1097 lines
		
	
	
	
		
			18 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1097 lines
		
	
	
	
		
			18 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*	fcc/fm2/fpc 
 | |
| 	Driver for fast ACK compilers.
 | |
| 
 | |
| 	Derived from the C compiler driver from Minix.
 | |
| 
 | |
| 	You can create cc-compatible versions by compiling with
 | |
| 		cc -O '-DFASTDIR="<this directory>"' -DF?? driver.c
 | |
| 	where F?? is either FCC, FM2, or FPC.
 | |
| 	Install resulting binaries in a public place (f.i. /usr/local/bin).
 | |
| 	Suggested names: fcc, fm2, and fpc.
 | |
| 
 | |
| 	If you have ACK, you can create ACK-compatible versions by compiling
 | |
| 	this file with
 | |
| 		cc -O -DACK_BIN -I<ACK home directory>/h -DF?? driver.c
 | |
| 	where F?? is either FCC, FPC, or FM2.
 | |
| 	Install the resulting binaries in the EM bin directory.
 | |
| 	Suggested names: afcc, afm2, and afpc.
 | |
| 	The ACK compatible version afpc is not compatible with the 4th
 | |
| 	ACK distribution, and thus is not on the tape.
 | |
| */
 | |
| 
 | |
| 
 | |
| #ifndef vax
 | |
| #ifndef sun
 | |
| In this case, no fast ACK compiler available
 | |
| #endif
 | |
| #endif
 | |
| 
 | |
| #if FM2+FPC+FCC > 1
 | |
| Something wrong here! Only one of FM2, FPC, or FCC must be defined
 | |
| #endif
 | |
| 
 | |
| #include <errno.h>
 | |
| #include <signal.h>
 | |
| #include <varargs.h>
 | |
| #include <stdio.h>
 | |
| #ifdef ACK_BIN
 | |
| #include <em_path.h>
 | |
| #define M2DEF	"/lib/m2"
 | |
| #define FASTDIR EM_DIR
 | |
| #else
 | |
| #define M2DEF "/def"
 | |
| #define TMP_DIR	"/tmp"
 | |
| #endif ACK_BIN
 | |
| 
 | |
| 
 | |
| #ifndef ACK_BIN
 | |
| #define BINDIR	"/bin/"
 | |
| #define LIBDIR	"/lib/"
 | |
| #else ACK_BIN
 | |
| #define BINDIR	"/lib/"
 | |
| #ifdef sun
 | |
| #define LIBDIR	"/lib/sun3/"
 | |
| #endif sun
 | |
| #ifdef vax
 | |
| #define LIBDIR	"/lib/vax4/"
 | |
| #endif vax
 | |
| #define CCINCL	"/include/_tail_cc"
 | |
| #endif ACK_BIN
 | |
| char *ROOT_DIR = FASTDIR;
 | |
| 
 | |
| /*
 | |
| #ifndef ACK_BIN
 | |
| 	Version producing VAX or SUN .o files in one pass.
 | |
| #else ACK_BIN
 | |
| 	Version producing ACK .o files in one pass.
 | |
| #endif ACK_BIN
 | |
| */
 | |
| #define MAXARGC	256	/* maximum number of arguments allowed in a list */
 | |
| #define USTR_SIZE	128	/* maximum length of string variable */
 | |
| 
 | |
| typedef char USTRING[USTR_SIZE];
 | |
| 
 | |
| USTRING INCLUDE = "-I";
 | |
| 
 | |
| struct arglist {
 | |
| 	int al_argc;
 | |
| 	char *al_argv[MAXARGC];
 | |
| };
 | |
| 
 | |
| #define LD	"/bin/ld"
 | |
| #define AS	"/bin/as"
 | |
| #define CPP	"*cpp"
 | |
| #if ACK_BIN && !vax
 | |
| #undef LD
 | |
| #undef AS
 | |
| #define LD	"*em_led"
 | |
| #define CV	"~cv"		/* machine dependent; resides in lib directory */
 | |
| #endif ACK_BIN && !vax
 | |
| #define SHELL	"/bin/sh"
 | |
| 
 | |
| /*	flags passed to linker */
 | |
| #define TO_LD	"dnNrsSi"
 | |
| #if ACK_BIN && !vax
 | |
| #undef TO_LD
 | |
| #define TO_LD	"rs"
 | |
| #endif ACK_BIN && !vax
 | |
| 
 | |
| int kids =  -1;
 | |
| int ecount = 0;
 | |
| 
 | |
| struct arglist CPP_FLAGS = {
 | |
| #ifdef vax
 | |
| 	8,
 | |
| #endif
 | |
| #ifdef sun
 | |
| 	10,
 | |
| #endif
 | |
| 	{
 | |
| 		"-Dunix",
 | |
| 		"-DEM_WSIZE=4",
 | |
| 		"-DEM_PSIZE=4",
 | |
| 		"-DEM_SSIZE=2",
 | |
| 		"-DEM_LSIZE=4",
 | |
| 		"-DEM_FSIZE=4",
 | |
| 		"-DEM_DSIZE=8",
 | |
| #ifdef vax
 | |
| 		"-Dvax"
 | |
| #endif
 | |
| #ifdef sun
 | |
| 		"-Dmc68000",
 | |
| 		"-Dsun",
 | |
| 		"-Dmc68020"
 | |
| #endif
 | |
| 	}
 | |
| };
 | |
