13e12c5d1SDavid du Colombier #include <u.h>
23e12c5d1SDavid du Colombier #include <libc.h>
33e12c5d1SDavid du Colombier
43e12c5d1SDavid du Colombier
53e12c5d1SDavid du Colombier typedef struct Objtype {
63e12c5d1SDavid du Colombier char *name;
73e12c5d1SDavid du Colombier char *cc;
83e12c5d1SDavid du Colombier char *ld;
93e12c5d1SDavid du Colombier char *o;
103e12c5d1SDavid du Colombier char *oname;
113e12c5d1SDavid du Colombier } Objtype;
123e12c5d1SDavid du Colombier
13fee9fd16SDavid du Colombier /* sync with /sys/src/ape/cmd/cc.c */
143e12c5d1SDavid du Colombier Objtype objtype[] = {
159acf0835SDavid du Colombier {"spim", "0c", "0l", "0", "0.out"},
16f1a26406SDavid du Colombier {"mips64", "4c", "4l", "4", "4.out"},
1743a33802SDavid du Colombier {"arm", "5c", "5l", "5", "5.out"},
1843a33802SDavid du Colombier {"amd64", "6c", "6l", "6", "6.out"},
1943a33802SDavid du Colombier {"386", "8c", "8l", "8", "8.out"},
20c1b29f07SDavid du Colombier {"power64", "9c", "9l", "9", "9.out"},
2143a33802SDavid du Colombier {"sparc", "kc", "kl", "k", "k.out"},
227dd7cddfSDavid du Colombier {"power", "qc", "ql", "q", "q.out"},
239acf0835SDavid du Colombier {"mips", "vc", "vl", "v", "v.out"},
24*ce95e1b3SDavid du Colombier {"riscv", "ic", "il", "i", "i.out"},
25*ce95e1b3SDavid du Colombier {"riscv64", "jc", "jl", "j", "j.out"},
26f1a26406SDavid du Colombier {"spim64", "xc", "xl", "x", "x.out"},
273e12c5d1SDavid du Colombier };
28*ce95e1b3SDavid du Colombier char *allos = "05689kqvij";
293e12c5d1SDavid du Colombier
303e12c5d1SDavid du Colombier enum {
313e12c5d1SDavid du Colombier Nobjs = (sizeof objtype)/(sizeof objtype[0]),
32fee9fd16SDavid du Colombier Maxlist = 2000,
333e12c5d1SDavid du Colombier };
343e12c5d1SDavid du Colombier
353e12c5d1SDavid du Colombier typedef struct List {
363e12c5d1SDavid du Colombier char *strings[Maxlist];
373e12c5d1SDavid du Colombier int n;
383e12c5d1SDavid du Colombier } List;
393e12c5d1SDavid du Colombier
403e12c5d1SDavid du Colombier List srcs, objs, cpp, cc, ld, ldargs;
413e12c5d1SDavid du Colombier int cflag, vflag, Eflag, Pflag;
423e12c5d1SDavid du Colombier
433e12c5d1SDavid du Colombier void append(List *, char *);
443e12c5d1SDavid du Colombier char *changeext(char *, char *);
453e12c5d1SDavid du Colombier void doexec(char *, List *);
463e12c5d1SDavid du Colombier void dopipe(char *, List *, char *, List *);
473e12c5d1SDavid du Colombier void fatal(char *);
483e12c5d1SDavid du Colombier Objtype *findoty(void);
493e12c5d1SDavid du Colombier void printlist(List *);
503e12c5d1SDavid du Colombier
513e12c5d1SDavid du Colombier void
main(int argc,char * argv[])523e12c5d1SDavid du Colombier main(int argc, char *argv[])
533e12c5d1SDavid du Colombier {
543e12c5d1SDavid du Colombier Objtype *ot;
553e12c5d1SDavid du Colombier char *s, *suf, *ccpath;
563e12c5d1SDavid du Colombier char *oname;
5759cc4ca5SDavid du Colombier int i, cppn, ccn, oflag;
583e12c5d1SDavid du Colombier
5959cc4ca5SDavid du Colombier oflag = 0;
603e12c5d1SDavid du Colombier ot = findoty();
613e12c5d1SDavid du Colombier oname = ot->oname;
623e12c5d1SDavid du Colombier append(&cpp, "cpp");
633e12c5d1SDavid du Colombier append(&cpp, "-D__STDC__=1"); /* ANSI says so */
643e12c5d1SDavid du Colombier append(&cpp, "-N"); /* turn off standard includes */
653e12c5d1SDavid du Colombier append(&cc, ot->cc);
663e12c5d1SDavid du Colombier append(&ld, ot->ld);
673e12c5d1SDavid du Colombier while(argc > 0) {
683e12c5d1SDavid du Colombier ARGBEGIN {
6959cc4ca5SDavid du Colombier case '+':
709a747e4fSDavid du Colombier append(&cpp, smprint("-%c", ARGC()));
7159cc4ca5SDavid du Colombier break;
723e12c5d1SDavid du Colombier case 'c':
733e12c5d1SDavid du Colombier cflag = 1;
743e12c5d1SDavid du Colombier break;
753e12c5d1SDavid du Colombier case 'l':
769a747e4fSDavid du Colombier append(&objs, smprint("/%s/lib/ape/lib%s.a", ot->name, ARGF()));
773e12c5d1SDavid du Colombier break;
783e12c5d1SDavid du Colombier case 'o':
7959cc4ca5SDavid du Colombier oflag = 1;
803e12c5d1SDavid du Colombier oname = ARGF();
813e12c5d1SDavid du Colombier if(!oname)
823e12c5d1SDavid du Colombier fatal("no -o argument");
833e12c5d1SDavid du Colombier break;
843e12c5d1SDavid du Colombier case 'w':
853e12c5d1SDavid du Colombier case 'B':
8643a33802SDavid du Colombier case 'F':
873e12c5d1SDavid du Colombier case 'N':
883e12c5d1SDavid du Colombier case 'S':
8943a33802SDavid du Colombier case 'T':
907dd7cddfSDavid du Colombier case 'V':
9162579e34SDavid du Colombier case 'W':
929a747e4fSDavid du Colombier append(&cc, smprint("-%c", ARGC()));
933e12c5d1SDavid du Colombier break;
947dd7cddfSDavid du Colombier case 's':
959a747e4fSDavid du Colombier append(&cc, smprint("-s%s", ARGF()));
967dd7cddfSDavid du Colombier break;
973e12c5d1SDavid du Colombier case 'D':
983e12c5d1SDavid du Colombier case 'I':
993e12c5d1SDavid du Colombier case 'U':
1009a747e4fSDavid du Colombier append(&cpp, smprint("-%c%s", ARGC(), ARGF()));
1013e12c5d1SDavid du Colombier break;
1023e12c5d1SDavid du Colombier case 'v':
1033e12c5d1SDavid du Colombier vflag = 1;
1043e12c5d1SDavid du Colombier append(&ldargs, "-v");
1053e12c5d1SDavid du Colombier break;
1063e12c5d1SDavid du Colombier case 'P':
1073e12c5d1SDavid du Colombier Pflag = 1;
1083e12c5d1SDavid du Colombier cflag = 1;
1093e12c5d1SDavid du Colombier break;
1103e12c5d1SDavid du Colombier case 'E':
1113e12c5d1SDavid du Colombier Eflag = 1;
1123e12c5d1SDavid du Colombier cflag = 1;
1133e12c5d1SDavid du Colombier break;
1143e12c5d1SDavid du Colombier case 'p':
1153e12c5d1SDavid du Colombier append(&ldargs, "-p");
1163e12c5d1SDavid du Colombier break;
1179b7bf7dfSDavid du Colombier case 'f':
1189b7bf7dfSDavid du Colombier if(strcmp(ot->name, "arm") == 0)
1199b7bf7dfSDavid du Colombier append(&ldargs, "-f");
1209b7bf7dfSDavid du Colombier break;
12162579e34SDavid du Colombier case 'x':
12262579e34SDavid du Colombier s = ARGF();
12362579e34SDavid du Colombier if(s == nil || *s == '-')
12462579e34SDavid du Colombier fatal("no -x argument");
12562579e34SDavid du Colombier append(&ldargs, smprint("-x %s", s));
12662579e34SDavid du Colombier break;
127219b2ee8SDavid du Colombier case 'a':
128219b2ee8SDavid du Colombier /* hacky look inside ARGBEGIN insides, to see if we have -aa */
129219b2ee8SDavid du Colombier if(*_args == 'a') {
130219b2ee8SDavid du Colombier append(&cc, "-aa");
131219b2ee8SDavid du Colombier _args++;
132219b2ee8SDavid du Colombier } else
133219b2ee8SDavid du Colombier append(&cc, "-a");
134219b2ee8SDavid du Colombier cflag = 1;
135219b2ee8SDavid du Colombier break;
1363e12c5d1SDavid du Colombier default:
1373e12c5d1SDavid du Colombier fprint(2, "pcc: flag -%c ignored\n", ARGC());
1383e12c5d1SDavid du Colombier break;
1393e12c5d1SDavid du Colombier } ARGEND
1403e12c5d1SDavid du Colombier if(argc > 0) {
1413e12c5d1SDavid du Colombier s = argv[0];
1423e12c5d1SDavid du Colombier suf = utfrrune(s, '.');
1433e12c5d1SDavid du Colombier if(suf) {
1443e12c5d1SDavid du Colombier suf++;
1453e12c5d1SDavid du Colombier if(strcmp(suf, "c") == 0) {
1463e12c5d1SDavid du Colombier append(&srcs, s);
1473e12c5d1SDavid du Colombier append(&objs, changeext(s, ot->o));
1483e12c5d1SDavid du Colombier } else if(strcmp(suf, ot->o) == 0 ||
1493e12c5d1SDavid du Colombier strcmp(suf, "a") == 0 ||
1503e12c5d1SDavid du Colombier (suf[0] == 'a' && strcmp(suf+1, ot->o) == 0)) {
1513e12c5d1SDavid du Colombier append(&objs, s);
1523e12c5d1SDavid du Colombier } else if(utfrune(allos, suf[0]) != 0) {
1533e12c5d1SDavid du Colombier fprint(2, "pcc: argument %s ignored: wrong architecture\n",
1543e12c5d1SDavid du Colombier s);
1553e12c5d1SDavid du Colombier }
1563e12c5d1SDavid du Colombier }
1573e12c5d1SDavid du Colombier }
1583e12c5d1SDavid du Colombier }
1593e12c5d1SDavid du Colombier if(objs.n == 0)
1603e12c5d1SDavid du Colombier fatal("no files to compile or load");
1619a747e4fSDavid du Colombier ccpath = smprint("/bin/%s", ot->cc);
1629a747e4fSDavid du Colombier append(&cpp, smprint("-I/%s/include/ape", ot->name));
1633e12c5d1SDavid du Colombier append(&cpp, "-I/sys/include/ape");
1643e12c5d1SDavid du Colombier cppn = cpp.n;
1653e12c5d1SDavid du Colombier ccn = cc.n;
1663e12c5d1SDavid du Colombier for(i = 0; i < srcs.n; i++) {
1673e12c5d1SDavid du Colombier append(&cpp, srcs.strings[i]);
1683e12c5d1SDavid du Colombier if(Pflag)
1690d35113bSDavid du Colombier append(&cpp, changeext(objs.strings[i], "i"));
1703e12c5d1SDavid du Colombier if(Eflag || Pflag)
1713e12c5d1SDavid du Colombier doexec("/bin/cpp", &cpp);
1723e12c5d1SDavid du Colombier else {
1733e12c5d1SDavid du Colombier append(&cc, "-o");
17459cc4ca5SDavid du Colombier if(oflag && cflag)
17559cc4ca5SDavid du Colombier append(&cc, oname);
17659cc4ca5SDavid du Colombier else
17759cc4ca5SDavid du Colombier append(&cc, changeext(srcs.strings[i], ot->o));
1783e12c5d1SDavid du Colombier dopipe("/bin/cpp", &cpp, ccpath, &cc);
1793e12c5d1SDavid du Colombier }
1803e12c5d1SDavid du Colombier cpp.n = cppn;
1813e12c5d1SDavid du Colombier cc.n = ccn;
1823e12c5d1SDavid du Colombier }
1833e12c5d1SDavid du Colombier if(!cflag) {
1843e12c5d1SDavid du Colombier append(&ld, "-o");
1853e12c5d1SDavid du Colombier append(&ld, oname);
1863e12c5d1SDavid du Colombier for(i = 0; i < ldargs.n; i++)
1873e12c5d1SDavid du Colombier append(&ld, ldargs.strings[i]);
1883e12c5d1SDavid du Colombier for(i = 0; i < objs.n; i++)
1893e12c5d1SDavid du Colombier append(&ld, objs.strings[i]);
190fb7f0c93SDavid du Colombier append(&ld, smprint("/%s/lib/ape/libap.a", ot->name));
1919a747e4fSDavid du Colombier doexec(smprint("/bin/%s", ot->ld), &ld);
1927dd7cddfSDavid du Colombier if(objs.n == 1){
1937dd7cddfSDavid du Colombier /* prevent removal of a library */
1947dd7cddfSDavid du Colombier if(strstr(objs.strings[0], ".a") == 0)
1953e12c5d1SDavid du Colombier remove(objs.strings[0]);
1963e12c5d1SDavid du Colombier }
1977dd7cddfSDavid du Colombier }
1983e12c5d1SDavid du Colombier
1993e12c5d1SDavid du Colombier exits(0);
2003e12c5d1SDavid du Colombier }
2013e12c5d1SDavid du Colombier
2023e12c5d1SDavid du Colombier void
append(List * l,char * s)2033e12c5d1SDavid du Colombier append(List *l, char *s)
2043e12c5d1SDavid du Colombier {
2053e12c5d1SDavid du Colombier if(l->n >= Maxlist-1)
2063e12c5d1SDavid du Colombier fatal("too many arguments");
2073e12c5d1SDavid du Colombier l->strings[l->n++] = s;
2083e12c5d1SDavid du Colombier l->strings[l->n] = 0;
2093e12c5d1SDavid du Colombier }
2103e12c5d1SDavid du Colombier
2113e12c5d1SDavid du Colombier void
doexec(char * c,List * a)2123e12c5d1SDavid du Colombier doexec(char *c, List *a)
2133e12c5d1SDavid du Colombier {
2149a747e4fSDavid du Colombier Waitmsg *w;
2153e12c5d1SDavid du Colombier
2163e12c5d1SDavid du Colombier if(vflag) {
2173e12c5d1SDavid du Colombier printlist(a);
2183e12c5d1SDavid du Colombier fprint(2, "\n");
2193e12c5d1SDavid du Colombier }
2203e12c5d1SDavid du Colombier switch(fork()) {
2213e12c5d1SDavid du Colombier case -1:
2223e12c5d1SDavid du Colombier fatal("fork failed");
2233e12c5d1SDavid du Colombier case 0:
2243e12c5d1SDavid du Colombier exec(c, a->strings);
2253e12c5d1SDavid du Colombier fatal("exec failed");
2263e12c5d1SDavid du Colombier }
2279a747e4fSDavid du Colombier w = wait();
2289a747e4fSDavid du Colombier if(w == nil)
2293e12c5d1SDavid du Colombier fatal("wait failed");
2309a747e4fSDavid du Colombier if(w->msg[0])
2319a747e4fSDavid du Colombier fatal(smprint("%s: %s", a->strings[0], w->msg));
2329a747e4fSDavid du Colombier free(w);
2333e12c5d1SDavid du Colombier }
2343e12c5d1SDavid du Colombier
2353e12c5d1SDavid du Colombier void
dopipe(char * c1,List * a1,char * c2,List * a2)2363e12c5d1SDavid du Colombier dopipe(char *c1, List *a1, char *c2, List *a2)
2373e12c5d1SDavid du Colombier {
2389a747e4fSDavid du Colombier Waitmsg *w;
2399a747e4fSDavid du Colombier int pid1, got;
2403e12c5d1SDavid du Colombier int fd[2];
2413e12c5d1SDavid du Colombier
2423e12c5d1SDavid du Colombier if(vflag) {
2433e12c5d1SDavid du Colombier printlist(a1);
2443e12c5d1SDavid du Colombier fprint(2, " | ");
2453e12c5d1SDavid du Colombier printlist(a2);
2463e12c5d1SDavid du Colombier fprint(2, "\n");
2473e12c5d1SDavid du Colombier }
2483e12c5d1SDavid du Colombier if(pipe(fd) < 0)
2493e12c5d1SDavid du Colombier fatal("pipe failed");
2503e12c5d1SDavid du Colombier switch((pid1 = fork())) {
2513e12c5d1SDavid du Colombier case -1:
2523e12c5d1SDavid du Colombier fatal("fork failed");
2533e12c5d1SDavid du Colombier case 0:
2543e12c5d1SDavid du Colombier dup(fd[0], 0);
2553e12c5d1SDavid du Colombier close(fd[0]);
2563e12c5d1SDavid du Colombier close(fd[1]);
2573e12c5d1SDavid du Colombier exec(c2, a2->strings);
2583e12c5d1SDavid du Colombier fatal("exec failed");
2593e12c5d1SDavid du Colombier }
2603e12c5d1SDavid du Colombier switch(fork()) {
2613e12c5d1SDavid du Colombier case -1:
2623e12c5d1SDavid du Colombier fatal("fork failed");
2633e12c5d1SDavid du Colombier case 0:
2643e12c5d1SDavid du Colombier close(0);
2653e12c5d1SDavid du Colombier dup(fd[1], 1);
2663e12c5d1SDavid du Colombier close(fd[0]);
2673e12c5d1SDavid du Colombier close(fd[1]);
2683e12c5d1SDavid du Colombier exec(c1, a1->strings);
2693e12c5d1SDavid du Colombier fatal("exec failed");
2703e12c5d1SDavid du Colombier }
2713e12c5d1SDavid du Colombier close(fd[0]);
2723e12c5d1SDavid du Colombier close(fd[1]);
2733e12c5d1SDavid du Colombier for(got = 0; got < 2; got++) {
2749a747e4fSDavid du Colombier w = wait();
2759a747e4fSDavid du Colombier if(w == nil)
2763e12c5d1SDavid du Colombier fatal("wait failed");
2779a747e4fSDavid du Colombier if(w->msg[0])
2789a747e4fSDavid du Colombier fatal(smprint("%s: %s",
2799a747e4fSDavid du Colombier (w->pid == pid1) ? a1->strings[0] : a2->strings[0], w->msg));
2809a747e4fSDavid du Colombier free(w);
2813e12c5d1SDavid du Colombier }
2823e12c5d1SDavid du Colombier }
2833e12c5d1SDavid du Colombier
2843e12c5d1SDavid du Colombier Objtype *
findoty(void)2853e12c5d1SDavid du Colombier findoty(void)
2863e12c5d1SDavid du Colombier {
2873e12c5d1SDavid du Colombier char *o;
2883e12c5d1SDavid du Colombier Objtype *oty;
2893e12c5d1SDavid du Colombier
2903e12c5d1SDavid du Colombier o = getenv("objtype");
2913e12c5d1SDavid du Colombier if(!o)
2923e12c5d1SDavid du Colombier fatal("no $objtype in environment");
2933e12c5d1SDavid du Colombier for(oty = objtype; oty < &objtype[Nobjs]; oty++)
2943e12c5d1SDavid du Colombier if(strcmp(o, oty->name) == 0)
2953e12c5d1SDavid du Colombier return oty;
2963e12c5d1SDavid du Colombier fatal("unknown $objtype");
2973e12c5d1SDavid du Colombier return 0; /* shut compiler up */
2983e12c5d1SDavid du Colombier }
2993e12c5d1SDavid du Colombier
3003e12c5d1SDavid du Colombier void
fatal(char * msg)3013e12c5d1SDavid du Colombier fatal(char *msg)
3023e12c5d1SDavid du Colombier {
3033e12c5d1SDavid du Colombier fprint(2, "pcc: %s\n", msg);
3043e12c5d1SDavid du Colombier exits(msg);
3053e12c5d1SDavid du Colombier }
3063e12c5d1SDavid du Colombier
3073e12c5d1SDavid du Colombier /* src ends in .something; return copy of basename with .ext added */
3083e12c5d1SDavid du Colombier char *
changeext(char * src,char * ext)3093e12c5d1SDavid du Colombier changeext(char *src, char *ext)
3103e12c5d1SDavid du Colombier {
3113e12c5d1SDavid du Colombier char *b, *e, *ans;
3123e12c5d1SDavid du Colombier
3133e12c5d1SDavid du Colombier b = utfrrune(src, '/');
3143e12c5d1SDavid du Colombier if(b)
3153e12c5d1SDavid du Colombier b++;
3163e12c5d1SDavid du Colombier else
3173e12c5d1SDavid du Colombier b = src;
3183e12c5d1SDavid du Colombier e = utfrrune(src, '.');
3193e12c5d1SDavid du Colombier if(!e)
3203e12c5d1SDavid du Colombier return 0;
3213e12c5d1SDavid du Colombier *e = 0;
3229a747e4fSDavid du Colombier ans = smprint("%s.%s", b, ext);
3233e12c5d1SDavid du Colombier *e = '.';
3243e12c5d1SDavid du Colombier return ans;
3253e12c5d1SDavid du Colombier }
3263e12c5d1SDavid du Colombier
3273e12c5d1SDavid du Colombier void
printlist(List * l)3283e12c5d1SDavid du Colombier printlist(List *l)
3293e12c5d1SDavid du Colombier {
3303e12c5d1SDavid du Colombier int i;
3313e12c5d1SDavid du Colombier
3323e12c5d1SDavid du Colombier for(i = 0; i < l->n; i++) {
3333e12c5d1SDavid du Colombier fprint(2, "%s", l->strings[i]);
3343e12c5d1SDavid du Colombier if(i < l->n - 1)
3353e12c5d1SDavid du Colombier fprint(2, " ");
3363e12c5d1SDavid du Colombier }
3373e12c5d1SDavid du Colombier }
338