371 lines
		
	
	
	
		
			6.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			371 lines
		
	
	
	
		
			6.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
#ifndef NORCSID
 | 
						|
static char rcsid[] = "$Id$";
 | 
						|
#endif
 | 
						|
 | 
						|
#include "param.h"
 | 
						|
#include "types.h"
 | 
						|
#include "tes.h"
 | 
						|
#include "assert.h"
 | 
						|
#include <em_spec.h>
 | 
						|
#include <em_pseu.h>
 | 
						|
#include <em_mnem.h>
 | 
						|
#include <em_flag.h>
 | 
						|
#include "alloc.h"
 | 
						|
#include "line.h"
 | 
						|
#include "lookup.h"
 | 
						|
#include "proinf.h"
 | 
						|
#include "optim.h"
 | 
						|
#include "ext.h"
 | 
						|
 | 
						|
/*
 | 
						|
 * (c) copyright 1987 by the Vrije Universiteit, Amsterdam, The Netherlands.
 | 
						|
 * See the copyright notice in the ACK home directory, in the file "Copyright".
 | 
						|
 *
 | 
						|
 * Author: Hans van Staveren
 | 
						|
 */
 | 
						|
 | 
						|
#define outbyte(b) putc(b,outfile)
 | 
						|
 | 
						|
