Annotation of 43BSD/sys/vax/inline/machdep.c, revision 1.1

1.1     ! root        1: /*
        !             2:  * Copyright (c) 1984, 1986 Regents of the University of California.
        !             3:  * All rights reserved.  The Berkeley software License Agreement
        !             4:  * specifies the terms and conditions for redistribution.
        !             5:  */
        !             6: 
        !             7: #ifndef lint
        !             8: static char sccsid[] = "@(#)machdep.c  7.1 (Berkeley) 6/5/86";
        !             9: #endif not lint
        !            10: 
        !            11: #include <stdio.h>
        !            12: #include <ctype.h>
        !            13: #include "inline.h"
        !            14: 
        !            15: extern char *strcpy();
        !            16: extern char *strcat();
        !            17: extern char *index();
        !            18: 
        !            19: /*
        !            20:  * The routines and tables in this file must be rewritten
        !            21:  * for each new machine that this program is ported to.
        !            22:  */
        !            23: 
        !            24: #ifdef vax
        !            25: /*
        !            26:  * Instruction stop table.
        !            27:  * All instructions that implicitly modify any of the temporary
        !            28:  * registers, change control flow, or implicitly loop must be
        !            29:  * listed in this table. It is used to find the end of a basic
        !            30:  * block when scanning backwards through the instruction stream
        !            31:  * trying to merge the inline expansion.
        !            32:  */
        !            33: struct inststoptbl inststoptable[] = {
        !            34:        { "jbc" }, { "jlbc" }, { "jbs" }, { "jlbs" }, { "jbcc" },
        !            35:        { "jbsc" }, { "jbcs" }, { "jbss" }, { "jbr" }, { "jcc" },
        !            36:        { "jcs" }, { "jvc" }, { "jvs" }, { "jlss" }, { "jlssu" },
        !            37:        { "jleq" }, { "jlequ" }, { "jeql" }, { "jeqlu" }, { "jneq" },
        !            38:        { "jnequ" }, { "jgeq" }, { "jgequ" }, { "jgtr" }, { "jgtru" },
        !            39:        { "chmk" }, { "chme" }, { "chms" }, { "chmu" }, { "rei" },
        !            40:        { "ldpctx" }, { "svpctx" }, { "xfc" }, { "bpt" },
        !            41:        { "bugw" }, { "bugl" }, { "halt" }, { "pushr" }, { "popr" },
        !            42:        { "polyf" }, { "polyd" }, { "polyg" }, { "polyh" },
        !            43:        { "bneq" }, { "bnequ" }, { "beql" }, { "beqlu" }, { "bgtr" },
        !            44:        { "bleq" }, { "bgeq" }, { "blss" }, { "bgtru" }, { "blequ" },
        !            45:        { "bvc" }, { "bvs" }, { "bgequ" }, { "bcc" }, { "blssu" },
        !            46:        { "bcs" }, { "brb" }, { "brw" }, { "jmp" },
        !            47:        { "bbs" }, { "bbc" }, { "bbss" }, { "bbcs" }, { "bbsc" },
        !            48:        { "bbcc" }, { "bbssi" }, { "bbcci" }, { "blbs" }, { "blbc" },
        !            49:        { "acbb" }, { "acbw" }, { "acbl" }, { "acbf" }, { "acbd" },
        !            50:        { "acbg" }, { "acbh" }, { "aoblss" }, { "aobleq" },
        !            51:        { "sobgeq" }, { "sobgtr" }, { "caseb" }, { "casew" }, { "casel" },
        !            52:        { "bsbb" }, { "bsbw" }, { "jsb" }, { "rsb" },
        !            53:        { "callg" }, { "calls" }, { "ret" },
        !            54:        { "movc3" }, { "movc5" }, { "movtc" }, { "movtuc" },
        !            55:        { "cmpc3" }, { "cmpc5" }, { "scanc" }, { "spanc" },
        !            56:        { "locc" }, { "skpc" }, { "matchc" }, { "crc" },
        !            57:        { "movp" }, { "cmpp3" }, { "cmpp4" }, { "addp4" }, { "addp6" },
        !            58:        { "subp4" }, { "subp6" }, { "mulp" }, { "divp" }, { "cvtlp" },
        !            59:        { "cvtpl" }, { "cvtpt" }, { "cvttp" }, { "cvtps" }, { "cvtsp" },
        !            60:        { "ashp" }, { "editpc" },
        !            61:        { "escd" }, { "esce" }, { "escf" },
        !            62:        { "" }
        !            63: };
        !            64: 
        !            65: /*
        !            66:  * Check to see if a line is a candidate for replacement.
        !            67:  * Return pointer to name to be looked up in pattern table.
        !            68:  */
        !            69: char *
        !            70: doreplaceon(cp)
        !            71:        char *cp;
        !            72: {
        !            73: 
        !            74:        if (bcmp(cp, "calls\t", 6) != 0)
        !            75:                return (0);
        !            76:        if ((cp = index(cp + 6, ',')) == 0)
        !            77:                return (0);
        !            78:        return (++cp);
        !            79: }
        !            80: 
        !            81: /*
        !            82:  * Find out how many arguments the function is being called with.
        !            83:  * A return value of -1 indicates that the count can't be determined.
        !            84:  */
        !            85: int
        !            86: countargs(cp)
        !            87:        char *cp;
        !            88: {
        !            89: 
        !            90:        if ((cp = index(cp, '$')) == 0)
        !            91:                return (-1);
        !            92:        if (!isdigit(*++cp))
        !            93:                return (-1);
        !            94:        return (atoi(cp));
        !            95: }
        !            96: 
        !            97: /*
        !            98:  * Find the next argument to the function being expanded.
        !            99:  */
        !           100: nextarg(argc, argv)
        !           101:        int argc;
        !           102:        char *argv[];
        !           103: {
        !           104:        register char *lastarg = argv[2];
        !           105: 
        !           106:        if (argc == 3 &&
        !           107:            bcmp(argv[0], "mov", 3) == 0 &&
        !           108:            bcmp(argv[1], "(sp)+", 6) == 0 &&
        !           109:            lastarg[0] == 'r' && isdigit(lastarg[1]) && lastarg[2] == '\0')
        !           110:                return (lastarg[1] - '0');
        !           111:        return (-1);
        !           112: }
        !           113: 
        !           114: /*
        !           115:  * Determine whether the current line pushes an argument.
        !           116:  */
        !           117: ispusharg(argc, argv)
        !           118:        int argc;
        !           119:        char *argv[];
        !           120: {
        !           121: 
        !           122:        if (argc < 2)
        !           123:                return (0);
        !           124:        if (argc == 2 && bcmp(argv[0], "push", 4) == 0)
        !           125:                return (1);
        !           126:        if (bcmp(argv[argc - 1], "-(sp)", 6) == 0)
        !           127:                return (1);
        !           128:        return (0);
        !           129: }
        !           130: 
        !           131: /*
        !           132:  * Determine which (if any) registers are modified
        !           133:  * Return register number that is modified, -1 if none are modified.
        !           134:  */
        !           135: modifies(argc, argv)
        !           136:        int argc;
        !           137:        char *argv[];
        !           138: {
        !           139:        /*
        !           140:         * For the VAX all we care about are r0 to r5
        !           141:         */
        !           142:        register char *lastarg = argv[argc - 1];
        !           143: 
        !           144:        if (lastarg[0] == 'r' && isdigit(lastarg[1]) && lastarg[2] == '\0')
        !           145:                return (lastarg[1] - '0');
        !           146:        return (-1);
        !           147: }
        !           148: 
        !           149: /*
        !           150:  * Rewrite the instruction in (argc, argv) to store its
        !           151:  * contents into arg instead of onto the stack. The new
        !           152:  * instruction is placed in the buffer that is provided.
        !           153:  */
        !           154: rewrite(instbuf, argc, argv, target)
        !           155:        char *instbuf;
        !           156:        int argc;
        !           157:        char *argv[];
        !           158:        int target;
        !           159: {
        !           160: 
        !           161:        switch (argc) {
        !           162:        case 0:
        !           163:                instbuf[0] = '\0';
        !           164:                fprintf(stderr, "blank line to rewrite?\n");
        !           165:                return;
        !           166:        case 1:
        !           167:                sprintf(instbuf, "\t%s\n", argv[0]);
        !           168:                fprintf(stderr, "rewrite?-> %s", instbuf);
        !           169:                return;
        !           170:        case 2:
        !           171:                if (bcmp(argv[0], "push", 4) == 0) {
        !           172:                        sprintf(instbuf, "\tmov%s\t%s,r%d\n",
        !           173:                                &argv[0][4], argv[1], target);
        !           174:                        return;
        !           175:                }
        !           176:                sprintf(instbuf, "\t%s\tr%d\n", argv[0], target);
        !           177:                return;
        !           178:        case 3:
        !           179:                sprintf(instbuf, "\t%s\t%s,r%d\n", argv[0], argv[1], target);
        !           180:                return;
        !           181:        case 4:
        !           182:                sprintf(instbuf, "\t%s\t%s,%s,r%d\n",
        !           183:                        argv[0], argv[1], argv[2], target);
        !           184:                return;
        !           185:        case 5:
        !           186:                sprintf(instbuf, "\t%s\t%s,%s,%s,r%d\n",
        !           187:                        argv[0], argv[1], argv[2], argv[3], target);
        !           188:                return;
        !           189:        default:
        !           190:                sprintf(instbuf, "\t%s\t%s", argv[0], argv[1]);
        !           191:                argc -= 2, argv += 2;
        !           192:                while (argc-- > 0) {
        !           193:                        (void) strcat(instbuf, ",");
        !           194:                        (void) strcat(instbuf, *argv++);
        !           195:                }
        !           196:                (void) strcat(instbuf, "\n");
        !           197:                fprintf(stderr, "rewrite?-> %s", instbuf);
        !           198:                return;
        !           199:        }
        !           200: }
        !           201: 
        !           202: /*
        !           203:  * Do any necessary post expansion cleanup.
        !           204:  */
        !           205: /*ARGSUSED*/
        !           206: cleanup(numargs)
        !           207:        int numargs;
        !           208: {
        !           209: 
        !           210:        return;
        !           211: }
        !           212: #endif vax
        !           213: 
        !           214: #ifdef mc68000
        !           215: /*
        !           216:  * Instruction stop table.
        !           217:  * All instructions that implicitly modify any of the temporary
        !           218:  * registers, change control flow, or implicitly loop must be
        !           219:  * listed in this table. It is used to find the end of a basic
        !           220:  * block when scanning backwards through the instruction stream
        !           221:  * trying to merge the inline expansion.
        !           222:  */
        !           223: struct inststoptbl inststoptable[] = {
        !           224:        { "" }
        !           225: };
        !           226: 
        !           227: /*
        !           228:  * Check to see if a line is a candidate for replacement.
        !           229:  * Return pointer to name to be looked up in pattern table.
        !           230:  */
        !           231: char *
        !           232: doreplaceon(cp)
        !           233:        char *cp;
        !           234: {
        !           235: 
        !           236:        if (bcmp(cp, "jbsr\t", 5) == 0)
        !           237:                return (cp + 5);
        !           238:        return (0);
        !           239: }
        !           240: 
        !           241: /*
        !           242:  * Find out how many arguments the function is being called with.
        !           243:  * A return value of -1 indicates that the count can't be determined.
        !           244:  */
        !           245: /* ARGSUSED */
        !           246: int
        !           247: countargs(cp)
        !           248:        char *cp;
        !           249: {
        !           250: 
        !           251:        /*
        !           252:         * TODO
        !           253:         * Figure out what the count should be. 
        !           254:         * Probably have to read the next instruction here
        !           255:         * instead of in cleanup() below.
        !           256:         */
        !           257:        return (-1);
        !           258: }
        !           259: 
        !           260: /*
        !           261:  * Find the next argument to the function being expanded.
        !           262:  */
        !           263: nextarg(argc, argv)
        !           264:        int argc;
        !           265:        char *argv[];
        !           266: {
        !           267:        register char *lastarg = argv[2];
        !           268: 
        !           269:        if (argc == 3 &&
        !           270:            bcmp(argv[0], "movl", 5) == 0 &&
        !           271:            bcmp(argv[1], "sp@+", 5) == 0 &&
        !           272:            (lastarg[1] == '0' || lastarg[1] == '1') &&
        !           273:            lastarg[2] == '\0') {
        !           274:                if (lastarg[0] == 'd')
        !           275:                        return (lastarg[1] - '0');
        !           276:                return (lastarg[1] - '0' + 8);
        !           277:        }
        !           278:        return (-1);
        !           279: }
        !           280: 
        !           281: /*
        !           282:  * Determine whether the current line pushes an argument.
        !           283:  */
        !           284: ispusharg(argc, argv)
        !           285:        int argc;
        !           286:        char *argv[];
        !           287: {
        !           288: 
        !           289:        if (argc < 2)
        !           290:                return (0);
        !           291:        if (argc == 2 && bcmp(argv[0], "pea", 4) == 0)
        !           292:                return (1);
        !           293:        if (bcmp(argv[argc - 1], "sp@-", 5) == 0)
        !           294:                return (1);
        !           295:        return (0);
        !           296: }
        !           297: 
        !           298: /*
        !           299:  * Determine which (if any) registers are modified
        !           300:  * Return register number that is modified, -1 if none are modified.
        !           301:  */
        !           302: modifies(argc, argv)
        !           303:        int argc;
        !           304:        char *argv[];
        !           305: {
        !           306:        /*
        !           307:         * For the MC68000 all we care about are d0, d1, a0, and a1.
        !           308:         */
        !           309:        register char *lastarg = argv[argc - 1];
        !           310: 
        !           311:        if (lastarg[0] == 'd' && isdigit(lastarg[1]) && lastarg[2] == '\0')
        !           312:                return (lastarg[1] - '0');
        !           313:        if (lastarg[0] == 'a' && isdigit(lastarg[1]) && lastarg[2] == '\0')
        !           314:                return (lastarg[1] - '0' + 8);
        !           315:        return (-1);
        !           316: }
        !           317: 
        !           318: /*
        !           319:  * Rewrite the instruction in (argc, argv) to store its
        !           320:  * contents into arg instead of onto the stack. The new
        !           321:  * instruction is placed in the buffer that is provided.
        !           322:  */
        !           323: rewrite(instbuf, argc, argv, target)
        !           324:        char *instbuf;
        !           325:        int argc;
        !           326:        char *argv[];
        !           327:        int target;
        !           328: {
        !           329:        int regno;
        !           330:        char regtype;
        !           331: 
        !           332:        if (target < 8) {
        !           333:                regtype = 'd';
        !           334:                regno = target;
        !           335:        } else {
        !           336:                regtype = 'a';
        !           337:                regno = target - 8;
        !           338:        }
        !           339:        switch (argc) {
        !           340:        case 0:
        !           341:                instbuf[0] = '\0';
        !           342:                fprintf(stderr, "blank line to rewrite?\n");
        !           343:                return;
        !           344:        case 1:
        !           345:                sprintf(instbuf, "\t%s\n", argv[0]);
        !           346:                fprintf(stderr, "rewrite?-> %s", instbuf);
        !           347:                return;
        !           348:        case 2:
        !           349:                if (bcmp(argv[0], "pea", 4) == 0) {
        !           350:                        if (regtype == 'a') {
        !           351:                                sprintf(instbuf, "\tlea\t%s,%c%d\n",
        !           352:                                        argv[1], regtype, regno);
        !           353:                                return;
        !           354:                        }
        !           355:                        if (argv[1][0] == '_' || isdigit(argv[1][0])) {
        !           356:                                sprintf(instbuf, "\tmovl\t#%s,%c%d\n",
        !           357:                                        argv[1], regtype, regno);
        !           358:                                return;
        !           359:                        }
        !           360:                        sprintf(instbuf,
        !           361:                                "\texg\ta0,d%d\n\tlea\t%s,a0\n\texg\ta0,d%d\n",
        !           362:                                regno, argv[1], regno);
        !           363:                        return;
        !           364:                }
        !           365:                sprintf(instbuf, "\t%s\t%c%d\n", argv[0], regtype, regno);
        !           366:                return;
        !           367:        case 3:
        !           368:                sprintf(instbuf, "\t%s\t%s,%c%d\n",
        !           369:                        argv[0], argv[1], regtype, regno);
        !           370:                return;
        !           371:        default:
        !           372:                sprintf(instbuf, "\t%s\t%s", argv[0], argv[1]);
        !           373:                argc -= 2, argv += 2;
        !           374:                while (argc-- > 0) {
        !           375:                        (void) strcat(instbuf, ",");
        !           376:                        (void) strcat(instbuf, *argv++);
        !           377:                }
        !           378:                (void) strcat(instbuf, "\n");
        !           379:                fprintf(stderr, "rewrite?-> %s", instbuf);
        !           380:                return;
        !           381:        }
        !           382: }
        !           383: 
        !           384: /*
        !           385:  * Do any necessary post expansion cleanup.
        !           386:  */
        !           387: cleanup(numargs)
        !           388:        int numargs;
        !           389: {
        !           390:        extern int lineno;
        !           391:        
        !           392:        if (numargs == 0)
        !           393:                return;
        !           394:        /*
        !           395:         * delete instruction to pop arguments.
        !           396:         * TODO:
        !           397:         *      CHECK FOR LABEL
        !           398:         *      CHECK THAT INSTRUCTION IS A POP
        !           399:         */
        !           400:        fgets(line[bufhead], MAXLINELEN, stdin);
        !           401:        lineno++;
        !           402: }
        !           403: #endif mc68000

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.