116960Smckusick /* Copyright (c) 1984 Regents of the University of California */ 216960Smckusick 316960Smckusick #ifndef lint 4*16979Smckusick static char sccsid[] = "@(#)machdep.c 1.3 (Berkeley) 08/20/84"; 516960Smckusick #endif not lint 616960Smckusick 716960Smckusick #include <stdio.h> 816960Smckusick #include <ctype.h> 9*16979Smckusick #include "inline.h" 1016960Smckusick 1116960Smckusick /* 12*16979Smckusick * The routines and tables in this file must be rewritten 13*16979Smckusick * for each new machine that this program is ported to. 1416960Smckusick */ 1516960Smckusick 1616960Smckusick /* 17*16979Smckusick * Instruction stop table. 18*16979Smckusick * All instructions that implicitly modify any of the temporary 19*16979Smckusick * registers, change control flow, or implicitly loop must be 20*16979Smckusick * listed in this table. It is used to find the end of a basic 21*16979Smckusick * block when scanning backwards through the instruction stream 22*16979Smckusick * trying to merge the inline expansion. 23*16979Smckusick */ 24*16979Smckusick struct inststoptbl inststoptable[] = { 25*16979Smckusick { "jbc" }, { "jlbc" }, { "jbs" }, { "jlbs" }, { "jbcc" }, 26*16979Smckusick { "jbsc" }, { "jbcs" }, { "jbss" }, { "jbr" }, { "jcc" }, 27*16979Smckusick { "jcs" }, { "jvc" }, { "jvs" }, { "jlss" }, { "jlssu" }, 28*16979Smckusick { "jleq" }, { "jlequ" }, { "jeql" }, { "jeqlu" }, { "jneq" }, 29*16979Smckusick { "jnequ" }, { "jgeq" }, { "jgequ" }, { "jgtr" }, { "jgtru" }, 30*16979Smckusick { "chmk" }, { "chme" }, { "chms" }, { "chmu" }, { "rei" }, 31*16979Smckusick { "ldpctx" }, { "svpctx" }, { "xfc" }, { "bpt" }, 32*16979Smckusick { "bugw" }, { "bugl" }, { "halt" }, { "pushr" }, { "popr" }, 33*16979Smckusick { "polyf" }, { "polyd" }, { "polyg" }, { "polyh" }, 34*16979Smckusick { "bneq" }, { "bnequ" }, { "beql" }, { "beqlu" }, { "bgtr" }, 35*16979Smckusick { "bleq" }, { "bgeq" }, { "blss" }, { "bgtru" }, { "blequ" }, 36*16979Smckusick { "bvc" }, { "bvs" }, { "bgequ" }, { "bcc" }, { "blssu" }, 37*16979Smckusick { "bcs" }, { "brb" }, { "brw" }, { "jmp" }, 38*16979Smckusick { "bbs" }, { "bbc" }, { "bbss" }, { "bbcs" }, { "bbsc" }, 39*16979Smckusick { "bbcc" }, { "bbssi" }, { "bbcci" }, { "blbs" }, { "blbc" }, 40*16979Smckusick { "acbb" }, { "acbw" }, { "acbl" }, { "acbf" }, { "acbd" }, 41*16979Smckusick { "acbg" }, { "acbh" }, { "aoblss" }, { "aobleq" }, 42*16979Smckusick { "sobgeq" }, { "sobgtr" }, { "caseb" }, { "casew" }, { "casel" }, 43*16979Smckusick { "bsbb" }, { "bsbw" }, { "jsb" }, { "rsb" }, 44*16979Smckusick { "callg" }, { "calls" }, { "ret" }, 45*16979Smckusick { "movc3" }, { "movc5" }, { "movtc" }, { "movtuc" }, 46*16979Smckusick { "cmpc3" }, { "cmpc5" }, { "scanc" }, { "spanc" }, 47*16979Smckusick { "locc" }, { "skpc" }, { "matchc" }, { "crc" }, 48*16979Smckusick { "movp" }, { "cmpp3" }, { "cmpp4" }, { "addp4" }, { "addp6" }, 49*16979Smckusick { "subp4" }, { "subp6" }, { "mulp" }, { "divp" }, { "cvtlp" }, 50*16979Smckusick { "cvtpl" }, { "cvtpt" }, { "cvttp" }, { "cvtps" }, { "cvtsp" }, 51*16979Smckusick { "ashp" }, { "editpc" }, 52*16979Smckusick { "escd" }, { "esce" }, { "escf" }, 53*16979Smckusick { "" } 54*16979Smckusick }; 55*16979Smckusick 56*16979Smckusick /* 5716960Smckusick * Check to see if a line is a candidate for replacement. 5816960Smckusick * Return pointer to name to be looked up in pattern table. 5916960Smckusick */ 6016960Smckusick char * 6116960Smckusick doreplaceon(cp) 6216960Smckusick char *cp; 6316960Smckusick { 6416960Smckusick 6516960Smckusick if (bcmp(cp, "calls\t$", 7) == 0) 6616960Smckusick return (cp + 7); 6716960Smckusick return (0); 6816960Smckusick } 6916960Smckusick 7016960Smckusick /* 7116960Smckusick * Find the next argument to the function being expanded. 7216960Smckusick * MACHINE DEPENDENT 7316960Smckusick */ 7416960Smckusick nextarg(argc, argv) 7516960Smckusick int argc; 7616960Smckusick char *argv[]; 7716960Smckusick { 7816960Smckusick register char *lastarg = argv[2]; 7916960Smckusick 8016960Smckusick if (argc == 3 && 8116960Smckusick bcmp(argv[0], "mov", 3) == 0 && 8216960Smckusick bcmp(argv[1], "(sp)+", 6) == 0 && 8316960Smckusick lastarg[0] == 'r' && isdigit(lastarg[1]) && lastarg[2] == '\0') 8416960Smckusick return (lastarg[1] - '0'); 8516960Smckusick return (-1); 8616960Smckusick } 8716960Smckusick 8816960Smckusick /* 8916960Smckusick * Determine whether the current line pushes an argument. 9016960Smckusick * MACHINE DEPENDENT 9116960Smckusick */ 9216960Smckusick ispusharg(argc, argv) 9316960Smckusick int argc; 9416960Smckusick char *argv[]; 9516960Smckusick { 9616960Smckusick 9716960Smckusick if (argc < 2) 9816960Smckusick return (0); 9916960Smckusick if (argc == 2 && bcmp(argv[0], "push", 4) == 0) 10016960Smckusick return (1); 10116960Smckusick if (bcmp(argv[argc - 1], "-(sp)", 6) == 0) 10216960Smckusick return (1); 10316960Smckusick return (0); 10416960Smckusick } 10516960Smckusick 10616960Smckusick /* 10716960Smckusick * Determine which (if any) registers are modified 10816960Smckusick * Return register number that is modified, -1 if none are modified. 10916960Smckusick * MACHINE DEPENDENT 11016960Smckusick */ 11116960Smckusick modifies(argc, argv) 11216960Smckusick int argc; 11316960Smckusick char *argv[]; 11416960Smckusick { 11516960Smckusick /* 11616960Smckusick * For the VAX all we care about are r0 to r5 11716960Smckusick */ 11816960Smckusick register char *lastarg = argv[argc - 1]; 11916960Smckusick 12016960Smckusick if (lastarg[0] == 'r' && isdigit(lastarg[1]) && lastarg[2] == '\0') 12116960Smckusick return (lastarg[1] - '0'); 12216960Smckusick return (-1); 12316960Smckusick } 12416960Smckusick 12516960Smckusick /* 12616960Smckusick * Rewrite the instruction in (argc, argv) to store its 12716960Smckusick * contents into arg instead of onto the stack. The new 12816960Smckusick * instruction is placed in the buffer that is provided. 12916960Smckusick * MACHINE DEPENDENT 13016960Smckusick */ 13116960Smckusick rewrite(instbuf, argc, argv, target) 13216960Smckusick char *instbuf; 13316960Smckusick int argc; 13416960Smckusick char *argv[]; 13516960Smckusick int target; 13616960Smckusick { 13716960Smckusick 13816960Smckusick switch (argc) { 13916960Smckusick case 0: 14016960Smckusick instbuf[0] = '\0'; 14116960Smckusick fprintf("blank line to rewrite?\n"); 14216960Smckusick return; 14316960Smckusick case 1: 14416960Smckusick sprintf(instbuf, "\t%s\n", argv[0]); 14516960Smckusick fprintf(stderr, "rewrite?-> %s", instbuf); 14616960Smckusick return; 14716960Smckusick case 2: 14816960Smckusick if (bcmp(argv[0], "push", 4) == 0) { 14916960Smckusick sprintf(instbuf, "\tmov%s\t%s,r%d\n", 15016960Smckusick &argv[0][4], argv[1], target); 15116960Smckusick return; 15216960Smckusick } 15316960Smckusick sprintf(instbuf, "\t%s\tr%d\n", argv[0], target); 15416960Smckusick return; 15516960Smckusick case 3: 15616960Smckusick sprintf(instbuf, "\t%s\t%s,r%d\n", argv[0], argv[1], target); 15716960Smckusick return; 15816960Smckusick case 4: 15916960Smckusick sprintf(instbuf, "\t%s\t%s,%s,r%d\n", 16016960Smckusick argv[0], argv[1], argv[2], target); 16116960Smckusick return; 16216971Smckusick case 5: 16316971Smckusick sprintf(instbuf, "\t%s\t%s,%s,%s,r%d\n", 16416971Smckusick argv[0], argv[1], argv[2], argv[3], target); 16516971Smckusick return; 16616960Smckusick default: 16716960Smckusick sprintf(instbuf, "\t%s\t%s", argv[0], argv[1]); 16816960Smckusick argc -= 2, argv += 2; 16916960Smckusick while (argc-- > 0) { 17016960Smckusick strcat(instbuf, ","); 17116960Smckusick strcat(instbuf, *argv++); 17216960Smckusick } 17316960Smckusick strcat(instbuf, "\n"); 17416960Smckusick fprintf(stderr, "rewrite?-> %s", instbuf); 17516960Smckusick return; 17616960Smckusick } 17716960Smckusick } 178