| 
 | |
| struct arglist COMP_FLAGS;
 | |
| 
 | |
| #ifdef FCC
 | |
| struct arglist LD_HEAD = {
 | |
| #ifdef ACK_BIN
 | |
| 	2,
 | |
| 	{
 | |
| 		"~head_em",
 | |
| 		"~head_cc"
 | |
| 	}
 | |
| #else ACK_BIN
 | |
| #ifdef sun
 | |
| #ifdef SunOs4
 | |
| 	7,
 | |
| #else
 | |
| 	5,
 | |
| #endif
 | |
| 	{
 | |
| 		"-e",
 | |
| 		"start",
 | |
| 		"-X",
 | |
| #ifdef SunOs4
 | |
| 		"-L/usr/lib/fsoft",
 | |
| 		"-Bstatic",
 | |
| #endif
 | |
| 		"/lib/crt0.o",
 | |
| 		"/lib/Fcrt1.o"
 | |
| 	}
 | |
| #endif
 | |
| #ifdef vax
 | |
| 	2,
 | |
| 	{
 | |
| 		"-X",
 | |
| 		"/lib/crt0.o"
 | |
| 	}
 | |
| #endif
 | |
| #endif ACK_BIN
 | |
| };
 | |
| 
 | |
| struct arglist LD_TAIL = {
 | |
| #ifdef ACK_BIN
 | |
| #ifdef sun
 | |
| 	6,
 | |
| 	{
 | |
| 		"~tail_cc.1s",
 | |
| 		"~tail_cc.2g",
 | |
| 		"*m68020/tail_fp",
 | |
| 		"*m68020/tail_em",
 | |
| 		"~tail_mon",
 | |
| 		"*m68020/end_em"
 | |
| 	}
 | |
| #endif sun
 | |
| #ifdef vax
 | |
| 	4,
 | |
| 	{
 | |
| 		"~tail_cc.1s",
 | |
| 		"~tail_cc.2g",
 | |
| 		"~tail_em",
 | |
| 		"~tail_mon"
 | |
| 	}
 | |
| #endif vax
 | |
| #else ACK_BIN
 | |
| #ifdef sun
 | |
| 	2,
 | |
| 	{
 | |
| 		"~tail_ext",
 | |
| 		"-lc"
 | |
| 	}
 | |
| #endif
 | |
| #ifdef vax
 | |
| 	2,
 | |
| 	{
 | |
| 		"~tail_em",
 | |
| 		"-lc"
 | |
| 	}
 | |
| #endif
 | |
| #endif ACK_BIN
 | |
| };
 | |
| #endif FCC
 | |
| 
 | |
| #ifdef FPC
 | |
| struct arglist LD_HEAD = {
 | |
| #ifdef ACK_BIN
 | |
| 	2,
 | |
| 	{
 | |
| 		"~head_em",
 | |
| #ifdef vax
 | |
| 		"~head_pc"
 | |
| #endif
 | |
| #ifdef sun
 | |
| 		"*m68020/head_pc"
 | |
| #endif
 | |
| 	}
 | |
| #else ACK_BIN
 | |
| #ifdef sun
 | |
| #ifdef SunOs4
 | |
| 	7,
 | |
| #else
 | |
| 	5,
 | |
| #endif
 | |
| 	{
 | |
| 		"-e",
 | |
| 		"begtext",
 | |
| 		"-X",
 | |
| #ifdef SunOs4
 | |
| 		"-L/usr/lib/fsoft",
 | |
| 		"-Bstatic",
 | |
| #endif
 | |
| 		"~head_em",
 | |
| 		"~head_pc"
 | |
| 	}
 | |
| #endif sun
 | |
| #ifdef vax
 | |
| 	3,
 | |
| 	{
 | |
| 		"-X",
 | |
| 		"~head_em",
 | |
| 		"~head_pc"
 | |
| 	}
 | |
| #endif vax
 | |
| #endif ACK_BIN
 | |
| };
 | |
| struct arglist LD_TAIL = {
 | |
| #ifndef ACK_BIN
 | |
| #ifdef sun
 | |
| 	3,
 | |
| 	{
 | |
| 		"~tail_pc",
 | |
| 		"~tail_ext",
 | |
| 		"-lc"
 | |
| 	}
 | |
| #endif sun
 | |
| #ifdef vax
 | |
| 	3,
 | |
| 	{
 | |
| 		"~tail_pc",
 | |
| 		"~tail_em",
 | |
| 		"-lc"
 | |
| 	}
 | |
| #endif vax
 | |
| #else ACK_BIN
 | |
| #ifdef sun
 | |
| 	5,
 | |
| 	{
 | |
| 		"*m68020/tail_pc",
 | |
| 		"*m68020/tail_fp",
 | |
| 		"*m68020/tail_em",
 | |
| 		"~tail_mon",
 | |
| 		"*m68020/end_em"
 | |
| 	}
 | |
| #endif sun
 | |
| #ifdef vax
 | |
| 	3,
 | |
| 	{
 | |
| 		"~tail_pc",
 | |
| 		"~tail_em",
 | |
| 		"~tail_mon"
 | |
| 	}
 | |
| #endif vax
 | |
| #endif ACK_BIN
 | |
| };
 | |