putlines(lnp) register line_p lnp; {
 | 
						|
	register arg_p ap;
 | 
						|
	line_p temp;
 | 
						|
	register instr;
 | 
						|
	short curlin= -2;
 | 
						|
	short thislin;
 | 
						|
 | 
						|
	while ( lnp != (line_p) 0) {
 | 
						|
		instr = lnp->l_instr&BMASK;
 | 
						|
		switch(lnp->l_optyp) {
 | 
						|
		case OPSYMBOL:
 | 
						|
			if ((lnp->l_instr&BMASK) == ps_sym)
 | 
						|
				outdef(lnp->l_a.la_sp);
 | 
						|
			else
 | 
						|
				outocc(lnp->l_a.la_sp);
 | 
						|
			break;
 | 
						|
		case OPSVAL:
 | 
						|
			outocc(lnp->l_a.la_sval.lasv_sp);
 | 
						|
			break;
 | 
						|
#ifdef LONGOFF
 | 
						|
		case OPLVAL:
 | 
						|
			outocc(lnp->l_a.la_lval.lalv_sp);
 | 
						|
			break;
 | 
						|
#endif
 | 
						|
		case OPLIST:
 | 
						|
			ap = lnp->l_a.la_arg;
 | 
						|
			while (ap != (arg_p) 0) {
 | 
						|
				switch(ap->a_typ) {
 | 
						|
				case ARGSYM:
 | 
						|
					outocc(ap->a_a.a_sp);
 | 
						|
					break;
 | 
						|
				case ARGVAL:
 | 
						|
					outocc(ap->a_a.a_val.av_sp);
 | 
						|
					break;
 | 
						|
				}
 | 
						|
				ap = ap->a_next;
 | 
						|
			}
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		/*
 | 
						|
		 * global symbols now taken care of
 | 
						|
		 */
 | 
						|
 | 
						|
 | 
						|
		switch(instr) {
 | 
						|
		case ps_sym:
 | 
						|
			break;
 | 
						|
		case op_lni:
 | 
						|
			if (curlin != -2)
 | 
						|
				curlin++;
 | 
						|
			outinst(instr);
 | 
						|
			break;
 | 
						|
		case op_lin:
 | 
						|
			switch(lnp->l_optyp) {
 | 
						|
			case OPNO:
 | 
						|
			case OPOFFSET:
 | 
						|
			case OPNUMLAB:
 | 
						|
			case OPSYMBOL:
 | 
						|
			case OPSVAL:
 | 
						|
			case OPLVAL:
 | 
						|
			case OPLIST:
 | 
						|
				outinst(instr);
 | 
						|
				goto processoperand;
 | 
						|
			case OPSHORT:
 | 
						|
				thislin = lnp->l_a.la_short;
 | 
						|
				break;
 | 
						|
			default:
 | 
						|
				thislin = (lnp->l_optyp&BMASK)-Z_OPMINI;
 | 
						|
				break;
 | 
						|
			}
 | 
						|
			if (thislin == curlin && !nflag) {
 | 
						|
				temp = lnp->l_next;
 | 
						|
				oldline(lnp);
 | 
						|
				lnp = temp;
 | 
						|
				OPTIM(O_LINGONE);
 | 
						|
				continue;
 | 
						|
			} else if (thislin == curlin+1 && !nflag) {
 | 
						|
				instr = op_lni;
 | 
						|
				outinst(instr);
 | 
						|
				temp = lnp->l_next;
 | 
						|
				oldline(lnp);
 | 
						|
				OPTIM(O_LINLNI);
 | 
						|
				lnp = newline(OPNO);
 | 
						|
				lnp->l_next = temp;
 | 
						|
				lnp->l_instr = instr;
 | 
						|
			} else {
 | 
						|
				outinst(instr);
 | 
						|
			}
 | 
						|
			curlin = thislin;
 | 
						|
			break;
 | 
						|
		case op_lab:
 | 
						|
			curlin = -2;
 | 
						|
			break;
 | 
						|
		default:
 | 
						|
			if ((em_flag[instr-sp_fmnem]&EM_FLO)==FLO_P)
 | 
						|
				curlin = -2;
 | 
						|
			outinst(instr);
 | 
						|
		}
 | 
						|
processoperand:
 | 
						|
		switch(lnp->l_optyp) {
 | 
						|
		case OPNO:
 | 
						|
			if ((em_flag[instr-sp_fmnem]&EM_PAR)!=PAR_NO)
 | 
						|
				outbyte( (byte) sp_cend) ;
 | 
						|
			break;
 | 
						|
		default:
 | 
						|
			outint((lnp->l_optyp&BMASK)-Z_OPMINI);
 | 
						|
			break;
 | 
						|
		case OPSHORT:
 | 
						|
			outint(lnp->l_a.la_short);
 | 
						|
			break;
 | 
						|
#ifdef LONGOFF
 | 
						|
		case OPOFFSET:
 | 
						|
			outoff(lnp->l_a.la_offset);
 | 
						|
			break;
 | 
						|
#endif
 | 
						|
		case OPNUMLAB:
 | 
						|
			if (instr == op_lab)
 | 
						|
				numlab(lnp->l_a.la_np->n_repl);
 | 
						|
			else if (instr < sp_fpseu) /* plain instruction */
 | 
						|
				outint((short) lnp->l_a.la_np->n_repl->n_number);
 | 
						|
			else
 | 
						|
				outnum(lnp->l_a.la_np->n_repl);
 | 
						|
			break;
 | 
						|
		case OPSYMBOL:
 | 
						|
			outsym(lnp->l_a.la_sp);
 | 
						|
			break;
 | 
						|
		case OPSVAL:
 | 
						|
			outbyte( (byte) sp_doff) ;
 | 
						|
			outsym(lnp->l_a.la_sval.lasv_sp);
 | 
						|
			outint(lnp->l_a.la_sval.lasv_short);
 | 
						|
			break;
 | 
						|
#ifdef LONGOFF
 | 
						|
		case OPLVAL:
 | 
						|
			outbyte( (byte) sp_doff) ;
 | 
						|
			outsym(lnp->l_a.la_lval.lalv_sp);
 | 
						|
			outoff(lnp->l_a.la_lval.lalv_offset);
 | 
						|
			break;
 | 
						|
#endif
 | 
						|
		case OPLIST:
 | 
						|
			putargs(lnp->l_a.la_arg);
 | 
						|
			switch(instr) {
 | 
						|
			case ps_con:
 | 
						|
			case ps_rom:
 | 
						|
			case ps_mes:
 | 
						|
				outbyte( (byte) sp_cend) ;
 | 
						|
			}
 | 
						|
		}
 | 
						|
		/*
 | 
						|
		 * instruction is output now.
 | 
						|
		 * remove its useless body
 | 
						|
		 */
 | 
						|
 | 
						|
		temp = lnp->l_next;
 | 
						|
		oldline(lnp);
 | 
						|
		lnp = temp;
 | 
						|
		if (ferror(outfile))
 | 
						|
			error("write error");
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
putargs(ap) register arg_p ap; {
 | 
						|
 | 
						|
	while (ap != (arg_p) 0) {
 | 
						|
		switch(ap->a_typ) {
 | 
						|
		default:
 | 
						|
			assert(FALSE);
 | 
						|
		case ARGOFF:
 | 
						|
			outoff(ap->a_a.a_offset);
 | 
						|
			break;
 | 
						|
		case ARGNUM:
 | 
						|
			outnum(ap->a_a.a_np->n_repl);
 | 
						|
			break;
 | 
						|
		case ARGSYM:
 | 
						|
			outsym(ap->a_a.a_sp);
 | 
						|
			break;
 | 
						|
		case ARGVAL:
 | 
						|
			outbyte( (byte) sp_doff) ;
 | 
						|
			outsym(ap->a_a.a_val.av_sp);
 | 
						|
			outoff(ap->a_a.a_val.av_offset);
 | 
						|
			break;
 | 
						|
		case ARGSTR:
 | 
						|
			outbyte( (byte) sp_scon) ;
 | 
						|
			putstr(&ap->a_a.a_string);
 | 
						|
			break;
 | 
						|
		case ARGICN:
 | 
						|
			outbyte( (byte) sp_icon) ;
 | 
						|
			goto casecon;
 | 
						|
		case ARGUCN:
 | 
						|
			outbyte( (byte) sp_ucon) ;
 | 
						|
			goto casecon;
 | 
						|
		case ARGFCN:
 | 
						|
			outbyte( (byte) sp_fcon) ;
 | 
						|
		casecon:
 | 
						|
			outint(ap->a_a.a_con.ac_length);
 | 
						|
			putstr(&ap->a_a.a_con.ac_con);
 | 
						|
			break;
 | 
						|
		}
 | 
						|
		ap = ap->a_next;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
putstr(abp) register argb_p abp; {
 | 
						|
	register argb_p tbp;
 | 
						|
	register length;
 | 
						|
 | 
						|
	length = 0;
 | 
						|
	tbp = abp;
 | 
						|
	while (tbp!= (argb_p) 0) {
 | 
						|
		length += tbp->ab_index;
 | 
						|
		tbp = tbp->ab_next;
 | 
						|
	}
 | 
						|
	outint(length);
 | 
						|
	while (abp != (argb_p) 0) {
 | 
						|
		for (length=0;length<abp->ab_index;length++)
 | 
						|
			outbyte( (byte) abp->ab_contents[length] );
 | 
						|
		abp = abp->ab_next;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
outdef(sp) register sym_p sp; {
 | 
						|
 | 
						|
	/*
 | 
						|
	 * The surrounding If statement is removed to be friendly
 | 
						|
	 * to Backend writers having to deal with assemblers
 | 
						|
	 * not following our conventions.
 | 
						|
	if ((sp->s_flags&SYMOUT)==0) {
 | 
						|
	 */
 | 
						|
		sp->s_flags |= SYMOUT;
 | 
						|
		if (sp->s_flags&SYMGLOBAL) {
 | 
						|
			outinst(sp->s_flags&SYMPRO ? ps_exp : ps_exa);
 | 
						|
			outsym(sp);
 | 
						|
		}
 | 
						|
	/*
 | 
						|
	}
 | 
						|
	 */
 | 
						|
}
 | 
						|
 | 
						|
outocc(sp) register sym_p sp; {
 | 
						|
 | 
						|
	if ((sp->s_flags&SYMOUT)==0) {
 | 
						|
		sp->s_flags |= SYMOUT;
 | 
						|
		if ((sp->s_flags&SYMGLOBAL)==0) {
 | 
						|
			outinst(sp->s_flags&SYMPRO ? ps_inp : ps_ina);
 | 
						|
			outsym(sp);
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
outpro() {
 | 
						|
 | 
						|
	outdef(curpro.symbol);
 | 
						|
	outinst(ps_pro);
 | 
						|
	outsym(curpro.symbol);
 | 
						|
	outoff(curpro.localbytes);
 | 
						|
}
 | 
						|
 | 
						|
outend() {
 | 
						|
 | 
						|
	outinst(ps_end);
 | 
						|
	outoff(curpro.localbytes);
 | 
						|
}
 | 
						|
 | 
						|
outinst(m) {
 | 
						|
 | 
						|
	outbyte( (byte) m );
 | 
						|
}
 | 
						|
 | 
						|
outoff(off) offset off; {
 | 
						|
 | 
						|
#ifdef LONGOFF
 | 
						|
	if ((short) off == off)
 | 
						|
#endif
 | 
						|
		outint((short) off);
 | 
						|
#ifdef LONGOFF
 | 
						|
	else {
 | 
						|
		outbyte( (byte) sp_cst4) ;
 | 
						|
		outshort( (short) (off&0177777L) );
 | 
						|
		outshort( (short) (off>>16) );
 | 
						|
	}
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
outint(i) short i; {
 | 
						|
 | 
						|
	if (i>= -sp_zcst0 && i< sp_ncst0-sp_zcst0)
 | 
						|
		outbyte( (byte) (i+sp_zcst0+sp_fcst0) );
 | 
						|
	else {
 | 
						|
		outbyte( (byte) sp_cst2) ;
 | 
						|
		outshort(i);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
outshort(i) short i; {
 | 
						|
 | 
						|
	outbyte( (byte) (i&BMASK) );
 | 
						|
	outbyte( (byte) (i>>8) );
 | 
						|
}
 | 
						|
 | 
						|
numlab(np) register num_p np; {
 | 
						|
 | 
						|
	if (np->n_number < sp_nilb0)
 | 
						|
		outbyte( (byte) (np->n_number + sp_filb0) );
 | 
						|
	else
 | 
						|
		outnum(np);
 | 
						|
}
 | 
						|
 | 
						|
outnum(np) register num_p np; {
 | 
						|
 | 
						|
	if(np->n_number<256) {
 | 
						|
		outbyte( (byte) sp_ilb1) ;
 | 
						|
		outbyte( (byte) (np->n_number) );
 | 
						|
	} else {
 | 
						|
		outbyte( (byte) sp_ilb2) ;
 | 
						|
		outshort((short) np->n_number);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
outsym(sp) register sym_p sp; {
 | 
						|
	register byte *p;
 | 
						|
	register unsigned num;
 | 
						|
 | 
						|
	if (sp->s_name[0] == '.') {
 | 
						|
		num = atoi(&sp->s_name[1]);
 | 
						|
		if (num < 256) {
 | 
						|
			outbyte( (byte) sp_dlb1) ;
 | 
						|
			outbyte( (byte) (num) );
 | 
						|
		} else {
 | 
						|
			outbyte( (byte) sp_dlb2) ;
 | 
						|
			outshort((short) num);
 | 
						|
		}
 | 
						|
	} else {
 | 
						|
		p= sp->s_name;
 | 
						|
		while (*p && p < &sp->s_name[IDL])
 | 
						|
			p++;
 | 
						|
		num = p - sp->s_name;
 | 
						|
		outbyte( (byte) (sp->s_flags&SYMPRO ? sp_pnam : sp_dnam) );
 | 
						|
		outint((short) num);
 | 
						|
		p = sp->s_name;
 | 
						|
		while (num--)
 | 
						|
			outbyte( (byte) *p++ );
 | 
						|
	}
 | 
						|
}
 |