*** empty log message ***
This commit is contained in:
539
mach/proto/as/comm4.c
Normal file
539
mach/proto/as/comm4.c
Normal file
@@ -0,0 +1,539 @@
|
||||
/* @(#)comm4.c 1.6 */
|
||||
/*
|
||||
* Micro processor assembler framework written by
|
||||
* Johan Stevenson, Vrije Universiteit, Amsterdam
|
||||
* modified by
|
||||
* Johan Stevenson, Han Schaminee and Hans de Vries
|
||||
* Philips S&I, T&M, PMDS, Eindhoven
|
||||
*/
|
||||
|
||||
#include "comm0.h"
|
||||
#include "comm1.h"
|
||||
#include "y.tab.h"
|
||||
|
||||
extern YYSTYPE yylval;
|
||||
|
||||
/* ========== Machine independent C routines ========== */
|
||||
|
||||
stop() {
|
||||
#if DEBUG < 2
|
||||
unlink(temppath);
|
||||
#ifdef LISTING
|
||||
unlink(listpath);
|
||||
#endif
|
||||
#endif
|
||||
exit(nerrors != 0);
|
||||
}
|
||||
|
||||
main(argc, argv)
|
||||
char **argv;
|
||||
{
|
||||
register char *p;
|
||||
register i;
|
||||
static char sigs[] = {
|
||||
SIGHUP, SIGINT, SIGQUIT, SIGTERM, 0
|
||||
};
|
||||
|
||||
/* this test should be performed by the
|
||||
* preprocessor, but it cannot
|
||||
*/
|
||||
if ((S_ETC|S_COM|S_VAR|S_DOT) != S_ETC)
|
||||
fatal("incorrect type bits");
|
||||
|
||||
progname = *argv++; argc--;
|
||||
for (p = sigs; i = *p++; )
|
||||
if (signal(i, SIG_IGN) != SIG_IGN)
|
||||
signal(i, stop);
|
||||
for (i = 0; i < argc; i++) {
|
||||
p = argv[i];
|
||||
if (*p++ != '-')
|
||||
continue;
|
||||
switch (*p++) {
|
||||
case 'o':
|
||||
if (*p != NULL) {
|
||||
aoutpath = p;
|
||||
break;
|
||||
}
|
||||
argv[i] = 0;
|
||||
if (++i >= argc)
|
||||
fatal("-o needs filename");
|
||||
aoutpath = argv[i];
|
||||
break;
|
||||
case 'd':
|
||||
#ifdef LISTING
|
||||
dflag = 0;
|
||||
while (*p >= '0' && *p <= '7')
|
||||
dflag = (dflag << 3) + *p++ - '0';
|
||||
if ((dflag & 0777) == 0)
|
||||
dflag |= 0700;
|
||||
dflag &= ~4;
|
||||
#endif
|
||||
break;
|
||||
case 's':
|
||||
sflag = 0;
|
||||
while (*p >= '0' && *p <= '7')
|
||||
sflag = (sflag << 3) + *p++ - '0';
|
||||
break;
|
||||
case 'r':
|
||||
#ifdef RELOCATION
|
||||
#ifdef ASLD
|
||||
rflag = 1;
|
||||
#endif ASLD
|
||||
#endif RELOCATION
|
||||
break;
|
||||
case 'b':
|
||||
#ifdef THREE_PASS
|
||||
bflag = 1;
|
||||
#endif
|
||||
break;
|
||||
#ifndef ASLD
|
||||
case 'u':
|
||||
case '\0':
|
||||
uflag = 1;
|
||||
break;
|
||||
#endif
|
||||
default:
|
||||
continue;
|
||||
}
|
||||
argv[i] = 0;
|
||||
}
|
||||
#ifdef RELOCATION
|
||||
if (rflag)
|
||||
sflag |= SYM_SCT;
|
||||
#endif RELOCATION
|
||||
pass_1(argc, argv);
|
||||
#ifdef THREE_PASS
|
||||
pass_23(PASS_2);
|
||||
#endif
|
||||
pass_23(PASS_3);
|
||||
stop();
|
||||
}
|
||||
|
||||
/* ---------- pass 1: arguments, modules, archives ---------- */
|
||||
|
||||
pass_1(argc, argv)
|
||||
char **argv;
|
||||
{
|
||||
register i;
|
||||
register char *p;
|
||||
register item_t *ip;
|
||||
#ifdef ASLD
|
||||
char armagic[SZMAGIC];
|
||||
#else
|
||||
register nfile = 0;
|
||||
#endif
|
||||
|
||||
tempfile = fftemp(temppath, "asTXXXXXX");
|
||||
#ifdef LISTING
|
||||
listmode = dflag;
|
||||
if (listmode & 0440)
|
||||
listfile = fftemp(listpath, "asLXXXXXX");
|
||||
#endif
|
||||
for (ip = keytab; ip->i_type; ip++)
|
||||
item_insert(ip, H_KEY+hash(ip->i_name));
|
||||
machstart(PASS_1);
|
||||
while (--argc >= 0) {
|
||||
p = *argv++;
|
||||
if (p == 0)
|
||||
continue;
|
||||
#ifdef ASLD
|
||||
if (p[0] == '-' && p[1] == '\0') {
|
||||
input = stdin;
|
||||
parse("STDIN");
|
||||
continue;
|
||||
}
|
||||
#else
|
||||
if (nfile != 0)
|
||||
fatal("second source file %s", p);
|
||||
nfile++;
|
||||
#endif
|
||||
if ((input = fopen(p, "r")) == NULL)
|
||||
fatal("can't open %s", p);
|
||||
#ifdef ASLD
|
||||
if (
|
||||
fread(armagic, SZMAGIC, 1, input) == 1
|
||||
&&
|
||||
strncmp(armagic, ARMAGIC, SZMAGIC) == 0
|
||||
) {
|
||||
archive();
|
||||
fclose(input);
|
||||
continue;
|
||||
}
|
||||
rewind(input);
|
||||
#endif
|
||||
parse(p);
|
||||
fclose(input);
|
||||
}
|
||||
commfinish();
|
||||
machfinish(PASS_1);
|
||||
#ifdef ASLD
|
||||
if (unresolved) {
|
||||
nerrors++;
|
||||
fflush(stdout);
|
||||
fprintf(stderr, "unresolved references:\n");
|
||||
for (i = 0; i < H_SIZE; i++) {
|
||||
ip = hashtab[H_GLOBAL+i];
|
||||
while (ip != 0) {
|
||||
if ((ip->i_type & (S_EXT|S_TYP)) == (S_EXT|S_UND))
|
||||
fprintf(stderr, "\t%s\n", ip->i_name);
|
||||
ip = ip->i_next;
|
||||
}
|
||||
}
|
||||
}
|
||||
#else
|
||||
if (unresolved)
|
||||
outhead.oh_flags |= HF_LINK;
|
||||
if (nfile == 0)
|
||||
fatal("no source file");
|
||||
#endif
|
||||
}
|
||||
|
||||
#ifdef ASLD
|
||||
archive()
|
||||
{
|
||||
register long offset;
|
||||
register i;
|
||||
register long modsize;
|
||||
char modhead[SZMHEAD];
|
||||
|
||||
archmode++;
|
||||
offset = SZMAGIC;
|
||||
for (;;) {
|
||||
if (unresolved == 0)
|
||||
break;
|
||||
fseek(input, offset, 0);
|
||||
if (fread(modhead, SZMHEAD, 1, input) != 1)
|
||||
break;
|
||||
if (
|
||||
strncmp(&modhead[OFF_BEG], STR_BEG, LEN_BEG)
|
||||
||
|
||||
strncmp(&modhead[OFF_END], STR_END, LEN_END)
|
||||
)
|
||||
fatal("bad archive");
|
||||
offset += SZMHEAD;
|
||||
modsize = atol(&modhead[OFF_SIZ]);
|
||||
archsize = modsize;
|
||||
if (needed()) {
|
||||
fseek(input, offset, 0);
|
||||
archsize = modsize;
|
||||
for (i = 0; i < LEN_NAM; i++)
|
||||
if (modhead[OFF_NAM+i] == ' ')
|
||||
break;
|
||||
modhead[OFF_NAM+i] = '\0';
|
||||
parse(remember(&modhead[OFF_NAM]));
|
||||
}
|
||||
offset += modsize;
|
||||
}
|
||||
archmode = 0;
|
||||
}
|
||||
|
||||
needed()
|
||||
{
|
||||
register c, first;
|
||||
register item_t *ip;
|
||||
register need;
|
||||
|
||||
#ifdef LISTING
|
||||
register save;
|
||||
|
||||
save = listflag; listflag = 0;
|
||||
#endif
|
||||
need = 0;
|
||||
peekc = -1;
|
||||
first = 1;
|
||||
for (;;) {
|
||||
c = nextchar();
|
||||
if (c == '\n') {
|
||||
first = 1;
|
||||
continue;
|
||||
}
|
||||
if (c == ' ' || c == '\t' || c == ',')
|
||||
continue;
|
||||
if (ISALPHA(c) == 0)
|
||||
break;
|
||||
if ((ip = item_search(readident(c))) == 0) {
|
||||
if (first)
|
||||
break;
|
||||
continue;
|
||||
}
|
||||
if (first) {
|
||||
if (ip != &keytab[KEYDEFINE])
|
||||
break;
|
||||
first = 0;
|
||||
}
|
||||
if ((ip->i_type & S_TYP) == S_UND) {
|
||||
need++;
|
||||
break;
|
||||
}
|
||||
}
|
||||
#ifdef LISTING
|
||||
listflag = save;
|
||||
#endif
|
||||
return(need);
|
||||
}
|
||||
#endif ASLD
|
||||
|
||||
parse(s)
|
||||
char *s;
|
||||
{
|
||||
register i;
|
||||
register item_t *ip;
|
||||
register char *p;
|
||||
|
||||
for (p = s; *p; )
|
||||
if (*p++ == '/')
|
||||
s = p;
|
||||
#ifdef ASLD
|
||||
yylval.y_strp = s;
|
||||
putval(MODULE);
|
||||
#endif
|
||||
for (i = 0; i < FB_SIZE; i++)
|
||||
fb_ptr[FB_BACK+i] = 0;
|
||||
newmodule(s);
|
||||
peekc = -1;
|
||||
yyparse();
|
||||
/*
|
||||
* Check for undefined symbols
|
||||
*/
|
||||
#ifdef ASLD
|
||||
for (i = 0; i < H_SIZE; i++) {
|
||||
while (ip = hashtab[H_LOCAL+i]) {
|
||||
/*
|
||||
* cleanup local queue
|
||||
*/
|
||||
hashtab[H_LOCAL+i] = ip->i_next;
|
||||
/*
|
||||
* make undefined references extern
|
||||
*/
|
||||
if ((ip->i_type & (S_VAR|S_TYP)) == S_UND)
|
||||
ip->i_type |= S_EXT;
|
||||
/*
|
||||
* relink externals in global queue
|
||||
*/
|
||||
if (ip->i_type & S_EXT)
|
||||
item_insert(ip, H_GLOBAL+i);
|
||||
}
|
||||
}
|
||||
#else
|
||||
for (i = 0; i < H_SIZE; i++) {
|
||||
for (ip = hashtab[H_LOCAL+i]; ip; ip = ip->i_next) {
|
||||
if (ip->i_type & S_EXT)
|
||||
continue;
|
||||
if (ip->i_type != S_UND)
|
||||
continue;
|
||||
if (uflag == 0)
|
||||
serror("undefined symbol %s", ip->i_name);
|
||||
ip->i_type |= S_EXT;
|
||||
}
|
||||
}
|
||||
#endif
|
||||
/*
|
||||
* Check for undefined numeric labels
|
||||
*/
|
||||
for (i = 0; i < FB_SIZE; i++) {
|
||||
if ((ip = fb_ptr[FB_FORW+i]) == 0)
|
||||
continue;
|
||||
serror("undefined label %d", i);
|
||||
fb_ptr[FB_FORW+i] = 0;
|
||||
}
|
||||
}
|
||||
|
||||
pass_23(n)
|
||||
{
|
||||
register i;
|
||||
#ifdef ASLD
|
||||
register addr_t base = 0;
|
||||
#endif
|
||||
register sect_t *sp;
|
||||
|
||||
if (nerrors)
|
||||
stop();
|
||||
pass = n;
|
||||
#ifdef LISTING
|
||||
listmode >>= 3;
|
||||
if (listmode & 4)
|
||||
ffreopen(listpath, listfile);
|
||||
listeoln = 1;
|
||||
#endif
|
||||
#ifdef THREE_PASS
|
||||
nbits = 0;
|
||||
#endif
|
||||
for (i = 0; i < FB_SIZE; i++)
|
||||
fb_ptr[FB_FORW+i] = fb_ptr[FB_HEAD+i];
|
||||
outhead.oh_nemit = 0;
|
||||
for (sp = sect; sp < §[outhead.oh_nsect]; sp++) {
|
||||
#ifdef ASLD
|
||||
if (sp->s_flag & BASED) {
|
||||
base = sp->s_base;
|
||||
if (base % sp->s_lign)
|
||||
fatal("base not aligned");
|
||||
} else {
|
||||
base += (sp->s_lign - 1);
|
||||
base -= (base % sp->s_lign);
|
||||
sp->s_base = base;
|
||||
}
|
||||
base += sp->s_size;
|
||||
base += sp->s_comm;
|
||||
#endif
|
||||
outhead.oh_nemit += sp->s_size - sp->s_zero;
|
||||
}
|
||||
if (pass == PASS_3)
|
||||
setupoutput();
|
||||
for (sp = sect; sp < §[outhead.oh_nsect]; sp++) {
|
||||
sp->s_size = 0;
|
||||
sp->s_zero = 0;
|
||||
#ifdef THREE_PASS
|
||||
sp->s_gain = 0;
|
||||
#endif
|
||||
}
|
||||
machstart(n);
|
||||
#ifndef ASLD
|
||||
newmodule(modulename);
|
||||
#endif ASLD
|
||||
ffreopen(temppath, tempfile);
|
||||
yyparse();
|
||||
commfinish();
|
||||
machfinish(n);
|
||||
}
|
||||
|
||||
newmodule(s)
|
||||
char *s;
|
||||
{
|
||||
switchsect(S_UND);
|
||||
modulename = s;
|
||||
lineno = 1;
|
||||
if ((sflag & (SYM_EXT|SYM_LOC|SYM_LAB)) && PASS_SYMB)
|
||||
newsymb(s, S_MOD, (short)0, (valu_t)0);
|
||||
#ifdef LISTING
|
||||
listtemp = 0;
|
||||
if (dflag & 01000)
|
||||
listtemp = listmode;
|
||||
listflag = listtemp;
|
||||
#endif
|
||||
}
|
||||
|
||||
setupoutput()
|
||||
{
|
||||
register sect_t *sp;
|
||||
register long off;
|
||||
struct outsect outsect;
|
||||
|
||||
#ifdef AOUTSEEK
|
||||
#define AOUTseek(p,o) {aoutseek[p]=o;}
|
||||
aoutfile = ffcreat(aoutpath);
|
||||
#else
|
||||
#define AOUTseek(p,o) {fseek(aoutfile[p],o,0);}
|
||||
aoutfile[PARTEMIT]=ffcreat(aoutpath);
|
||||
#ifdef RELOCATION
|
||||
aoutfile[PARTRELO]=ffcreat(aoutpath);
|
||||
#endif
|
||||
aoutfile[PARTNAME]=ffcreat(aoutpath);
|
||||
aoutfile[PARTCHAR]=ffcreat(aoutpath);
|
||||
#endif
|
||||
/*
|
||||
* header generation
|
||||
*/
|
||||
AOUTseek(PARTEMIT, 0);
|
||||
putofmt((char *)&outhead, SF_HEAD, PARTEMIT);
|
||||
/*
|
||||
* section table generation
|
||||
*/
|
||||
off = SZ_HEAD;
|
||||
off += (long)outhead.oh_nsect * SZ_SECT;
|
||||
for (sp = sect; sp < §[outhead.oh_nsect]; sp++) {
|
||||
sp->s_foff = off;
|
||||
outsect.os_base = SETBASE(sp);
|
||||
outsect.os_size = sp->s_size + sp->s_comm;
|
||||
outsect.os_foff = sp->s_foff;
|
||||
outsect.os_flen = sp->s_size - sp->s_zero;
|
||||
outsect.os_lign = sp->s_lign;
|
||||
off += outsect.os_flen;
|
||||
putofmt((char *)&outsect, SF_SECT, PARTEMIT);
|
||||
}
|
||||
#ifdef RELOCATION
|
||||
AOUTseek(PARTRELO, off);
|
||||
off += (long)outhead.oh_nrelo * SZ_RELO;
|
||||
#endif
|
||||
if (sflag == 0)
|
||||
return;
|
||||
AOUTseek(PARTNAME, off);
|
||||
off += (long)outhead.oh_nname * SZ_NAME;
|
||||
AOUTseek(PARTCHAR, off);
|
||||
outhead.oh_nchar = off; /* see newsymb() */
|
||||
#undef AOUTseek
|
||||
}
|
||||
|
||||
commfinish()
|
||||
{
|
||||
register i;
|
||||
register item_t *ip;
|
||||
register sect_t *sp;
|
||||
register valu_t addr;
|
||||
|
||||
switchsect(S_UND);
|
||||
#ifdef ASLD
|
||||
/*
|
||||
* assign .comm labels and produce .comm symbol table entries
|
||||
*/
|
||||
for (i = 0; i<H_SIZE; i++)
|
||||
for (ip = hashtab[H_GLOBAL+i]; ip; ip = ip->i_next) {
|
||||
if ((ip->i_type & S_COM) == 0)
|
||||
continue;
|
||||
sp = §[(ip->i_type & S_TYP) - S_MIN];
|
||||
if (pass == PASS_1) {
|
||||
addr = sp->s_size + sp->s_comm;
|
||||
sp->s_comm += ip->i_valu;
|
||||
ip->i_valu = addr;
|
||||
}
|
||||
#ifdef THREE_PASS
|
||||
if (pass == PASS_2)
|
||||
ip->i_valu -= sp->s_gain;
|
||||
#endif
|
||||
if ((sflag & SYM_EXT) && PASS_SYMB)
|
||||
newsymb(
|
||||
ip->i_name,
|
||||
ip->i_type & (S_EXT|S_TYP),
|
||||
(short)0,
|
||||
load(ip)
|
||||
);
|
||||
}
|
||||
#endif
|
||||
if (PASS_SYMB == 0)
|
||||
return;
|
||||
#ifndef ASLD
|
||||
/*
|
||||
* produce symbol table entries for undefined's
|
||||
*/
|
||||
for (i = 0; i<H_SIZE; i++)
|
||||
for (ip = hashtab[H_LOCAL+i]; ip; ip = ip->i_next) {
|
||||
if (ip->i_type != (S_EXT|S_UND))
|
||||
continue;
|
||||
if (pass != PASS_3)
|
||||
/*
|
||||
* save symbol table index
|
||||
* for possible relocation
|
||||
*/
|
||||
ip->i_valu = outhead.oh_nname;
|
||||
if (sflag & SYM_SCT)
|
||||
newsymb(
|
||||
ip->i_name,
|
||||
S_EXT|S_UND,
|
||||
(short)0,
|
||||
(valu_t)0
|
||||
);
|
||||
}
|
||||
#endif ASLD
|
||||
/*
|
||||
* produce symbol table entries for sections
|
||||
*/
|
||||
if (sflag & SYM_SCT)
|
||||
for (sp = sect; sp < §[outhead.oh_nsect]; sp++) {
|
||||
ip = sp->s_item;
|
||||
newsymb(
|
||||
ip->i_name,
|
||||
(short)(ip->i_type | S_SCT),
|
||||
(short)0,
|
||||
load(ip)
|
||||
);
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user