| #endif FPC
 | |
| 
 | |
| #ifdef FM2
 | |
| struct arglist LD_HEAD = {
 | |
| #ifdef ACK_BIN
 | |
| 	2,
 | |
| 	{
 | |
| 		"~head_em",
 | |
| 		"~head_m2"
 | |
| 	}
 | |
| #else ACK_BIN
 | |
| #ifdef sun
 | |
| #ifdef SunOs4
 | |
| 	7,
 | |
| #else
 | |
| 	5,
 | |
| #endif
 | |
| 	{
 | |
| 		"-e",
 | |
| 		"begtext",
 | |
| 		"-X",
 | |
| #ifdef SunOs4
 | |
| 		"-L/usr/lib/fsoft",
 | |
| 		"-Bstatic",
 | |
| #endif
 | |
| 		"~head_em",
 | |
| 		"~head_m2"
 | |
| 	}
 | |
| #endif sun
 | |
| #ifdef vax
 | |
| 	3,
 | |
| 	{
 | |
| 		"-X",
 | |
| 		"~head_em",
 | |
| 		"~head_m2"
 | |
| 	}
 | |
| #endif vax
 | |
| #endif ACK_BIN
 | |
| };
 | |
| struct arglist LD_TAIL = {
 | |
| #ifndef ACK_BIN
 | |
| #ifdef sun
 | |
| 	3,
 | |
| 	{
 | |
| 		"~tail_m2",
 | |
| 		"~tail_ext",
 | |
| 		"-lc"
 | |
| 	}
 | |
| #endif sun
 | |
| #ifdef vax
 | |
| 	3,
 | |
| 	{
 | |
| 		"~tail_m2",
 | |
| 		"~tail_em",
 | |
| 		"-lc"
 | |
| 	}
 | |
| #endif vax
 | |
| #else ACK_BIN
 | |
| #ifdef sun
 | |
| 	5,
 | |
| 	{
 | |
| 		"~tail_m2",
 | |
| 		"*m68020/tail_fp",
 | |
| 		"*m68020/tail_em",
 | |
| 		"~tail_mon",
 | |
| 		"*m68020/end_em"
 | |
| 	}
 | |
| #endif sun
 | |
| #ifdef vax
 | |
| 	3,
 | |
| 	{
 | |
| 		"~tail_m2",
 | |
| 		"~tail_em",
 | |
| 		"~tail_mon"
 | |
| 	}
 | |
| #endif vax
 | |
| #endif ACK_BIN
 | |
| };
 | |
| #endif FM2
 | |
| 
 | |
| #if ACK_BIN && !vax
 | |
| #ifdef sun
 | |
| struct arglist def_align = {
 | |
| 	5,
 | |
| 	{
 | |
| 		"-a0:4",
 | |
| 		"-a1:4",
 | |
| 		"-a2:0x20000",
 | |
| 		"-a3:4",
 | |
| 		"-b0:0x2020"
 | |
| 	}
 | |
| };
 | |
| 
 | |
| struct arglist n_align = {
 | |
| 	5,
 | |
| 	{
 | |
| 		"-a0:4",
 | |
| 		"-a1:4",
 | |
| 		"-a2:0x20000",
 | |
| 		"-a3:4",
 | |
| 		"-b0:0x2000"
 | |
| 	}
 | |
| };
 | |
| 
 | |
| struct arglist N_align = {
 | |
| 	5,
 | |
| 	{
 | |
| 		"-a0:4",
 | |
| 		"-a1:4",
 | |
| 		"-a2:4",
 | |
| 		"-a3:4",
 | |
| 		"-b0:0x2000"
 | |
| 	}
 | |
| };
 | |
| #endif sun
 | |
| 
 | |
| struct arglist *sect_align = &def_align;
 | |
| 
 | |
| #endif ACK_BIN && !vax
 | |
| 
 | |
| char *o_FILE = "a.out"; /* default name for executable file */
 | |
| 
 | |
| #define remove(str)	((noexec || unlink(str)), (str)[0] = '\0')
 | |
| #define cleanup(str)		(str && str[0] && remove(str))
 | |
| #define init(al)		((al)->al_argc = 1)
 | |
| 
 | |
| char ProgCall[128];
 | |
| 
 | |
| struct arglist SRCFILES;
 | |
| struct arglist LDFILES;
 | |
| struct arglist GEN_LDFILES;
 | |
| 
 | |
| int RET_CODE = 0;
 | |
| 
 | |
| struct arglist LD_FLAGS;
 | |
| 
 | |
| struct arglist CALL_VEC;
 | |
