⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 cc.c

📁 minix软件源代码
💻 C
📖 第 1 页 / 共 2 页
字号:
/*	Driver for Minix compilers.	Written june 1987 by Ceriel J.H. Jacobs, partly derived from old	cc-driver, written by Erik Baalbergen.	This driver is mostly table-driven, the table being in the form of	some global initialized structures.*//* $Header: cc.c,v 1.35 91/04/12 15:33:50 ceriel Exp $ */#include <sys/types.h>#include <sys/wait.h>#include <errno.h>#include <signal.h>#include <string.h>#include <stdlib.h>#include <fcntl.h>#include <unistd.h>/* Paths.  (Executables in /usr are first tried with /usr stripped off.) */#define SHELL		"/bin/sh"#define PP		"/usr/lib/ncpp"#define IRREL		"/usr/lib/irrel"#define CEM		"/usr/lib/ncem"#define M2EM		"/usr/lib/nm2em"#define OPT		"/usr/lib/nopt"#define CG		"/usr/lib/ncg"#define AS		"/usr/lib/as"#define LD		"/usr/lib/ld"#define CV		"/usr/lib/cv"#define LIBDIR		"/usr/lib"#define CRT		"/usr/lib/ncrtso.o"#define PEM		"/usr/lib/npem"#define PRT		"/usr/lib/nprtso.o"#define M2RT		"/usr/lib/nm2rtso.o"#define LIBC            "/usr/lib/libd.a", "/usr/lib/libc.a"#define LIBP		"/usr/lib/libp.a", "/usr/lib/libc.a"#define LIBM2		"/usr/lib/libm2.a", "/usr/lib/libc.a"#define END             "/usr/lib/libe.a", "/usr/lib/end.a"#define M2DEF		"-I/usr/lib/m2"/*	every pass that this program knows about has associated with it	a structure, containing such information as its name, where it	resides, the flags it accepts, and the like.*/struct passinfo {	char *p_name;		/* name of this pass */	char *p_path;		/* where is it */	char *p_from;		/* suffix of source (comma-separated list) */	char *p_to;		/* suffix of destination */	char *p_acceptflags;	/* comma separated list; format:			   		flag			   		flag*			   		flag=xxx					flag*=xxx[*]				   where a star matches a, possibly empty, 				   string				*/	int  p_flags;#define INPUT	01		/* needs input file as argument */#define OUTPUT	02		/* needs output file as argument */#define LOADER	04		/* this pass is the loader */#define STDIN	010		/* reads from standard input */#define STDOUT	020		/* writes on standard output */#define NOCLEAN	040		/* do not remove target if this pass fails */#define O_OUTPUT 0100		/* -o outputfile, hack for as */#define PREPALWAYS	0200	/* always to be preprocessed */#define PREPCOND	0400	/* preprocessed when starting with '#' */};#define MAXHEAD	6#define MAXTAIL	6#define MAXPASS	7/*	Every language handled by this program has a "compile" structure	associated with it, describing the start-suffix, how the driver for	this language is called, which passes must be called, which flags	and arguments must be passed to these passes, etc.	The language is determined by the suffix of the argument program.	However, if this suffix does not determine a language (DEFLANG),	the callname is used.	Notice that the 's' suffix does not determine a language, because	the input file could have been derived from f.i. a C-program.	So, if you use "cc x.s", the C-runtime system will be used, but if	you use "as x.s", it will not.*/struct compile {	char *c_suffix;		/* starting suffix of this list of passes */	char *c_callname;	/* affects runtime system loaded with program */	struct pass {		char *pp_name;		/* name of the pass */		char *pp_head[MAXHEAD];	/* args in front of filename */		char *pp_tail[MAXTAIL];	/* args after filename */	} c_passes[MAXPASS];	int  c_flags;#define DEFLANG		010	/* this suffix determines a language */};struct passinfo passinfo[] = {	{ "cpp", PP, "CPP", "i", "wo=o,I*,D*,U*,P", INPUT|STDOUT },	{ "irrel", IRREL, "i", "i", "", INPUT},	{ "cem", CEM, "i,c", "k", "m,p,wa=a,wo=o,ws=s,w,T*", INPUT|OUTPUT|PREPALWAYS },	{ "pc", PEM, "i,p", "k", "n=L,w,a,A,R", INPUT|OUTPUT|PREPCOND },	{ "m2", M2EM, "i,mod", "k", "n=L,w*,A,R,W*,3,I*", INPUT|OUTPUT|PREPCOND },	{ "opt", OPT, "k", "m", "", STDIN|STDOUT },	{ "cg", CG, "m", "s", "O=p4", INPUT|OUTPUT },	{ "as", AS, "i,s", "o", "T*", INPUT|O_OUTPUT|PREPCOND },	{ "ld", LD, "o", "out", "i,s", INPUT|LOADER },	/* changed */	{ "cv", CV, "out", 0, "", INPUT|OUTPUT|NOCLEAN },	/* must come after loader */	{ 0}};#define	PREP_FLAGS	"-D_EM_WSIZE=2", "-D_EM_PSIZE=2", "-D_EM_LSIZE=4" \			, "-D__ACK__", "-D__minix", "-D__i86",struct pass preprocessor = { "cpp",			    { PREP_FLAGS }			    , {0}			    };struct pass irrel = { "irrel",			    {0}			};/* The "*" in the arguments for the loader indicates the place where the * fp-emulation library should come. */struct compile passes[] = {{	"c", "cc", 	{	{ "cem", {"-L"}, {0} },	/* changed */		{ "opt", {0}, {0} },		{ "cg", {0}, {0} },		{ "as", {"-"}, {0} },		{ "ld", {CRT}, /* changed */			  {LIBC, "*",  END}},		{ "cv", {0}, {0} }	},	DEFLANG},{	"p", "pc",	{	{ "pc", {0}, {0} },		{ "opt", {0}, {0} },		{ "cg", {0}, {0} },		{ "as", {"-"}, {0} },		{ "ld", {PRT}, 			  {LIBP,			    "*", END}},		{ "cv", {0}, {0} }	},	DEFLANG},{	"mod", "m2",	{	{ "m2", {M2DEF}, {0} },		{ "opt", {0}, {0} },		{ "cg", {0}, {0} },		{ "as", {"-"}, {0} },		{ "ld", {M2RT}, 			  {LIBM2,			    "*", END}},		{ "cv", {0}, {0} }	},	DEFLANG},{	"s", "as",	{	{ "as", {0}, {0}}	},	0},{	"CPP", "cpp",	{	{ "cpp", {PREP_FLAGS}, {0}}	},	DEFLANG},{	0},};#define MAXARGC	150	/* maximum number of arguments allowed in a list */#define USTR_SIZE	64	/* maximum length of string variable */typedef char USTRING[USTR_SIZE];struct arglist {	int al_argc;	char *al_argv[MAXARGC];};struct arglist CALLVEC;int kids = -1;char *o_FILE = "a.out"; /* default name for executable file */#define init(a)		((a)->al_argc = 1)#define cleanup(str)		(str && remove(str))char *ProgCall = 0;int RET_CODE = 0;char *stopsuffix;int m_flag = 0;int v_flag = 0;int t_flag = 0;int noexec = 0;int fp_lib = 0;int E_flag = 0;USTRING curfil;USTRING newfil;struct arglist SRCFILES;struct arglist LDIRS;struct arglist LDFILES;struct arglist GEN_LDFILES;struct arglist FLAGS;char *tmpdir = "/tmp";char tmpname[64];struct compile *compbase;struct pass *loader;struct passinfo *loaderinfo;char *source;int maxLlen;_PROTOTYPE(char *library, (char *nm ));_PROTOTYPE(void trapcc, (int sig ));_PROTOTYPE(int main, (int argc, char *argv []));_PROTOTYPE(int remove, (char *str ));_PROTOTYPE(char *alloc, (unsigned u ));_PROTOTYPE(int append, (struct arglist *al, char *arg ));_PROTOTYPE(int concat, (struct arglist *al1, struct arglist *al2 ));_PROTOTYPE(char *mkstr, (char *dst, char *arg1, char *arg2, char *arg3 ));_PROTOTYPE(int basename, (char *str, char *dst ));_PROTOTYPE(char *extension, (char *fln ));_PROTOTYPE(int runvec, (struct arglist *vec, struct passinfo *pass, char *in, char *out ));_PROTOTYPE(int prnum, (unsigned x ));_PROTOTYPE(int prs, (char *str ));_PROTOTYPE(int panic, (char *str ));_PROTOTYPE(int pr_vec, (struct arglist *vec ));_PROTOTYPE(int ex_vec, (struct arglist *vec ));_PROTOTYPE(int mktempname, (char *nm ));_PROTOTYPE(int mkbase, (void));_PROTOTYPE(int mkloader, (void));_PROTOTYPE(int needsprep, (char *name ));_PROTOTYPE(int cfile, (char *name ));_PROTOTYPE(char *apply, (struct passinfo *pinf, struct compile *cp, char *name, int passindex, int noremove, int first, char *resultname ));_PROTOTYPE(int applicable, (struct passinfo *pinf, char *suffix ));_PROTOTYPE(char *process, (char *name, int noremove ));_PROTOTYPE(int mkvec, (struct arglist *call, char *in, char *out, struct pass *pass, struct passinfo *pinf ));_PROTOTYPE(int callld, (struct arglist *in, char *out, struct pass *pass, struct passinfo *pinf ));_PROTOTYPE(int clean, (struct arglist *c ));_PROTOTYPE(int scanflags, (struct arglist *call, struct passinfo *pinf ));char *library(nm)	char	*nm;{	static char	f[512];	int	Lcount;	for (Lcount = 0; Lcount < LDIRS.al_argc; Lcount++) {		mkstr(f, LDIRS.al_argv[Lcount], "/lib", nm);		strcat(f, ".a");		if (access(f, 0) != 0) {			f[strlen(f)-1] = 'a';			if (access(f, 0) != 0) continue;		}		return f;	}	mkstr(f, LIBDIR, "/lib", nm);	strcat(f, ".a");	if (access(f, 0) != 0) {		int i = strlen(f) - 1;		f[i] = 'a';		if (access(f, 0) != 0) f[i] = 'A';	}	return f;}void trapcc(sig)	int sig;{	signal(sig, SIG_IGN);	if (kids != -1) kill(kids, sig);	cleanup(newfil);	cleanup(curfil);	exit(1);}main(argc, argv)	char *argv[];{	char *str;	char **argvec;	int count;	char *file;	maxLlen = strlen(LIBDIR);	ProgCall = *argv++;	mkbase();	if (signal(SIGHUP, SIG_IGN) != SIG_IGN)		signal(SIGHUP, trapcc);	if (signal(SIGINT, SIG_IGN) != SIG_IGN)		signal(SIGINT, trapcc);	if (signal(SIGQUIT, SIG_IGN) != SIG_IGN)		signal(SIGQUIT, trapcc);	while (--argc > 0) {		if (*(str = *argv++) != '-' || str[1] == 0) {			append(&SRCFILES, str);			continue;		}		switch (str[1]) {		case 'c':			stopsuffix = "o";			if (str[2] == '.') stopsuffix = str + 3;			break;		case 'f':			fp_lib = 1;			break;		case 'F':		case 'W':			/* Ignore. */			break;		case 'L':			append(&LDIRS, &str[2]);			count = strlen(&str[2]);			if (count > maxLlen) maxLlen = count;			break;		case 'l':			append(&SRCFILES, library(&str[2]));			break;		case 'm':			if (str[2] == 0)				m_flag++;			break;		case 'o':			if (argc-- >= 0)				o_FILE = *argv++;			break;		case 'S':			stopsuffix = "s";			break;		case 'E':			E_flag = 1;			stopsuffix = "i";			break;		case 'P':			stopsuffix = "i";			append(&FLAGS, str);			break;		case 'v':			v_flag++;			if (str[2] == 'n')				noexec = 1;			break;		case 't':			/* save temporaries */			t_flag++;			break;		case '.':			if (str[2] == 'o') {				/* no runtime start-off */				loader->pp_head[0] = 0;			}			break;		case 'T':			tmpdir = &str[2];			/*FALLTHROUGH*/		default:			append(&FLAGS, str);		}	}	mktempname(tmpname);	count = SRCFILES.al_argc;	argvec = &(SRCFILES.al_argv[0]);	while (count-- > 0) {		file = *argvec++;		source = file;		file = process(file, 1);			if (file && ! stopsuffix) append(&LDFILES, file);	}	clean(&SRCFILES);	/* loader ... */	if (RET_CODE == 0 && LDFILES.al_argc > 0) {		register struct passinfo *pp = passinfo;		while (!(pp->p_flags & LOADER)) pp++;		mkstr(newfil, tmpname, pp->p_to, "");		callld(&LDFILES, !((pp+1)->p_name) ? o_FILE : newfil, loader, pp);		if (RET_CODE == 0) {			register int i = GEN_LDFILES.al_argc;			while (i-- > 0) {				remove(GEN_LDFILES.al_argv[i]);				free(GEN_LDFILES.al_argv[i]);			}			if ((++pp)->p_name) {				process(newfil, 0);			}		}	}	exit(RET_CODE);}remove(str)	char *str;{	if (t_flag)		return;	if (v_flag) {		prs("rm ");		prs(str);		prs("\n");	}	if (noexec)		return;	unlink(str);}char *alloc(u)	unsigned u;{	register char *p = malloc(u);	if (p == 0) panic("no space\n");	return p;}append(al, arg)	struct arglist *al;	char *arg;{	char *a = alloc((unsigned) (strlen(arg) + 1));	strcpy(a, arg);	if (al->al_argc >= MAXARGC)		panic("argument list overflow\n");	al->al_argv[(al->al_argc)++] = a;}concat(al1, al2)	struct arglist *al1, *al2;{	register i = al2->al_argc;	register char **p = &(al1->al_argv[al1->al_argc]);	register char **q = &(al2->al_argv[0]);	if ((al1->al_argc += i) >= MAXARGC)		panic("argument list overflow\n");	while (i-- > 0)		*p++ = *q++;}char *mkstr(dst, arg1, arg2, arg3)	char *dst, *arg1, *arg2, *arg3;{	register char *p;	register char *q = dst;	p = arg1;	while (*q++ = *p++);	q--;	p = arg2;	while (*q++ = *p++);	q--;	p = arg3;	while (*q++ = *p++);	q--;	return dst;}basename(str, dst)	char *str;	register char *dst;{	register char *p1 = str;	register char *p2 = p1;	while (*p1)		if (*p1++ == '/')			p2 = p1;	p1--;	while (*p1 != '.' && p1 > p2) p1--;	if (*p1 == '.') {		*p1 = '\0';		while (*dst++ = *p2++);		*p1 = '.';	}	else		while (*dst++ = *p2++);}char *extension(fln)	char *fln;{	register char *fn = fln;

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -