1197 lines
23 KiB
C
1197 lines
23 KiB
C
/* fcc/fm2/fpc
|
|
Driver for fast ACK compilers.
|
|
|
|
Derived from the C compiler driver from Minix.
|
|
|
|
Compile this file with
|
|
cc -O -I<ACK home directory>/config -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.
|
|
*/
|
|
|
|
#if FM2+FPC+FCC > 1
|
|
Something wrong here! Only one of FM2, FPC, or FCC must be defined
|
|
#endif
|
|
|
|
|
|
#ifdef i386
|
|
#define MACHNAME "i386"
|
|
#define SYSNAME "i386"
|
|
#endif
|
|
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include <errno.h>
|
|
#include <signal.h>
|
|
#include <stdio.h>
|
|
#include <stdarg.h>
|
|
#include <limits.h>
|
|
#include <stddef.h>
|
|
#include "smap.h"
|
|
#include "stringlist.h"
|
|
#include "em_path.h"
|
|
#include "system.h"
|
|
|
|
#if(CHAR_BIT!=8)
|
|
We do not support machines with non 8-bit characters.
|
|
#endif
|
|
|
|
/* Create em based compiler if otherwise not specified.
|
|
* Size of EM machine depends on current machine. */
|
|
#ifndef MACHNAME
|
|
#if(INT_MAX==32767)
|
|
#define MACHNAME "em22"
|
|
#define SYSNAME "em22"
|
|
#else
|
|
#define MACHNAME "em44"
|
|
#define SYSNAME "em44"
|
|
#endif
|
|
#endif
|
|
|
|
/*
|
|
Version producing ACK .o files in one pass.
|
|
*/
|
|
#define MAXARGC 256 /* maximum number of arguments allowed in a list */
|
|
#define USTR_SIZE FILENAME_MAX /* maximum length of string variable */
|
|
|
|
typedef char USTRING[USTR_SIZE];
|
|
|
|
struct arglist
|
|
{
|
|
int al_argc;
|
|
char *al_argv[MAXARGC];
|
|
};
|
|
|
|
#define CPP_NAME "$H/lib/ack/em_cpp"
|
|
#define LD_NAME "$H/lib/ack/em_ass"
|
|
#define CV_NAME "$H/lib/ack/$S/cv"
|
|
|
|
static char *CPP;
|
|
static char *COMP;
|
|
static char *LD;
|
|
static char *cc = "cc";
|
|
|
|
static int kids = -1;
|
|
static int ecount = 0;
|
|
|
|
|
|
struct size_info
|
|
{
|
|
/* size in bytes */
|
|
int size;
|
|
/* alignment */
|
|
int align;
|
|
};
|
|
|
|
/** Contains information on the different systems supported. */
|
|
struct system_information
|
|
{
|
|
/** Architecture / machine name */
|
|
char* arch;
|
|
/** Platform name */
|
|
char* platform;
|
|
/** Linker path and name */
|
|
char *linker;
|
|
/** Runtime startup file */
|
|
char *startup;
|
|
/** Converter path and name */
|
|
char *cv;
|
|
/** C preprocessor command line arguments. */
|
|
struct arglist cpp_flags;
|
|
/** Compiler flags. */
|
|
struct arglist compiler_flags;
|
|
};
|
|
|
|
const struct system_information machine_info[2] =
|
|
{
|
|
{
|
|
"em22" /* arch */,
|
|
"em22" /* platform */,
|
|
"$H/lib/ack/em_ass" /* linker */,
|
|
#ifdef FCC
|
|
"c-ansi.m" /* startup file */,
|
|
#else
|
|
#ifdef FPC
|
|
"pascal.m" /* startup file */,
|
|
#ifdef FM2
|
|
"modula2.m" /* startup file */,
|
|
#endif
|
|
#endif
|
|
#endif
|
|
NULL /* converter */,
|
|
/** CPP Flags */
|
|
{17,{
|
|
"-DEM_WSIZE=2",
|
|
"-DEM_PSIZE=2",
|
|
"-DEM_SSIZE=2",
|
|
"-DEM_LSIZE=4",
|
|
"-DEM_FSIZE=4",
|
|
"-DEM_DSIZE=8",
|
|
"-DEM_XSIZE=8",
|
|
"-D_EM_WSIZE=2",
|
|
"-D_EM_PSIZE=2",
|
|
"-D_EM_SSIZE=2",
|
|
"-D_EM_LSIZE=4",
|
|
"-D_EM_FSIZE=4",
|
|
"-D_EM_DSIZE=8",
|
|
"-D_EM_XSIZE=8",
|
|
"-Dem22",
|
|
"-D__em22",
|
|
"-D__unix"}},
|
|
/* compiler flags */
|
|
#ifdef FCC
|
|
{1, {
|
|
"-Vw2.2i2.2p2.2f4.2s2.2l4.2d8.2x8.2"
|
|
}}
|
|
#else
|
|
#ifdef FPC
|
|
/* pc Flags */
|
|
{1, {
|
|
"-Vw2.2i2.2l4.2p2.2f8.2S2.2"
|
|
}}
|
|
#else
|
|
#ifdef FM2
|
|
/* modula-2 Flags */
|
|
{1, {
|
|
"-Vw2.2i2.2l4.2p2.2f8.2S2.2"
|
|
}}
|
|
#endif
|
|
#endif
|
|
#endif
|
|
},
|
|
{
|
|
"em24" /* arch */,
|
|
"em24" /* platform */,
|
|
"$H/lib/ack/em_ass" /* linker */,
|
|
#ifdef FCC
|
|
"c-ansi.m" /* startup file */,
|
|
#else
|
|
#ifdef FPC
|
|
"pascal.m" /* startup file */,
|
|
#ifdef FM2
|
|
"modula2.m" /* startup file */,
|
|
#endif
|
|
#endif
|
|
#endif
|
|
NULL /* converter */,
|
|
/** CPP Flags */
|
|
{17,{
|
|
"-DEM_WSIZE=2",
|
|
"-DEM_PSIZE=4",
|
|
"-DEM_SSIZE=2",
|
|
"-DEM_LSIZE=4",
|
|
"-DEM_FSIZE=4",
|
|
"-DEM_DSIZE=8",
|
|
"-DEM_XSIZE=8",
|
|
"-D_EM_WSIZE=2",
|
|
"-D_EM_PSIZE=4",
|
|
"-D_EM_SSIZE=2",
|
|
"-D_EM_LSIZE=4",
|
|
"-D_EM_FSIZE=4",
|
|
"-D_EM_DSIZE=8",
|
|
"-D_EM_XSIZE=8",
|
|
"-Dem24",
|
|
"-D__em24",
|
|
"-D__unix"}},
|
|
/* compiler flags */
|
|
#ifdef FCC
|
|
{1, {
|
|
"-Vw2.2i2.2p4.2f4.2s2.2l4.2d8.2x8.2"
|
|
}}
|
|
#else
|
|
#ifdef FPC
|
|
/* pc Flags */
|
|
{1, {
|
|
"-Vw2.2i2.2l4.2p4.2f8.2S2.2"
|
|
}}
|
|
#else
|
|
#ifdef FM2
|
|
/* modula-2 Flags */
|
|
{1, {
|
|
"-Vw2.2i2.2l4.2p4.2f8.2S2.2"
|
|
}}
|
|
#endif
|
|
#endif
|
|
#endif
|
|
}
|
|
};
|
|
|
|
|
|
struct arglist CPP_FLAGS = {0,{NULL}};
|
|
|
|
|
|
struct arglist LD_HEAD =
|
|
{ 2,
|
|
{ "$H/lib/$S/head_em",
|
|
#ifdef FCC
|
|
"$H/lib/$S/head_$A"
|
|
#endif
|
|
#ifdef FM2
|
|
"$H/lib/$S/head_m2"
|
|
#endif
|
|
#ifdef FPC
|
|
"$H/lib/$S/head_pc"
|
|
#endif
|
|
}};
|
|
|
|
|
|
struct arglist align =
|
|
{ 5,
|
|
{
|
|
#ifdef sun3
|
|
"-a0:4",
|
|
"-a1:4",
|
|
"-a2:0x20000",
|
|
"-a3:4",
|
|
"-b0:0x2020"
|
|
#endif
|
|
#ifdef vax4
|
|
"-a0:4",
|
|
"-a1:4",
|
|
"-a2:0x400",
|
|
"-a3:4",
|
|
"-b0:0"
|
|
#endif
|
|
#ifdef i386
|
|
"-a0:4",
|
|
"-a1:4",
|
|
"-a2:4",
|
|
"-a3:4",
|
|
"-b1:0x1880000"
|
|
#endif
|
|
} };
|
|
|
|
struct arglist COMP_FLAGS;
|
|
|
|
static char *o_FILE = "a.out"; /* default name for executable file */
|
|
|
|
|
|
#define cleanup(str) (str && str[0] && remove(str))
|
|
#define init(al) ((al)->al_argc = 0)
|
|
|
|
static char ProgCall[FILENAME_MAX];
|
|
|
|
static struct arglist SRCFILES;
|
|
static struct arglist LDFILES;
|
|
|
|
static int RET_CODE = 0;
|
|
|
|
/* The environment variable pointing to ACK_HOME */
|
|
static char* ackhome = 0;
|
|
/* The environment variable pointing to the temporary directory. */
|
|
static char* tmpdir = 0;
|
|
static struct arglist LD_FLAGS;
|
|
|
|
static struct arglist CALL_VEC;
|
|
|
|
static int o_flag = 0;
|
|
static int c_flag = 0;
|
|
static int g_flag = 0;
|
|
static int v_flag = 0;
|
|
static int O_flag = 0;
|
|
/* Only ANSI C is now supported. */
|
|
static int ansi_c = 1;
|
|
static int cv_flag = 0;
|
|
|
|
char *mkstr(char *, ...);
|
|
static char *alloc(unsigned int);
|
|
static char *extension(char *);
|
|
static char *expand_string(char *s, struct system_information *);
|
|
static void error(char *, char *, char *);
|
|
static void warning(char *, char *, char *);
|
|
static void panic(char *);
|
|
static void append(register struct arglist *, char *);
|
|
static void expand(register struct arglist *, struct system_information *);
|
|
static void concat(struct arglist *, struct arglist *);
|
|
static int runvec(struct arglist *, char *);
|
|
static int needsprep(char *);
|
|
|
|
static USTRING ofile;
|
|
static USTRING BASE;
|
|
static char tmp_file[L_tmpnam];
|
|
|
|
int noexec = 0;
|
|
|
|
void trapcc(int sig)
|
|
{
|
|
cleanup(ofile);
|
|
cleanup(tmp_file);
|
|
if (ackhome != 0)
|
|
{
|
|
free(ackhome);
|
|
}
|
|
if (tmpdir != 0)
|
|
{
|
|
free(tmpdir);
|
|
}
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
|
|
#ifdef FCC
|
|
#define lang_suffix() "c"
|
|
#define comp_name() "$H/lib/ack/em_cemcom.ansi"
|
|
#define ansi_c_name() "$H/lib/ack/em_cemcom.ansi"
|
|
#endif /* FCC */
|
|
|
|
#ifdef FM2
|
|
#define lang_suffix() "mod"
|
|
#define comp_name() "$H/lib/ack/em_m2"
|
|
#endif /* FM2 */
|
|
|
|
#ifdef FPC
|
|
#define lang_suffix() "p"
|
|
#define comp_name() "$H/lib/ack/em_pc"
|
|
#endif /* FPC */
|
|
|
|
/** Default library directories to search in. */
|
|
|
|
#define LIB_DIR_1 "$H/share/ack/$S"
|
|
#define LIB_DIR_2 "$H/lib/ack/plat/$S"
|
|
|
|
|
|
#ifdef FCC
|
|
int lang_opt(char *str)
|
|
{
|
|
switch (str[1])
|
|
{
|
|
case 'R':
|
|
if (!ansi_c)
|
|
{
|
|
append(&COMP_FLAGS, str);
|
|
return 1;
|
|
}
|
|
break;
|
|
case '-': /* debug options */
|
|
append(&COMP_FLAGS, str);
|
|
return 1;
|
|
case 'w': /* disable warnings */
|
|
if (!ansi_c)
|
|
{
|
|
append(&COMP_FLAGS, str);
|
|
return 1;
|
|
}
|
|
if (str[2])
|
|
{
|
|
str[1] = '-';
|
|
append(&COMP_FLAGS, &str[1]);
|
|
}
|
|
else
|
|
append(&COMP_FLAGS, "-a");
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
#endif /* FCC */
|
|
|
|
#ifdef FM2
|
|
int lang_opt(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(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 */
|
|
|
|
char* stringdup(const char* s)
|
|
{
|
|
char *p;
|
|
if (s == NULL)
|
|
return NULL;
|
|
p = alloc(strlen(s) + sizeof(char));
|
|
strcpy(p, s);
|
|
return p;
|
|
}
|
|
|
|
char* getackhome(void)
|
|
{
|
|
char *value = getenv("ACK_HOME");
|
|
if (value == NULL)
|
|
{
|
|
#ifndef EM_DIR
|
|
panic("ACK_HOME must be set.");
|
|
#else
|
|
return EM_DIR;
|
|
#endif
|
|
}
|
|
return value;
|
|
}
|
|
|
|
|
|
#define LIB_PREFIX "lib"
|
|
#define LIB_SUFFIX ".a"
|
|
|
|
/** From a library library directory list,
|
|
* search for the libraries that exist, and return
|
|
* the full path to the directory where the
|
|
* library is located.
|
|
*
|
|
* The search starts from the first added directory
|
|
* item at the command line.
|
|
*
|
|
* @param[in] dirs The directory list to search in.
|
|
* @param[in] lib The library name to search for.
|
|
* @returns The full path specification, allocated in the heap, it must
|
|
* be freed by the caller.
|
|
*/
|
|
char* search_library_path(struct stringlist *dirs, char* lib)
|
|
{
|
|
char fname[FILENAME_MAX];
|
|
FILE *fd;
|
|
int len;
|
|
int dirs_count;
|
|
int index = stringlist_count(dirs)-1;
|
|
|
|
dirs_count = stringlist_count(dirs);
|
|
for (index = 0; index < dirs_count; index++)
|
|
{
|
|
strcpy(fname, stringlist_get(dirs,index));
|
|
len = strlen(fname);
|
|
/* len-1 is the NULL character */
|
|
if (fname[len-1] != '/')
|
|
{
|
|
/* Add trailing slash */
|
|
fname[len] = '/';
|
|
fname[len+1] = 0;
|
|
len++;
|
|
}
|
|
fname[len] = 0;
|
|
strcat(fname,lib);
|
|
fd = fopen(fname,"rb");
|
|
if (fd!=NULL)
|
|
{
|
|
/* Return the path! */
|
|
fclose(fd);
|
|
return stringdup(fname);
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
/* Contains the directories being searched for libraries */
|
|
struct stringlist library_dirs;
|
|
/* Contains the library lists */
|
|
struct stringlist libraries;
|
|
/* Contains the directory paths. */
|
|
struct smap paths;
|
|
/* Contains the src file list */
|
|
struct stringlist srcfiles;
|
|
char *str;
|
|
char *startup_file;
|
|
char **argvec;
|
|
int count;
|
|
int index;
|
|
int libs_count;
|
|
char *ext;
|
|
FILE* fd;
|
|
register struct arglist *call = &CALL_VEC;
|
|
char tmpbuffer[256];
|
|
char *file;
|
|
char *ldfile;
|
|
char *INCLUDE = NULL;
|
|
int compile_cnt = 0;
|
|
struct system_information* sys_info;
|
|
|
|
startup_file = NULL;
|
|
sys_info = &machine_info[0];
|
|
strcpy(tmpbuffer, LIB_PREFIX);
|
|
stringlist_init(&library_dirs);
|
|
stringlist_init(&libraries);
|
|
stringlist_init(&srcfiles);
|
|
smap_init(&paths);
|
|
|
|
ackhome = stringdup(getackhome());
|
|
if (ackhome == NULL)
|
|
{
|
|
panic("ACK_HOME Environment variable is not set.");
|
|
}
|
|
tmpdir = stringdup(sys_gettmpdir());
|
|
if (tmpdir == NULL)
|
|
{
|
|
panic("TMPDIR Environment variable is not set.");
|
|
}
|
|
|
|
setbuf(stdout, (char *) 0);
|
|
/* get basebame of application. */
|
|
sys_basename(*argv++, ProgCall);
|
|
|
|
/* get compiler to use. */
|
|
COMP = expand_string(comp_name(),sys_info);
|
|
/* get c pre-processor to use */
|
|
CPP = expand_string(CPP_NAME,sys_info);
|
|
/** get linker to use */
|
|
LD = expand_string(sys_info->linker,sys_info);
|
|
|
|
/* Add system directory path. */
|
|
stringlist_add(&library_dirs,expand_string(LIB_DIR_1,sys_info));
|
|
stringlist_add(&library_dirs,expand_string(LIB_DIR_2,sys_info));
|
|
|
|
|
|
|
|
if (signal(SIGINT, SIG_IGN) != SIG_IGN)
|
|
signal(SIGINT, trapcc);
|
|
if (signal(SIGTERM, SIG_IGN) != SIG_IGN)
|
|
signal(SIGTERM, trapcc);
|
|
while (--argc > 0)
|
|
{
|
|
if (*(str = *argv++) != '-')
|
|
{
|
|
stringlist_add(&srcfiles,stringdup(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 'g': /* debugger support */
|
|
append(&COMP_FLAGS, str);
|
|
g_flag = 1;
|
|
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()))
|
|
{
|
|
error("-o would overwrite %s", o_FILE, NULL);
|
|
}
|
|
}
|
|
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 */
|
|
O_flag = 1;
|
|
break;
|
|
case 'v': /* verbose */
|
|
v_flag++;
|
|
if (str[2] == 'n')
|
|
noexec = 1;
|
|
break;
|
|
case 'L':
|
|
stringlist_add(&library_dirs,stringdup(str+2));
|
|
break;
|
|
case 'l': /* library file */
|
|
stringlist_add(&srcfiles,stringdup(str));
|
|
break;
|
|
case 'M': /* use other compiler (for testing) */
|
|
free(COMP);
|
|
COMP = alloc(strlen(str)+2-1);
|
|
strcpy(COMP, str + 2);
|
|
break;
|
|
case 'P': /* use other cpp (for testing) */
|
|
free(CPP);
|
|
CPP = alloc(strlen(str)+2-1);
|
|
strcpy(CPP, str + 2);
|
|
break;
|
|
case 'X': /* use other linker (for testing) */
|
|
LD = alloc(strlen(str)+2-1);
|
|
strcpy(LD, str + 2);
|
|
break;
|
|
case 's': /* strip */
|
|
if (str[2] == '\0')
|
|
{
|
|
append(&LD_FLAGS, str);
|
|
break;
|
|
}
|
|
/* fall through */
|
|
default:
|
|
warning("%s flag ignored", str, NULL);
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* Add C preprocessor flags. */
|
|
for (count = 0; count < sys_info->cpp_flags.al_argc; count++)
|
|
{
|
|
append(&CPP_FLAGS, sys_info->cpp_flags.al_argv[count]);
|
|
}
|
|
/* Add compiler flags. */
|
|
for (count = 0; count < sys_info->compiler_flags.al_argc; count++)
|
|
{
|
|
append(&COMP_FLAGS, sys_info->compiler_flags.al_argv[count]);
|
|
}
|
|
#ifdef FPC
|
|
append(&COMP_FLAGS, "-d");
|
|
#endif
|
|
|
|
/* Now for each srcfile, if it starts with -l then replace it with the correct real
|
|
* path immediately.
|
|
*/
|
|
count = stringlist_count(&srcfiles);
|
|
for (index = 0; index < count; index++)
|
|
{
|
|
str = stringlist_get(&srcfiles,index);
|
|
/* -l parameter */
|
|
if (*str == '-')
|
|
{
|
|
/* Search for the directory where this library might be located. */
|
|
strcpy(tmpbuffer,LIB_PREFIX);
|
|
strcat(tmpbuffer,str+2);
|
|
strcat(tmpbuffer,LIB_SUFFIX);
|
|
str = search_library_path(&library_dirs,tmpbuffer);
|
|
if (str != NULL)
|
|
{
|
|
stringlist_add(&srcfiles,str);
|
|
append(&SRCFILES,str);
|
|
}
|
|
} else
|
|
{
|
|
append(&SRCFILES,str);
|
|
}
|
|
}
|
|
|
|
|
|
if (ecount)
|
|
{
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
|
|
count = SRCFILES.al_argc;
|
|
argvec = &(SRCFILES.al_argv[0]);
|
|
while (count-- > 0)
|
|
{
|
|
ext = extension(*argvec);
|
|
if (*argvec[0] != '-' && ext != *argvec++
|
|
&& (!strcmp(ext, lang_suffix())))
|
|
{
|
|
compile_cnt++;
|
|
}
|
|
}
|
|
|
|
if (compile_cnt > 1 && c_flag && o_flag)
|
|
{
|
|
warning("-o flag ignored", NULL, NULL);
|
|
o_flag = 0;
|
|
}
|
|
|
|
#ifdef FM2
|
|
INCLUDE = expand_string("-I$H/lib/m2",sys_info);
|
|
#endif /* FM2 */
|
|
#ifdef FCC
|
|
/* INCLUDE = expand_string(
|
|
ansi_c ? "-I$H/include/tail_ac" : "-I$H/include/_tail_cc"); */
|
|
INCLUDE = expand_string("-I$H/share/ack/include",sys_info);
|
|
/* append(&COMP_FLAGS, "-L");*/
|
|
#endif /* FCC */
|
|
count = SRCFILES.al_argc;
|
|
argvec = &(SRCFILES.al_argv[0]);
|
|
|
|
/* For argument file */
|
|
while (count-- > 0)
|
|
{
|
|
register char *f;
|
|
sys_basename(file = *argvec++, BASE);
|
|
|
|
ext = extension(file);
|
|
|
|
/* if not standard input and file is equal to the supported language suffix,
|
|
* then compile it */
|
|
if (file[0] != '-' && ext != file && (!strcmp(ext, lang_suffix())))
|
|
{
|
|
if (compile_cnt > 1)
|
|
printf("%s\n", file);
|
|
|
|
ldfile = c_flag ? ofile : alloc((unsigned) strlen(BASE) + 3);
|
|
|
|
/****************************************************
|
|
* Run C preprocessor on the source files and save
|
|
* result in temporary output file.
|
|
****************************************************/
|
|
if (
|
|
#ifdef FCC
|
|
/* (!strcmp(ext, "s")) && */
|
|
#endif
|
|
needsprep(file))
|
|
{
|
|
if (sys_tmpnam(tmp_file)==NULL)
|
|
{
|
|
panic("Cannot get temporary filename.");
|
|
}
|
|
fd = fopen(tmp_file,"w+");
|
|
if (fd==NULL)
|
|
{
|
|
fclose(fd);
|
|
panic("Cannot write temporary file.");
|
|
}
|
|
init(call);
|
|
append(call, CPP);
|
|
concat(call, &CPP_FLAGS);
|
|
append(call, INCLUDE);
|
|
append(call, file);
|
|
if (runvec(call, tmp_file)==EXIT_SUCCESS)
|
|
{
|
|
/* The input file is now the preprocessor
|
|
* output file.
|
|
*/
|
|
file = tmp_file;
|
|
}
|
|
else
|
|
{
|
|
remove(tmp_file);
|
|
tmp_file[0] = '\0';
|
|
continue;
|
|
}
|
|
}
|
|
|
|
/****************************************************
|
|
* Compile the source file.
|
|
****************************************************/
|
|
init(call);
|
|
if (o_flag && c_flag)
|
|
{
|
|
f = o_FILE;
|
|
}
|
|
else
|
|
{
|
|
f = mkstr(ldfile, BASE, ".", "o", (char *) 0);
|
|
}
|
|
append(call, COMP);
|
|
concat(call, &COMP_FLAGS);
|
|
#ifdef FM2
|
|
append(call, INCLUDE);
|
|
#endif
|
|
append(call, file);
|
|
append(call, f);
|
|
if (runvec(call, (char *) 0)==EXIT_SUCCESS)
|
|
{
|
|
file = f;
|
|
}
|
|
else
|
|
{
|
|
remove(f);
|
|
continue;
|
|
}
|
|
cleanup(tmp_file);
|
|
tmp_file[0] = '\0';
|
|
} /* endif compiling source file. */
|
|
|
|
|
|
else if (file[0] != '-' && strcmp(ext, "o") && strcmp(ext, "a"))
|
|
{
|
|
warning("file with unknown suffix (%s) passed to the loader", ext,
|
|
NULL);
|
|
}
|
|
|
|
if (c_flag)
|
|
continue;
|
|
|
|
/* Add the output object file to the list of files to link. */
|
|
append(&LDFILES, file);
|
|
}
|
|
|
|
|
|
/* See if we need to convert the flags. */
|
|
cv_flag = sys_info->cv!=NULL;
|
|
|
|
/* *.s to a.out */
|
|
if (RET_CODE == 0 && LDFILES.al_argc > 0)
|
|
{
|
|
init(call);
|
|
/* expand(&LD_HEAD);
|
|
// cc = "cc.2g";
|
|
expand(&LD_TAIL);*/
|
|
append(call, LD);
|
|
/* concat(call, &align);*/
|
|
append(call, "-p");
|
|
append(call, "-sx");
|
|
append(call, "-o");
|
|
|
|
if (cv_flag)
|
|
{
|
|
if (sys_tmpnam(tmp_file)==NULL)
|
|
{
|
|
panic("Cannot get temporary filename.");
|
|
}
|
|
append(call, tmp_file);
|
|
} else
|
|
{
|
|
append(call, o_FILE);
|
|
}
|
|
/* concat(call, &LD_HEAD);
|
|
concat(call, &LD_FLAGS);*/
|
|
|
|
if (sys_info->startup!=NULL)
|
|
{
|
|
startup_file = search_library_path(&library_dirs,sys_info->startup);
|
|
if (startup_file!=NULL)
|
|
{
|
|
append(call,startup_file);
|
|
} else
|
|
{
|
|
panic("Cannot find startup file.");
|
|
}
|
|
}
|
|
concat(call, &LDFILES);
|
|
/* if (g_flag)
|
|
append(call, expand_string("$H/lib/$M/tail_db"));
|
|
#ifdef FCC
|
|
if (!ansi_c)
|
|
append(call, expand_string("$H/lib/$S/tail_cc.1s"));
|
|
#endif
|
|
concat(call, &LD_TAIL);*/
|
|
if (runvec(call, (char *) 0)==EXIT_FAILURE)
|
|
{
|
|
cleanup(tmp_file);
|
|
exit(RET_CODE);
|
|
}
|
|
|
|
/* Check if we need to convert the file to
|
|
* a standard executable file for the specific
|
|
* target.
|
|
*
|
|
*/
|
|
if (cv_flag)
|
|
{
|
|
init(call);
|
|
append(call, expand_string(sys_info->cv, sys_info));
|
|
append(call, tmp_file);
|
|
append(call, o_FILE);
|
|
if (runvec(call, (char *) 0)==EXIT_FAILURE)
|
|
{
|
|
cleanup(tmp_file);
|
|
exit(RET_CODE);
|
|
}
|
|
cleanup(tmp_file);
|
|
}
|
|
}
|
|
if ((LDFILES.al_argc == 0) && (SRCFILES.al_argc==0))
|
|
{
|
|
panic("No input source files or input object files specified.");
|
|
}
|
|
stringlist_free(&library_dirs,1);
|
|
stringlist_free(&libraries,1);
|
|
if (startup_file!=NULL)
|
|
free(startup_file);
|
|
smap_free(&paths,1,1);
|
|
exit(RET_CODE);
|
|
}
|
|
|
|
static int needsprep(char *name)
|
|
{
|
|
/* FILE *file;
|
|
char fc;
|
|
|
|
file = fopen(name, "r");
|
|
if (file == 0)
|
|
return 0;
|
|
if (fread(file, &fc, 1) != 1)
|
|
fc = 0;
|
|
fclose(file);
|
|
return fc == '#';*/
|
|
return 1;
|
|
}
|
|
|
|
static char * alloc(unsigned int u)
|
|
{
|
|
char *p = malloc(u);
|
|
|
|
if (p == NULL)
|
|
panic("no space");
|
|
return p;
|
|
}
|
|
|
|
static char * expand_string(char *s, struct system_information *sysinfo)
|
|
{
|
|
char buf[1024];
|
|
register char *p = s;
|
|
register char *q = &buf[0];
|
|
int expanded = 0;
|
|
|
|
if (!p)
|
|
return p;
|
|
while (*p)
|
|
{
|
|
if (*p == '$')
|
|
{
|
|
p++;
|
|
expanded = 1;
|
|
switch (*p++)
|
|
{
|
|
case 'A':
|
|
if (ansi_c)
|
|
strcpy(q, "ac");
|
|
else
|
|
strcpy(q, cc);
|
|
break;
|
|
case 'H':
|
|
strcpy(q, ackhome);
|
|
break;
|
|
case 'M':
|
|
strcpy(q, sysinfo->arch);
|
|
break;
|
|
case 'S':
|
|
strcpy(q, sysinfo->platform);
|
|
break;
|
|
default:
|
|
panic("internal error");
|
|
break;
|
|
}
|
|
while (*q)
|
|
q++;
|
|
}
|
|
else
|
|
*q++ = *p++;
|
|
}
|
|
if (!expanded)
|
|
return s;
|
|
*q++ = '\0';
|
|
/* Do not forget the missing null character. */
|
|
p = alloc((unsigned int) (q - buf));
|
|
return strcpy(p, buf);
|
|
}
|
|
|
|
static void append(register struct arglist *al, char *arg)
|
|
{
|
|
if (!arg || !*arg)
|
|
return;
|
|
if (al->al_argc >= MAXARGC)
|
|
panic("argument list overflow");
|
|
al->al_argv[(al->al_argc)++] = arg;
|
|
}
|
|
|
|
static void expand(register struct arglist *al, struct system_information *sysinfo)
|
|
{
|
|
register int i = al->al_argc;
|
|
register char **p = &(al->al_argv[0]);
|
|
|
|
while (i-- > 0)
|
|
{
|
|
*p = expand_string(*p,sysinfo);
|
|
p++;
|
|
}
|
|
}
|
|
|
|
static void concat(struct arglist *al1, struct arglist *al2)
|
|
{
|
|
register int 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)
|
|
{
|
|
*p++ = *q++;
|
|
}
|
|
}
|
|
|
|
char *mkstr(char *dst, ...)
|
|
{
|
|
va_list ap;
|
|
|
|
va_start(ap, dst);
|
|
{
|
|
register char *p;
|
|
register char *q;
|
|
|
|
q = dst;
|
|
p = va_arg(ap, char *);
|
|
|
|
while (p)
|
|
{
|
|
while ((*q++ = *p++) != 0)
|
|
;
|
|
q--;
|
|
p = va_arg(ap, char *);
|
|
}
|
|
}
|
|
va_end(ap);
|
|
|
|
return dst;
|
|
}
|
|
|
|
static char *extension(char *fn)
|
|
{
|
|
register char *c = fn;
|
|
|
|
while (*c++)
|
|
;
|
|
while (*--c != '.' && c >= fn)
|
|
{
|
|
}
|
|
if (c++ < fn || !*c)
|
|
return fn;
|
|
return c;
|
|
}
|
|
|
|
/* Converts an argument structure to a string.
|
|
* The memory allocated should be freed by
|
|
* the caller.
|
|
* */
|
|
char *arg2str(struct arglist *vec)
|
|
{
|
|
int i;
|
|
unsigned int length;
|
|
char *p;
|
|
|
|
length = 0;
|
|
for (i = 0; i < vec->al_argc; i++)
|
|
{
|
|
length += strlen(vec->al_argv[i]);
|
|
}
|
|
/* Add space character */
|
|
length += vec->al_argc * sizeof(char);
|
|
p = alloc(length + sizeof(char));
|
|
*p = '\0';
|
|
for (i = 0; i < vec->al_argc; i++)
|
|
{
|
|
strcat(p, vec->al_argv[i]);
|
|
strcat(p, " ");
|
|
}
|
|
return p;
|
|
}
|
|
|
|
/* Run the system command, and outputs to "outp" if not
|
|
* NULL.
|
|
*/
|
|
static int runvec(struct arglist *vec, char *outp)
|
|
{
|
|
int status;
|
|
char *p;
|
|
char *redirect;
|
|
|
|
redirect = NULL;
|
|
|
|
/* Check if a command interpreter is available. */
|
|
if (system(NULL) == 0)
|
|
{
|
|
panic("No command interpreter available!");
|
|
}
|
|
if(outp != NULL)
|
|
{
|
|
/* Don't forget the null character. */
|
|
redirect = alloc(strlen(outp)+sizeof(char)*3);
|
|
strcpy(redirect,"1>");
|
|
strcat(redirect,outp);
|
|
append(vec,redirect);
|
|
}
|
|
p = arg2str(vec);
|
|
if ((v_flag) && (p != NULL))
|
|
{
|
|
fprintf(stdout,"Executing : %s\n",p);
|
|
}
|
|
|
|
status = system(p);
|
|
free(p);
|
|
if (outp != NULL)
|
|
{
|
|
free(redirect);
|
|
}
|
|
if (status == 0)
|
|
{
|
|
RET_CODE = EXIT_SUCCESS;
|
|
}
|
|
else
|
|
{
|
|
RET_CODE = EXIT_FAILURE;
|
|
}
|
|
return status;
|
|
}
|
|
|
|
/*VARARGS1*/
|
|
static void error(char *str, char *s1, char *s2)
|
|
{
|
|
fprintf(stderr, "%s: ", ProgCall);
|
|
fprintf(stderr, str, s1, s2);
|
|
putc('\n', stderr);
|
|
ecount++;
|
|
}
|
|
|
|
/*VARARGS1*/
|
|
static void warning(char *str, char *s1, char *s2)
|
|
{
|
|
fprintf(stderr, "%s: (warning) ", ProgCall);
|
|
fprintf(stderr, str, s1, s2);
|
|
putc('\n', stderr);
|
|
}
|
|
|
|
static void panic(char *str)
|
|
{
|
|
error(str, NULL, NULL);
|
|
trapcc(SIGINT);
|
|
}
|
|
|