| 
 | |
| int o_flag = 0;
 | |
| int c_flag = 0;
 | |
| int v_flag = 0;
 | |
| int O_flag = 0;
 | |
| 
 | |
| char *mkstr();
 | |
| char *malloc();
 | |
| char *alloc();
 | |
| char *extension();
 | |
| 
 | |
| USTRING ofile;
 | |
| USTRING BASE;
 | |
| USTRING tmp_file;
 | |
| 
 | |
| int noexec = 0;
 | |
| 
 | |
| extern char *strcat(), *strcpy(), *mktemp(), *strchr();
 | |
| 
 | |
| trapcc(sig)
 | |
| 	int sig;
 | |
| {
 | |
| 	signal(sig, SIG_IGN);
 | |
| 	if (kids != -1) kill(kids, sig);
 | |
| 	cleanup(ofile);
 | |
| 	cleanup(tmp_file);
 | |
| 	exit(1);
 | |
| }
 | |
| 
 | |
| #ifdef FCC
 | |
| #define lang_suffix()	"c"
 | |
| #ifndef ACK_BIN
 | |
| #define comp_name()	"*cemcom_ce"
 | |
| #else
 | |
| #ifdef vax
 | |
| #define comp_name()	"~cemcom_ce"
 | |
| #endif
 | |
| #ifdef sun
 | |
| #define comp_name()	"*m68020/cemcom_ce"
 | |
| #endif
 | |
| #endif ACK_BIN
 | |
| #endif FCC
 | |
| 
 | |
| #ifdef FM2
 | |
| #define lang_suffix()	"mod"
 | |
| #ifndef ACK_BIN
 | |
| #define comp_name()	"*m2_ce"
 | |
| #else
 | |
| #ifdef vax
 | |
| #define comp_name()	"~m2_ce"
 | |
| #endif
 | |
| #ifdef sun
 | |
| #define comp_name()	"*m68020/m2_ce"
 | |
| #endif
 | |
| #endif ACK_BIN
 | |
| #endif FM2
 | |
| 
 | |
| #ifdef FPC
 | |
| #define lang_suffix()	"p"
 | |
| #ifndef ACK_BIN
 | |
| #define comp_name()	"*pc_ce"
 | |
| #else
 | |
| #ifdef vax
 | |
| #define comp_name()	"~pc_ce"
 | |
| #endif
 | |
| #ifdef sun
 | |
| #define comp_name()	"*m68020/pc_ce"
 | |
| #endif
 | |
| #endif ACK_BIN
 | |
| 
 | |
| #endif FPC
 | |
| 
 | |
| 
 | |
| #ifdef FCC
 | |
| int
 | |
| lang_opt(str)
 | |
| 	char *str;
 | |
| {
 | |
| 	switch(str[1]) {
 | |
| 	case '-':	/* debug options */
 | |
| 	case 'w':	/* disable warnings */
 | |
| 	case 'R':	/* strict K&R C.  */
 | |
| 		append(&COMP_FLAGS, str);
 | |
| 		return 1;
 | |
| 	}
 | |
| 	return 0;
 | |
| }
 | |
| #endif FCC
 | |
| 
 | |
| #ifdef FM2
 | |
| int
 | |
| lang_opt(str)
 | |
| 	char *str;
 | |
| {
 | |
| 	switch(str[1]) {
 | |
| 	case '-':	/* debug options */
 | |
| 	case 'w':	/* disable warnings */
 | |
| 	case 'R':	/* no runtime checks */
 | |
| 	case 'W':	/* add warnings */
 | |
| 	case 'L':	/* no line numbers */
 | |
| 	case 'A':	/* extra array bound checks */
 | |
| 	case '3':	/* only accept 3rd edition Modula-2 */
 | |
| 		append(&COMP_FLAGS, str);
 | |
| 		return 1;
 | |
| 	case 'I':
 | |
| 		append(&COMP_FLAGS, str);
 | |
| 		break;	/* !!! */
 | |
| 	case 'U':	/* underscores in identifiers allowed */
 | |
| 		if (str[2] == '\0') {
 | |
| 			append(&COMP_FLAGS, str);
 | |
| 			return 1;
 | |
| 		}
 | |
| 		break;
 | |
| 	case 'e':	/* local extension for Modula-2 compiler:
 | |
| 			   procedure constants
 | |
| 			*/
 | |
| 		str[1] = 'l';
 | |
| 		append(&COMP_FLAGS, str);
 | |
| 		return 1;
 | |
| 	}
 | |
| 	return 0;
 | |
| }
 | |
| #endif FM2
 | |
| 
 | |
| #ifdef FPC
 | |
| int
 | |
| lang_opt(str)
 | |
| 	char *str;
 | |
