xref: /plan9/sys/src/cmd/pcc.c (revision fee9fd16fbcb514a4f462b56ff20c7199c0fc677)
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 
13*fee9fd16SDavid du Colombier /* sync with /sys/src/ape/cmd/cc.c */
143e12c5d1SDavid du Colombier Objtype objtype[] = {
159acf0835SDavid du Colombier 	{"spim",	"0c", "0l", "0", "0.out"},
1643a33802SDavid du Colombier 	{"arm",		"5c", "5l", "5", "5.out"},
1743a33802SDavid du Colombier 	{"amd64",	"6c", "6l", "6", "6.out"},
1843a33802SDavid du Colombier 	{"386",		"8c", "8l", "8", "8.out"},
19c1b29f07SDavid du Colombier 	{"power64",	"9c", "9l", "9", "9.out"},
2043a33802SDavid du Colombier 	{"sparc",	"kc", "kl", "k", "k.out"},
217dd7cddfSDavid du Colombier 	{"power",	"qc", "ql", "q", "q.out"},
229acf0835SDavid du Colombier 	{"mips",	"vc", "vl", "v", "v.out"},
233e12c5d1SDavid du Colombier };
24*fee9fd16SDavid du Colombier char	*allos = "05689kqv";
253e12c5d1SDavid du Colombier 
263e12c5d1SDavid du Colombier enum {
273e12c5d1SDavid du Colombier 	Nobjs = (sizeof objtype)/(sizeof objtype[0]),
28*fee9fd16SDavid du Colombier 	Maxlist = 2000,
293e12c5d1SDavid du Colombier };
303e12c5d1SDavid du Colombier 
313e12c5d1SDavid du Colombier typedef struct List {
323e12c5d1SDavid du Colombier 	char	*strings[Maxlist];
333e12c5d1SDavid du Colombier 	int	n;
343e12c5d1SDavid du Colombier } List;
353e12c5d1SDavid du Colombier 
363e12c5d1SDavid du Colombier List	srcs, objs, cpp, cc, ld, ldargs;
373e12c5d1SDavid du Colombier int	cflag, vflag, Eflag, Pflag;
383e12c5d1SDavid du Colombier 
393e12c5d1SDavid du Colombier void	append(List *, char *);
403e12c5d1SDavid du Colombier char	*changeext(char *, char *);
413e12c5d1SDavid du Colombier void	doexec(char *, List *);
423e12c5d1SDavid du Colombier void	dopipe(char *, List *, char *, List *);
433e12c5d1SDavid du Colombier void	fatal(char *);
443e12c5d1SDavid du Colombier Objtype	*findoty(void);
453e12c5d1SDavid du Colombier void	printlist(List *);
463e12c5d1SDavid du Colombier 
473e12c5d1SDavid du Colombier void
main(int argc,char * argv[])483e12c5d1SDavid du Colombier main(int argc, char *argv[])
493e12c5d1SDavid du Colombier {
503e12c5d1SDavid du Colombier 	Objtype *ot;
513e12c5d1SDavid du Colombier 	char *s, *suf, *ccpath;
523e12c5d1SDavid du Colombier 	char *oname;
5359cc4ca5SDavid du Colombier 	int i, cppn, ccn, oflag;
543e12c5d1SDavid du Colombier 
5559cc4ca5SDavid du Colombier 	oflag = 0;
563e12c5d1SDavid du Colombier 	ot = findoty();
573e12c5d1SDavid du Colombier 	oname = ot->oname;
583e12c5d1SDavid du Colombier 	append(&cpp, "cpp");
593e12c5d1SDavid du Colombier 	append(&cpp, "-D__STDC__=1");	/* ANSI says so */
603e12c5d1SDavid du Colombier 	append(&cpp, "-N");		/* turn off standard includes */
613e12c5d1SDavid du Colombier 	append(&cc, ot->cc);
623e12c5d1SDavid du Colombier 	append(&ld, ot->ld);
633e12c5d1SDavid du Colombier 	while(argc > 0) {
643e12c5d1SDavid du Colombier 		ARGBEGIN {
6559cc4ca5SDavid du Colombier 		case '+':
669a747e4fSDavid du Colombier 			append(&cpp, smprint("-%c", ARGC()));
6759cc4ca5SDavid du Colombier 			break;
683e12c5d1SDavid du Colombier 		case 'c':
693e12c5d1SDavid du Colombier 			cflag = 1;
703e12c5d1SDavid du Colombier 			break;
713e12c5d1SDavid du Colombier 		case 'l':
729a747e4fSDavid du Colombier 			append(&objs, smprint("/%s/lib/ape/lib%s.a", ot->name, ARGF()));
733e12c5d1SDavid du Colombier 			break;
743e12c5d1SDavid du Colombier 		case 'o':
7559cc4ca5SDavid du Colombier 			oflag = 1;
763e12c5d1SDavid du Colombier 			oname = ARGF();
773e12c5d1SDavid du Colombier 			if(!oname)
783e12c5d1SDavid du Colombier 				fatal("no -o argument");
793e12c5d1SDavid du Colombier 			break;
803e12c5d1SDavid du Colombier 		case 'w':
813e12c5d1SDavid du Colombier 		case 'B':
8243a33802SDavid du Colombier 		case 'F':
833e12c5d1SDavid du Colombier 		case 'N':
843e12c5d1SDavid du Colombier 		case 'S':
8543a33802SDavid du Colombier 		case 'T':
867dd7cddfSDavid du Colombier 		case 'V':
8762579e34SDavid du Colombier 		case 'W':
889a747e4fSDavid du Colombier 			append(&cc, smprint("-%c", ARGC()));
893e12c5d1SDavid du Colombier 			break;
907dd7cddfSDavid du Colombier 		case 's':
919a747e4fSDavid du Colombier 			append(&cc, smprint("-s%s", ARGF()));
927dd7cddfSDavid du Colombier 			break;
933e12c5d1SDavid du Colombier 		case 'D':
943e12c5d1SDavid du Colombier 		case 'I':
953e12c5d1SDavid du Colombier 		case 'U':
969a747e4fSDavid du Colombier 			append(&cpp, smprint("-%c%s", ARGC(), ARGF()));
973e12c5d1SDavid du Colombier 			break;
983e12c5d1SDavid du Colombier 		case 'v':
993e12c5d1SDavid du Colombier 			vflag = 1;
1003e12c5d1SDavid du Colombier 			append(&ldargs, "-v");
1013e12c5d1SDavid du Colombier 			break;
1023e12c5d1SDavid du Colombier 		case 'P':
1033e12c5d1SDavid du Colombier 			Pflag = 1;
1043e12c5d1SDavid du Colombier 			cflag = 1;
1053e12c5d1SDavid du Colombier 			break;
1063e12c5d1SDavid du Colombier 		case 'E':
1073e12c5d1SDavid du Colombier 			Eflag = 1;
1083e12c5d1SDavid du Colombier 			cflag = 1;
1093e12c5d1SDavid du Colombier 			break;
1103e12c5d1SDavid du Colombier 		case 'p':
1113e12c5d1SDavid du Colombier 			append(&ldargs, "-p");
1123e12c5d1SDavid du Colombier 			break;
1139b7bf7dfSDavid du Colombier 		case 'f':
1149b7bf7dfSDavid du Colombier 			if(strcmp(ot->name, "arm") == 0)
1159b7bf7dfSDavid du Colombier 				append(&ldargs, "-f");
1169b7bf7dfSDavid du Colombier 			break;
11762579e34SDavid du Colombier 		case 'x':
11862579e34SDavid du Colombier 			s = ARGF();
11962579e34SDavid du Colombier 			if(s == nil || *s == '-')
12062579e34SDavid du Colombier 				fatal("no -x argument");
12162579e34SDavid du Colombier 			append(&ldargs, smprint("-x %s", s));
12262579e34SDavid du Colombier 			break;
123219b2ee8SDavid du Colombier 		case 'a':
124219b2ee8SDavid du Colombier 			/* hacky look inside ARGBEGIN insides, to see if we have -aa */
125219b2ee8SDavid du Colombier 			if(*_args == 'a') {
126219b2ee8SDavid du Colombier 				append(&cc, "-aa");
127219b2ee8SDavid du Colombier 				_args++;
128219b2ee8SDavid du Colombier 			} else
129219b2ee8SDavid du Colombier 				append(&cc, "-a");
130219b2ee8SDavid du Colombier 			cflag = 1;
131219b2ee8SDavid du Colombier 			break;
1323e12c5d1SDavid du Colombier 		default:
1333e12c5d1SDavid du Colombier 			fprint(2, "pcc: flag -%c ignored\n", ARGC());
1343e12c5d1SDavid du Colombier 			break;
1353e12c5d1SDavid du Colombier 		} ARGEND
1363e12c5d1SDavid du Colombier 		if(argc > 0) {
1373e12c5d1SDavid du Colombier 			s = argv[0];
1383e12c5d1SDavid du Colombier 			suf = utfrrune(s, '.');
1393e12c5d1SDavid du Colombier 			if(suf) {
1403e12c5d1SDavid du Colombier 				suf++;
1413e12c5d1SDavid du Colombier 				if(strcmp(suf, "c") == 0) {
1423e12c5d1SDavid du Colombier 					append(&srcs, s);
1433e12c5d1SDavid du Colombier 					append(&objs, changeext(s, ot->o));
1443e12c5d1SDavid du Colombier 				} else if(strcmp(suf, ot->o) == 0 ||
1453e12c5d1SDavid du Colombier 					  strcmp(suf, "a") == 0 ||
1463e12c5d1SDavid du Colombier 					  (suf[0] == 'a' && strcmp(suf+1, ot->o) == 0)) {
1473e12c5d1SDavid du Colombier 					append(&objs, s);
1483e12c5d1SDavid du Colombier 				} else if(utfrune(allos, suf[0]) != 0) {
1493e12c5d1SDavid du Colombier 					fprint(2, "pcc: argument %s ignored: wrong architecture\n",
1503e12c5d1SDavid du Colombier 						s);
1513e12c5d1SDavid du Colombier 				}
1523e12c5d1SDavid du Colombier 			}
1533e12c5d1SDavid du Colombier 		}
1543e12c5d1SDavid du Colombier 	}
1553e12c5d1SDavid du Colombier 	if(objs.n == 0)
1563e12c5d1SDavid du Colombier 		fatal("no files to compile or load");
1579a747e4fSDavid du Colombier 	ccpath = smprint("/bin/%s", ot->cc);
1589a747e4fSDavid du Colombier 	append(&cpp, smprint("-I/%s/include/ape", ot->name));
1593e12c5d1SDavid du Colombier 	append(&cpp, "-I/sys/include/ape");
1603e12c5d1SDavid du Colombier 	cppn = cpp.n;
1613e12c5d1SDavid du Colombier 	ccn = cc.n;
1623e12c5d1SDavid du Colombier 	for(i = 0; i < srcs.n; i++) {
1633e12c5d1SDavid du Colombier 		append(&cpp, srcs.strings[i]);
1643e12c5d1SDavid du Colombier 		if(Pflag)
1650d35113bSDavid du Colombier 			append(&cpp, changeext(objs.strings[i], "i"));
1663e12c5d1SDavid du Colombier 		if(Eflag || Pflag)
1673e12c5d1SDavid du Colombier 			doexec("/bin/cpp", &cpp);
1683e12c5d1SDavid du Colombier 		else {
1693e12c5d1SDavid du Colombier 			append(&cc, "-o");
17059cc4ca5SDavid du Colombier 			if(oflag && cflag)
17159cc4ca5SDavid du Colombier 				append(&cc, oname);
17259cc4ca5SDavid du Colombier 			else
17359cc4ca5SDavid du Colombier 				append(&cc, changeext(srcs.strings[i], ot->o));
1743e12c5d1SDavid du Colombier 			dopipe("/bin/cpp", &cpp, ccpath, &cc);
1753e12c5d1SDavid du Colombier 		}
1763e12c5d1SDavid du Colombier 		cpp.n = cppn;
1773e12c5d1SDavid du Colombier 		cc.n = ccn;
1783e12c5d1SDavid du Colombier 	}
1793e12c5d1SDavid du Colombier 	if(!cflag) {
1803e12c5d1SDavid du Colombier 		append(&ld, "-o");
1813e12c5d1SDavid du Colombier 		append(&ld, oname);
1823e12c5d1SDavid du Colombier 		for(i = 0; i < ldargs.n; i++)
1833e12c5d1SDavid du Colombier 			append(&ld, ldargs.strings[i]);
1843e12c5d1SDavid du Colombier 		for(i = 0; i < objs.n; i++)
1853e12c5d1SDavid du Colombier 			append(&ld, objs.strings[i]);
186fb7f0c93SDavid du Colombier 		append(&ld, smprint("/%s/lib/ape/libap.a", ot->name));
1879a747e4fSDavid du Colombier 		doexec(smprint("/bin/%s", ot->ld), &ld);
1887dd7cddfSDavid du Colombier 		if(objs.n == 1){
1897dd7cddfSDavid du Colombier 			/* prevent removal of a library */
1907dd7cddfSDavid du Colombier 			if(strstr(objs.strings[0], ".a") == 0)
1913e12c5d1SDavid du Colombier 				remove(objs.strings[0]);
1923e12c5d1SDavid du Colombier 		}
1937dd7cddfSDavid du Colombier 	}
1943e12c5d1SDavid du Colombier 
1953e12c5d1SDavid du Colombier 	exits(0);
1963e12c5d1SDavid du Colombier }
1973e12c5d1SDavid du Colombier 
1983e12c5d1SDavid du Colombier void
append(List * l,char * s)1993e12c5d1SDavid du Colombier append(List *l, char *s)
2003e12c5d1SDavid du Colombier {
2013e12c5d1SDavid du Colombier 	if(l->n >= Maxlist-1)
2023e12c5d1SDavid du Colombier 		fatal("too many arguments");
2033e12c5d1SDavid du Colombier 	l->strings[l->n++] = s;
2043e12c5d1SDavid du Colombier 	l->strings[l->n] = 0;
2053e12c5d1SDavid du Colombier }
2063e12c5d1SDavid du Colombier 
2073e12c5d1SDavid du Colombier void
doexec(char * c,List * a)2083e12c5d1SDavid du Colombier doexec(char *c, List *a)
2093e12c5d1SDavid du Colombier {
2109a747e4fSDavid du Colombier 	Waitmsg *w;
2113e12c5d1SDavid du Colombier 
2123e12c5d1SDavid du Colombier 	if(vflag) {
2133e12c5d1SDavid du Colombier 		printlist(a);
2143e12c5d1SDavid du Colombier 		fprint(2, "\n");
2153e12c5d1SDavid du Colombier 	}
2163e12c5d1SDavid du Colombier 	switch(fork()) {
2173e12c5d1SDavid du Colombier 	case -1:
2183e12c5d1SDavid du Colombier 		fatal("fork failed");
2193e12c5d1SDavid du Colombier 	case 0:
2203e12c5d1SDavid du Colombier 		exec(c, a->strings);
2213e12c5d1SDavid du Colombier 		fatal("exec failed");
2223e12c5d1SDavid du Colombier 	}
2239a747e4fSDavid du Colombier 	w = wait();
2249a747e4fSDavid du Colombier 	if(w == nil)
2253e12c5d1SDavid du Colombier 		fatal("wait failed");
2269a747e4fSDavid du Colombier 	if(w->msg[0])
2279a747e4fSDavid du Colombier 		fatal(smprint("%s: %s", a->strings[0], w->msg));
2289a747e4fSDavid du Colombier 	free(w);
2293e12c5d1SDavid du Colombier }
2303e12c5d1SDavid du Colombier 
2313e12c5d1SDavid du Colombier void
dopipe(char * c1,List * a1,char * c2,List * a2)2323e12c5d1SDavid du Colombier dopipe(char *c1, List *a1, char *c2, List *a2)
2333e12c5d1SDavid du Colombier {
2349a747e4fSDavid du Colombier 	Waitmsg *w;
2359a747e4fSDavid du Colombier 	int pid1, got;
2363e12c5d1SDavid du Colombier 	int fd[2];
2373e12c5d1SDavid du Colombier 
2383e12c5d1SDavid du Colombier 	if(vflag) {
2393e12c5d1SDavid du Colombier 		printlist(a1);
2403e12c5d1SDavid du Colombier 		fprint(2, " | ");
2413e12c5d1SDavid du Colombier 		printlist(a2);
2423e12c5d1SDavid du Colombier 		fprint(2, "\n");
2433e12c5d1SDavid du Colombier 	}
2443e12c5d1SDavid du Colombier 	if(pipe(fd) < 0)
2453e12c5d1SDavid du Colombier 		fatal("pipe failed");
2463e12c5d1SDavid du Colombier 	switch((pid1 = fork())) {
2473e12c5d1SDavid du Colombier 	case -1:
2483e12c5d1SDavid du Colombier 		fatal("fork failed");
2493e12c5d1SDavid du Colombier 	case 0:
2503e12c5d1SDavid du Colombier 		dup(fd[0], 0);
2513e12c5d1SDavid du Colombier 		close(fd[0]);
2523e12c5d1SDavid du Colombier 		close(fd[1]);
2533e12c5d1SDavid du Colombier 		exec(c2, a2->strings);
2543e12c5d1SDavid du Colombier 		fatal("exec failed");
2553e12c5d1SDavid du Colombier 	}
2563e12c5d1SDavid du Colombier 	switch(fork()) {
2573e12c5d1SDavid du Colombier 	case -1:
2583e12c5d1SDavid du Colombier 		fatal("fork failed");
2593e12c5d1SDavid du Colombier 	case 0:
2603e12c5d1SDavid du Colombier 		close(0);
2613e12c5d1SDavid du Colombier 		dup(fd[1], 1);
2623e12c5d1SDavid du Colombier 		close(fd[0]);
2633e12c5d1SDavid du Colombier 		close(fd[1]);
2643e12c5d1SDavid du Colombier 		exec(c1, a1->strings);
2653e12c5d1SDavid du Colombier 		fatal("exec failed");
2663e12c5d1SDavid du Colombier 	}
2673e12c5d1SDavid du Colombier 	close(fd[0]);
2683e12c5d1SDavid du Colombier 	close(fd[1]);
2693e12c5d1SDavid du Colombier 	for(got = 0; got < 2; got++) {
2709a747e4fSDavid du Colombier 		w = wait();
2719a747e4fSDavid du Colombier 		if(w == nil)
2723e12c5d1SDavid du Colombier 			fatal("wait failed");
2739a747e4fSDavid du Colombier 		if(w->msg[0])
2749a747e4fSDavid du Colombier 			fatal(smprint("%s: %s",
2759a747e4fSDavid du Colombier 			   (w->pid == pid1) ? a1->strings[0] : a2->strings[0], w->msg));
2769a747e4fSDavid du Colombier 		free(w);
2773e12c5d1SDavid du Colombier 	}
2783e12c5d1SDavid du Colombier }
2793e12c5d1SDavid du Colombier 
2803e12c5d1SDavid du Colombier Objtype *
findoty(void)2813e12c5d1SDavid du Colombier findoty(void)
2823e12c5d1SDavid du Colombier {
2833e12c5d1SDavid du Colombier 	char *o;
2843e12c5d1SDavid du Colombier 	Objtype *oty;
2853e12c5d1SDavid du Colombier 
2863e12c5d1SDavid du Colombier 	o = getenv("objtype");
2873e12c5d1SDavid du Colombier 	if(!o)
2883e12c5d1SDavid du Colombier 		fatal("no $objtype in environment");
2893e12c5d1SDavid du Colombier 	for(oty = objtype; oty < &objtype[Nobjs]; oty++)
2903e12c5d1SDavid du Colombier 		if(strcmp(o, oty->name) == 0)
2913e12c5d1SDavid du Colombier 			return oty;
2923e12c5d1SDavid du Colombier 	fatal("unknown $objtype");
2933e12c5d1SDavid du Colombier 	return 0;			/* shut compiler up */
2943e12c5d1SDavid du Colombier }
2953e12c5d1SDavid du Colombier 
2963e12c5d1SDavid du Colombier void
fatal(char * msg)2973e12c5d1SDavid du Colombier fatal(char *msg)
2983e12c5d1SDavid du Colombier {
2993e12c5d1SDavid du Colombier 	fprint(2, "pcc: %s\n", msg);
3003e12c5d1SDavid du Colombier 	exits(msg);
3013e12c5d1SDavid du Colombier }
3023e12c5d1SDavid du Colombier 
3033e12c5d1SDavid du Colombier /* src ends in .something; return copy of basename with .ext added */
3043e12c5d1SDavid du Colombier char *
changeext(char * src,char * ext)3053e12c5d1SDavid du Colombier changeext(char *src, char *ext)
3063e12c5d1SDavid du Colombier {
3073e12c5d1SDavid du Colombier 	char *b, *e, *ans;
3083e12c5d1SDavid du Colombier 
3093e12c5d1SDavid du Colombier 	b = utfrrune(src, '/');
3103e12c5d1SDavid du Colombier 	if(b)
3113e12c5d1SDavid du Colombier 		b++;
3123e12c5d1SDavid du Colombier 	else
3133e12c5d1SDavid du Colombier 		b = src;
3143e12c5d1SDavid du Colombier 	e = utfrrune(src, '.');
3153e12c5d1SDavid du Colombier 	if(!e)
3163e12c5d1SDavid du Colombier 		return 0;
3173e12c5d1SDavid du Colombier 	*e = 0;
3189a747e4fSDavid du Colombier 	ans = smprint("%s.%s", b, ext);
3193e12c5d1SDavid du Colombier 	*e = '.';
3203e12c5d1SDavid du Colombier 	return ans;
3213e12c5d1SDavid du Colombier }
3223e12c5d1SDavid du Colombier 
3233e12c5d1SDavid du Colombier void
printlist(List * l)3243e12c5d1SDavid du Colombier printlist(List *l)
3253e12c5d1SDavid du Colombier {
3263e12c5d1SDavid du Colombier 	int i;
3273e12c5d1SDavid du Colombier 
3283e12c5d1SDavid du Colombier 	for(i = 0; i < l->n; i++) {
3293e12c5d1SDavid du Colombier 		fprint(2, "%s", l->strings[i]);
3303e12c5d1SDavid du Colombier 		if(i < l->n - 1)
3313e12c5d1SDavid du Colombier 			fprint(2, " ");
3323e12c5d1SDavid du Colombier 	}
3333e12c5d1SDavid du Colombier }
334