| {
 | |
| 	switch(str[1]) {
 | |
| 	case '-':	/* debug options */
 | |
| 	case 'a':	/* enable assertions */
 | |
| 	case 'd':	/* allow doubles (longs) */
 | |
| 	case 'i':	/* set size of integer sets */
 | |
| 	case 't':	/* tracing */
 | |
| 	case 'w':	/* disable warnings */
 | |
| 	case 'A':	/* extra array bound checks */
 | |
| 	case 'C':	/* distinguish between lower case and upper case */
 | |
| 	case 'L':	/* no FIL and LIN instructions */
 | |
| 	case 'R':	/* no runtime checks */
 | |
| 		append(&COMP_FLAGS, str);
 | |
| 		return 1;
 | |
| 	case 'u':
 | |
| 	case 'U':
 | |
| 		/* underscores in identifiers */
 | |
| 	case 's':
 | |
| 		/* only compile standard pascal */
 | |
| 	case 'c':
 | |
| 		/* C type strings */
 | |
| 		if (str[2] == '+' && str[3] == '\0') {
 | |
| 			str[2] = 0;
 | |
| 			append(&COMP_FLAGS, str);
 | |
| 			return 1;
 | |
| 		}
 | |
| 	}
 | |
| 	return 0;
 | |
| }
 | |
| #endif FPC
 | |
| 
 | |
| main(argc, argv)
 | |
| 	char *argv[];
 | |
| {
 | |
| 	char *str;
 | |
| 	char **argvec;
 | |
| 	int count;
 | |
| 	char *ext;
 | |
| 	register struct arglist *call = &CALL_VEC;
 | |
| 	char *file;
 | |
| 	char *ldfile = 0;
 | |
| 	USTRING COMP;
 | |
| 	int compile_cnt = 0;
 | |
| 
 | |
| 	setbuf(stdout, (char *) 0);
 | |
| 	basename(*argv++,ProgCall);
 | |
| 	strcat(INCLUDE, ROOT_DIR);
 | |
| #ifdef FM2
 | |
| 	strcat(INCLUDE, M2DEF);
 | |
| #endif FM2
 | |
| #ifdef FCC
 | |
| #ifdef ACK_BIN
 | |
| 	strcat(INCLUDE, CCINCL);
 | |
| #else
 | |
| 	INCLUDE[0] = '\0';
 | |
| #endif ACK_BIN
 | |
| #endif FCC
 | |
| #ifdef FPC
 | |
| 	INCLUDE[0] = '\0';
 | |
| #endif FPC
 | |
| 	strcpy(COMP,comp_name());
 | |
| 
 | |
| 	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++) != '-') {
 | |
| 			append(&SRCFILES, str);
 | |
| 			continue;
 | |
| 		}
 | |
| 
 | |
| 		if (lang_opt(str)) {
 | |
| 		}
 | |
| 		else switch (str[1]) {
 | |
| 
 | |
| 		case 'c':	/* stop after producing .o files */
 | |
| 			c_flag = 1;
 | |
| 			break;
 | |
| 		case 'D':	/* preprocessor #define */
 | |
| 		case 'U':	/* preprocessor #undef */
 | |
| 			append(&CPP_FLAGS, str);
 | |
| 			break;
 | |
| 		case 'I':	/* include directory */
 | |
| 			append(&CPP_FLAGS, str);
 | |
| 			break;
 | |
| 		case 'o':	/* target file */
 | |
| 			if (argc-- >= 0) {
 | |
| 				o_flag = 1;
 | |
| 				o_FILE = *argv++;
 | |
| 				ext = extension(o_FILE);
 | |
| 				if (ext != o_FILE && ! strcmp(ext, lang_suffix())
 | |
| #ifndef ACK_BIN
 | |
| 				     || !strcmp(ext, "s")
 | |
| #endif
 | |
| 				) {
 | |
| 					error("-o would overwrite %s", o_FILE);
 | |
| 				}
 | |
| 			}
 | |
| 			break;
 | |
| 		case 'u':	/* mark identifier as undefined */
 | |
| 			append(&LD_FLAGS, str);
 | |
| 			if (argc-- >= 0)
 | |
| 				append(&LD_FLAGS, *argv++);
 | |
| 			break;
 | |
| 		case 'O':	/* use built in peephole optimizer */
 | |
| 			if (! O_flag) strcat(COMP, "opt");
 | |
| 			O_flag = 1;
 | |
| 			break;
 | |
| 		case 'v':	/* verbose */
 | |
| 			v_flag++;
 | |
| 			if (str[2] == 'n')
 | |
| 				noexec = 1;
 | |
| 			break;
 | |
| 		case 'l':	/* library file */
 | |
| 			append(&SRCFILES, str);
 | |
| 			break;
 | |
| 		case 'M':	/* use other compiler (for testing) */
 | |
| 			strcpy(COMP, str+2);
 | |
| 			break;
 | |
| #if ACK_BIN && !vax
 | |
| 		case 'n':	/* text portion sharable, read-only. No
 | |
| 				   demand paging
 | |
| 				*/
 | |
| 			sect_align = &n_align;
 | |
| 			break;
 | |
| 		case 'N':	/* text portion not sharable, not read-only */
 | |
| 			sect_align = &N_align;
 | |
| 			break;
 | |
| 		case 'd':	/* force definition of common storage */
 | |
| 			append(&LD_FLAGS, "-c");
 | |
| 			break;
 | |
| #endif ACK_BIN && !vax
 | |
| 		default:
 | |
| #ifdef sun
 | |
| 			if (! strcmp(str, "-sun3")) {
 | |
| 				/* SunOs4.0 make passes this flag to
 | |
| 				   $(CC). Ignore it silently.
 | |
| 				*/
 | |
| 				break;
 | |
| 			}
 | |
| #endif
 | |
| 			if (strchr(TO_LD, str[1]) == 0)
 | |
| 				warning("%s flag ignored", str);
 | |
| 			else {
 | |
| 				append(&LD_FLAGS, str);
 | |
| 			}
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (ecount) exit(1);
 | |
| 
 | |
| #if FCC && !ACK_BIN
 | |
| 	append(&COMP_FLAGS, "-L"); /* disable line number and filename admin */
 | |
| #endif FCC && !ACK_BIN
 | |
| 	count = SRCFILES.al_argc;
 | |
| 	argvec = &(SRCFILES.al_argv[0]);
 | |
| 	while (count-- > 0) {
 | |
| 		ext = extension(*argvec);
 | |
| 		if (*argvec[0] != '-' && 
 | |
| 		    ext != *argvec++ && (! strcmp(ext, lang_suffix())
 | |
| #ifndef ACK_BIN
 | |
| 		     || !strcmp(ext, "s")
 | |
| #endif
 | |
| 		)) {
 | |
| 			compile_cnt++;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (compile_cnt > 1 && c_flag && o_flag) {
 | |
| 		warning("-o flag ignored");
 | |
| 		o_flag = 0;
 | |
| 	}
 | |
| 
 | |
| 	count = SRCFILES.al_argc;
 | |
| 	argvec = &(SRCFILES.al_argv[0]);
 | |
| 	while (count-- > 0) {
 | |
| 		register char *f;
 | |
| 		basename(file = *argvec++, BASE);
 | |
| 
 | |
| 		ext = extension(file);
 | |
| 
 | |
| 		if (file[0] != '-' &&
 | |
| 		    ext != file && (!strcmp(ext, lang_suffix())
 | |
| #ifndef ACK_BIN
 | |
| 		    || !strcmp(ext, "s")
 | |
| #endif
 | |
| 		)) {
 | |
| #ifndef ACK_BIN
 | |
| 			if (compile_cnt > 1)
 | |
| 				printf("%s:\n", file);
 | |
| #else
 | |
| 			printf("%s\n", file);
 | |
| #endif
 | |
| 
 | |
| 			ldfile = c_flag ? ofile : alloc((unsigned)strlen(BASE)+3);
 | |
| 			if (
 | |
| #ifdef FCC
 | |
| 			    !strcmp(ext, "s") &&
 | |
| #endif
 | |
| 			    needsprep(file)) {
 | |
| 				strcpy(tmp_file, TMP_DIR);
 | |
| 				strcat(tmp_file, "/F_XXXXXX");
 | |
| 				mktemp(tmp_file);
 | |
| 				init(call);
 | |
| 				append(call, CPP);
 | |
| 				concat(call, &CPP_FLAGS);
 | |
| 				append(call, INCLUDE);
 | |
| 				append(call, file);
 | |
| 				if (runvec(call, tmp_file)) {
 | |
| 					file = tmp_file;
 | |
| 				}
 | |
| 				else {
 | |
| 					remove(tmp_file);
 | |
| 					tmp_file[0] = '\0';
 | |
| 					continue;
 | |
| 				}
 | |
| 			}
 | |
| 			init(call);
 | |
| 			if (o_flag && c_flag) {
 | |
| 				f = o_FILE;
 | |
| 			}
 | |
| 			else	f = mkstr(ldfile, BASE, ".o", (char *)0);
 | |
| #ifndef ACK_BIN
 | |
| 			if (strcmp(ext, "s")) {
 | |
| #endif
 | |
| 				append(call, COMP);
 | |
| #ifdef FCC
 | |
| 				concat(call, &CPP_FLAGS);
 | |
| #endif
 | |
| 				concat(call, &COMP_FLAGS);
 | |
| #if FM2 || FCC
 | |
| 				append(call, INCLUDE);
 | |
| #endif
 | |
| 				append(call, file);
 | |
| 				append(call, f);
 | |
| #ifndef ACK_BIN
 | |
| 			}
 | |
| 			else {
 | |
| 				append(call, AS);
 | |
| 				append(call, "-o");
 | |
| 				append(call, f);
 | |
| #ifdef sun
 | |
| 				append(call, "-mc68020");
 | |
| #endif
 | |
| 				append(call, file);
 | |
| 			}
 | |
| #endif
 | |
| 			if (runvec(call, (char *) 0)) {
 | |
| 				file = f;
 | |
| 			}
 | |
| 			else {
 | |
| 				remove(f);
 | |
| 				continue;
 | |
| 			}
 | |
| 			cleanup(tmp_file);
 | |
| 			tmp_file[0] = '\0';
 | |
| 		}
 | |
| 
 | |
| 		else if (file[0] != '-' &&
 | |
| 			 strcmp(ext, "o") && strcmp(ext, "a")) {
 | |
| 			warning("file with unknown suffix (%s) passed to the loader", ext);
 | |
| 		}
 | |
| 
 | |
| 		if (c_flag)
 | |
| 			continue;
 | |
| 
 | |
| 		append(&LDFILES, file);
 | |
| 		if (ldfile) {
 | |
| 			append(&GEN_LDFILES, ldfile);
 | |
| 			ldfile = 0;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/* *.s to a.out */
 | |
| 	if (RET_CODE == 0 && LDFILES.al_argc > 0) {
 | |
| #ifndef ACK_BIN
 | |
| 		if (compile_cnt > 1) {
 | |
| 			printf("Linking:\n");
 | |
| 		}
 | |
| #endif
 | |
| 		init(call);
 | |
| 		append(call, LD);
 | |
| 		concat(call, &LD_HEAD);
 | |
| 		concat(call, &LD_FLAGS);
 | |
| 		append(call, "-o");
 | |
| #if ACK_BIN && !vax
 | |
| 		strcpy(tmp_file, TMP_DIR);
 | |
| 		strcat(tmp_file, "/LF_XXXXXX");
 | |
| 		mktemp(tmp_file);
 | |
| 		append(call, tmp_file);
 | |
| 		concat(call, sect_align);
 | |
| #else ACK_BIN && !vax
 | |
| 		append(call, o_FILE);
 | |
| #endif ACK_BIN && !vax
 | |
| 		concat(call, &LDFILES);
 | |
| 		concat(call, &LD_TAIL);
 | |
| #if ACK_BIN && !vax
 | |
| 		if (! runvec(call, (char *) 0)) {
 | |
| 			cleanup(tmp_file);
 | |
| 			exit(RET_CODE);
 | |
| 		}
 | |
| 		init(call);
 | |
| 		append(call, CV);
 | |
| 		append(call, tmp_file);
 | |
| 		append(call, o_FILE);
 | |
| #endif ACK_BIN && !vax
 | |
| 		if (runvec(call, (char *) 0) && GEN_LDFILES.al_argc == 1)
 | |
| #ifdef ACK_BIN
 | |
| 			;
 | |
| #else
 | |
| 			remove(GEN_LDFILES.al_argv[0]);
 | |
| #endif ACK_BIN
 | |
| #if ACK_BIN && !vax
 | |
| 		cleanup(tmp_file);
 | |
| #endif ACK_BIN && !vax
 | |
| 	}
 | |
| 	exit(RET_CODE);
 | |
| }
 | |
| 
 | |
| needsprep(name)
 | |
| 	char *name;
 | |
| {
 | |
| 	int file;
 | |
| 	char fc;
 | |
| 
 | |
| 	file = open(name,0);
 | |
| 	if (file < 0) return 0;
 | |
| 	if (read(file, &fc, 1) != 1) fc = 0;
 | |
| 	close(file);
 | |
| 	return fc == '#';
 | |
| }
 | |
| 
 | |
| char *
 | |
| alloc(u)
 | |
| 	unsigned u;
 | |
| {
 | |
| 	char *p = malloc(u);
 | |
| 
 | |
| 	if (p == 0)
 | |
| 		panic("no space");
 | |
| 	return p;
 | |
| }
 | |
| 
 | |
| append(al, arg)
 | |
| 	struct arglist *al;
 | |
| 	char *arg;
 | |
| {
 | |
| 	if (!arg || !*arg) return;
 | |
| 	if (al->al_argc >= MAXARGC)
 | |
| 		panic("argument list overflow");
 | |
| 	if (*arg == '~' || *arg == '*') {
 | |
| 		char *p;
 | |
| 		char *lb = *arg == '~' ? LIBDIR : BINDIR;
 | |
| 		p = alloc((unsigned)strlen(ROOT_DIR)+strlen(lb)+strlen(arg+1)+1);
 | |
| 		strcpy(p, ROOT_DIR);
 | |
| 		strcat(p, lb);
 | |
| 		strcat(p, arg+1);
 | |
| 		arg = p;
 | |
| 	}
 | |
| 	al->al_argv[(al->al_argc)++] = arg;
 | |
| }
 | |
| 
 | |
| 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");
 | |
| 	while (i-- > 0) {
 | |
| 		if (**q == '~' || **q == '*') {
 | |
| 			char *lb = **q == '~' ? LIBDIR : BINDIR;
 | |
| 			*p = alloc((unsigned)strlen(ROOT_DIR)+strlen(lb)+strlen(*q+1)+2);
 | |
| 			strcpy(*p, ROOT_DIR);
 | |
| 			strcat(*p, lb);
 | |
| 			strcat(*p++, *q+1);
 | |
| 			q++;
 | |
| 		}
 | |
| 		else	*p++ = *q++;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /*VARARGS*/
 | |
| char *
 | |
| mkstr(va_alist)
 | |
| 	va_dcl
 | |
| {
 | |
| 	va_list ap;
 | |
| 	char *dst;
 | |
| 
 | |
| 	va_start(ap);
 | |
| 	{
 | |
| 		register char *p;
 | |
| 		register char *q;
 | |
| 
 | |
| 		dst = q = va_arg(ap, char *);
 | |
| 		p = va_arg(ap, char *);
 | |
| 
 | |
| 		while (p) {
 | |
| 			while (*q++ = *p++);
 | |
| 			q--;
 | |
| 			p = va_arg(ap, char *);
 | |
| 		}
 | |
| 	}
 | |
| 	va_end(ap);
 | |
| 
 | |
| 	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 >= p2) {
 | |
| 		*p1 = '\0';
 | |
| 		while (*dst++ = *p2++);
 | |
| 		*p1 = '.';
 | |
| 	}
 | |
| 	else
 | |
| 		while (*dst++ = *p2++);
 | |
| }
 | |
| 
 | |
| char *
 | |
| extension(fn)
 | |
| 	char *fn;
 | |
| {
 | |
| 	register char *c = fn;
 | |
| 
 | |
| 	while (*c++) ;
 | |
| 	while (*--c != '.' && c >= fn) { }
 | |
| 	if (c++ < fn || !*c) return fn;
 | |
| 	return c;
 | |
| }
 | |
| 
 | |
| runvec(vec, outp)
 | |
| 	struct arglist *vec;
 | |
| 	char *outp;
 | |
| {
 | |
| 	int pid, status;
 | |
| 
 | |
| 	if (v_flag) {
 | |
| 		pr_vec(vec);
 | |
| 		putc('\n', stderr);
 | |
| 	}
 | |
| 	if ((pid = fork()) == 0) {	/* start up the process */
 | |
| 		if (outp) {	/* redirect standard output	*/
 | |
| 			close(1);
 | |
| 			if (creat(outp, 0666) != 1)
 | |
| 				panic("cannot create output file");
 | |
| 		}
 | |
| 		ex_vec(vec);
 | |
| 	}
 | |
| 	if (pid == -1)
 | |
| 		panic("no more processes");
 | |
| 	kids = pid;
 | |
| 	wait(&status);
 | |
| 	if (status) switch(status & 0177) {
 | |
| 	case SIGHUP:
 | |
| 	case SIGINT:
 | |
| 	case SIGQUIT:
 | |
| 	case SIGTERM:
 | |
| 	case 0:
 | |
| 		break;
 | |
| 	default:
 | |
| 		error("%s died with signal %d\n", vec->al_argv[1], status&0177);
 | |
| 	}
 | |
| 	kids = -1;
 | |
| 	return status ? ((RET_CODE = 1), 0) : 1;
 | |
| }
 | |
| 
 | |
| /*VARARGS1*/
 | |
| error(str, s1, s2)
 | |
| 	char *str, *s1, *s2;
 | |
| {
 | |
| 	fprintf(stderr, "%s: ", ProgCall);
 | |
| 	fprintf(stderr, str, s1, s2);
 | |
| 	putc('\n', stderr);
 | |
| 	ecount++;
 | |
| }
 | |
| 
 | |
| /*VARARGS1*/
 | |
| warning(str, s1, s2)
 | |
| 	char *str, *s1, *s2;
 | |
| {
 | |
| 	fprintf(stderr, "%s: (warning) ", ProgCall);
 | |
| 	fprintf(stderr, str, s1, s2);
 | |
| 	putc('\n', stderr);
 | |
| }
 | |
| 
 | |
| panic(str)
 | |
| 	char *str;
 | |
| {
 | |
| 	error(str);
 | |
| 	trapcc(SIGINT);
 | |
| }
 | |
| 
 | |
| pr_vec(vec)
 | |
| 	register struct arglist *vec;
 | |
| {
 | |
| 	register char **ap = &vec->al_argv[1];
 | |
| 
 | |
| 	vec->al_argv[vec->al_argc] = 0;
 | |
| 	fputs(*ap, stderr);
 | |
| 	while (*++ap) {
 | |
| 		putc(' ', stderr);
 | |
| 		fputs(*ap, stderr);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| extern int errno;
 | |
| 
 | |
| ex_vec(vec)
 | |
| 	register struct arglist *vec;
 | |
| {
 | |
| 	if (noexec)
 | |
| 		exit(0);
 | |
| 	vec->al_argv[vec->al_argc] = 0;
 | |
| 	execv(vec->al_argv[1], &(vec->al_argv[1]));
 | |
| 	if (errno == ENOEXEC) { /* not an a.out, try it with the SHELL */
 | |
| 		vec->al_argv[0] = SHELL;
 | |
| 		execv(SHELL, &(vec->al_argv[0]));
 | |
| 	}
 | |
| 	if (access(vec->al_argv[1], 1) == 0) {
 | |
| 		/* File is executable. */
 | |
| 		error("cannot execute %s", vec->al_argv[1]);
 | |
| 	} else {
 | |
| 		error("%s is not executable", vec->al_argv[1]);
 | |
| 	}
 | |
| 	exit(1);
 | |
| }
 |