This needed lots of refactoring to ego --- not all platforms have ego descr files, and ego will just crash if you invoke it without one. I think originally it was never intended that these platforms would be used at -O2 or above. Plats now only specify the ego descr file if they have one.
		
			
				
	
	
		
			361 lines
		
	
	
	
		
			8.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			361 lines
		
	
	
	
		
			8.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* $Id$ */
 | 
						|
/*
 | 
						|
 * (c) copyright 1987 by the Vrije Universiteit, Amsterdam, The Netherlands.
 | 
						|
 * See the copyright notice in the ACK home directory, in the file "Copyright".
 | 
						|
 */
 | 
						|
/*  I N L I N E   S U B S T I T U T I O N */
 | 
						|
 | 
						|
#include <stdlib.h>
 | 
						|
#include <stdio.h>
 | 
						|
#include <string.h>
 | 
						|
#include <em_path.h>
 | 
						|
#include <em_mnem.h>
 | 
						|
#include <em_pseu.h>
 | 
						|
#include "../share/types.h"
 | 
						|
#include "il.h"
 | 
						|
#include "../share/debug.h"
 | 
						|
#include "../share/alloc.h"
 | 
						|
#include "../share/global.h"
 | 
						|
#include "../share/lset.h"
 | 
						|
#include "../share/files.h"
 | 
						|
#include "../share/map.h"
 | 
						|
#include "il_aux.h"
 | 
						|
#include "il1_anal.h"
 | 
						|
#include "il2_aux.h"
 | 
						|
#include "il3_subst.h"
 | 
						|
#include "../share/get.h"
 | 
						|
#include "../share/put.h"
 | 
						|
#include "../share/go.h"
 | 
						|
 | 
						|
int calnr;
 | 
						|
int complete_program;
 | 
						|
calcnt_p cchead; /* call-count info of current proc */
 | 
						|
STATIC long space = 0;
 | 
						|
STATIC long total_size = 0;
 | 
						|
 | 
						|
STATIC char cname[128] = TMP_DIR;
 | 
						|
STATIC char ccname[128] = TMP_DIR;
 | 
						|
 | 
						|
/* For debugging only */
 | 
						|
STATIC char sname[128] = TMP_DIR;
 | 
						|
STATIC int kp_temps = 0;
 | 
						|
 | 
						|
int Ssubst;
 | 
						|
#ifdef VERBOSE
 | 
						|
int Senv, Srecursive, Slocals, Sinstrlab, Sparsefails, Spremoved, Scals;
 | 
						|
int Sbig_caller, Sdispensable, Schangedcallee, Sbigcallee, Sspace, Szeroratio;
 | 
						|
#endif
 | 
						|
 | 
						|
/* P A S S  1
 | 
						|
 *
 | 
						|
 * Pass 1 reads and analyses the EM text and the CFG.
 | 
						|
 * It determines for every procedure if it may be expanded
 | 
						|
 * in line and how it uses its formal parameters.
 | 
						|
 * It also collects all calls appearing in the program and
 | 
						|
 * recognizes the actual parameters of every call.
 | 
						|
 * The call descriptors are put in a file (calfile).
 | 
						|
 */
 | 
						|
 | 
						|
pass1(lnam, bnam, cnam) char* lnam, *bnam, *cnam;
 | 
						|
{
 | 
						|
	FILE* f, *gf, *cf, *ccf; /* The EM input, the basic block graph,
 | 
						|
				  * the call-list file and the calcnt file.
 | 
						|
				  */
 | 
						|
	long laddr;
 | 
						|
	bblock_p g;
 | 
						|
	short kind;
 | 
						|
	line_p l;
 | 
						|
 | 
						|
	f = openfile(lnam, "r");
 | 
						|
	gf = openfile(bnam, "r");
 | 
						|
	cf = openfile(cnam, "w");
 | 
						|
	ccf = openfile(ccname, "w");
 | 
						|
	mesregs = Lempty_set();
 | 
						|
	apriori(fproc);
 | 
						|
	/* use information from the procedure table to
 | 
						|
	 * see which calls certainly cannot be expanded.
 | 
						|
	 */
 | 
						|
	while (TRUE)
 | 
						|
	{
 | 
						|
		laddr = ftell(f);
 | 
						|
		if (!getunit(gf, f, &kind, &g, &l, &curproc, TRUE))
 | 
						|
			break;
 | 
						|
		/* Read the control flow graph and EM text of
 | 
						|
		 * one procedure and analyze it.
 | 
						|
		 */
 | 
						|
		if (kind == LDATA)
 | 
						|
		{
 | 
						|
			remunit(LDATA, (proc_p)0, l);
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
		OUTTRACE("flow graph of proc %d read", curproc->p_id);
 | 
						|
		assert(INSTR(g->b_start) == ps_pro);
 | 
						|
		curproc->p_start = g;
 | 
						|
		curproc->P_LADDR = laddr;
 | 
						|
		/* address of em text in em-file */
 | 
						|
		/* address of graph in basic block file */
 | 
						|
		curproc->P_SIZE = proclength(curproc); /* #instructions */
 | 
						|
		total_size += curproc->P_SIZE;
 | 
						|
		if (BIG_PROC(curproc))
 | 
						|
		{
 | 
						|
			/* curproc is too large to be expanded in line */
 | 
						|
			UNSUITABLE(curproc);
 | 
						|
		}
 | 
						|
		calnr = 0;
 | 
						|
		anal_proc(curproc, cf, ccf);
 | 
						|
		OUTTRACE("proc %d processed", curproc->p_id);
 | 
						|
		remunit(LTEXT, curproc, (line_p)0);
 | 
						|
		/* remove control flow graph + text */
 | 
						|
		OUTTRACE("graph of proc %d removed", curproc->p_id);
 | 
						|
		Ldeleteset(mesregs);
 | 
						|
		mesregs = Lempty_set();
 | 
						|
	}
 | 
						|
	fclose(f);
 | 
						|
	fclose(gf);
 | 
						|
	fclose(cf);
 | 
						|
	fclose(ccf);
 | 
						|
}
 | 
						|
 | 
						|
/* P A S S  2
 | 
						|
 *
 | 
						|
 * Pass 2 reads the calfile and determines which calls should
 | 
						|
 * be expanded in line. It does not use the EM text.
 | 
						|
 */
 | 
						|
 | 
						|
STATIC char cname2[128] = TMP_DIR;
 | 
						|
 | 
						|
pass2(cnam, space) char* cnam;
 | 
						|
long space;
 | 
						|
{
 | 
						|
	FILE* cf, *cf2, *ccf;
 | 
						|
	call_p c, a;
 | 
						|
 | 
						|
	cf = openfile(cnam, "r");
 | 
						|
	cf2 = openfile(cname2, "w");
 | 
						|
	ccf = openfile(ccname, "r");
 | 
						|
	while ((c = getcall(cf)) != (call_p)0)
 | 
						|
	{
 | 
						|
		/* process all calls */
 | 
						|
		if (SUITABLE(c->cl_proc) && anal_params(c))
 | 
						|
		{
 | 
						|
			/* called proc. may be put in line */
 | 
						|
			/* see which parameters may be put in line */
 | 
						|
			assign_ratio(c); /* assign a rank */
 | 
						|
			a = abstract(c); /* abstract essential info */
 | 
						|
			append_abstract(a, a->cl_caller);
 | 
						|
			/* put it in call-list of calling proc. */
 | 
						|
			putcall(c, cf2, (short)0);
 | 
						|
		}
 | 
						|
		else
 | 
						|
		{
 | 
						|
			rem_call(c);
 | 
						|
		}
 | 
						|
	}
 | 
						|
	select_calls(fproc, ccf, space);
 | 
						|
	fclose(cf);
 | 
						|
	if (!kp_temps)
 | 
						|
		unlink(cnam);
 | 
						|
	fclose(cf2);
 | 
						|
	fclose(ccf);
 | 
						|
	if (!kp_temps)
 | 
						|
		unlink(ccname);
 | 
						|
	cf2 = openfile(cname2, "r");
 | 
						|
	add_actuals(fproc, cf2);
 | 
						|
	cleancals(fproc); /* remove calls that were not selected */
 | 
						|
	/* add actual parameters to each selected call */
 | 
						|
	fclose(cf2);
 | 
						|
	if (!kp_temps)
 | 
						|
		unlink(cname2);
 | 
						|
}
 | 
						|
 | 
						|
/* P A S S  3
 | 
						|
 *
 | 
						|
 * pass 3 reads the substitution file and performs all
 | 
						|
 * substitutions described in that file. It reads the
 | 
						|
 * original EM text and produced a new (optimized)
 | 
						|
 * EM textfile.
 | 
						|
 */
 | 
						|
 | 
						|
pass3(lnam, lnam2) char* lnam, *lnam2;
 | 
						|
{
 | 
						|
	bool verbose = TRUE;
 | 
						|
	FILE* lfile, *lfilerand, *lfile2, *sfile;
 | 
						|
	call_p c, next;
 | 
						|
	line_p l, startscan, cal;
 | 
						|
	short lastcid; /* last call-id seen */
 | 
						|
 | 
						|
	lfile = openfile(lnam, "r");
 | 
						|
	lfilerand = openfile(lnam, "r");
 | 
						|
	lfile2 = openfile(lnam2, "w");
 | 
						|
	if (verbose)
 | 
						|
	{
 | 
						|
		sfile = openfile(sname, "w");
 | 
						|
	}
 | 
						|
	mesregs = Lempty_set();
 | 
						|
	while ((l = get_text(lfile, &curproc)) != (line_p)0)
 | 
						|
	{
 | 
						|
		if (curproc == (proc_p)0)
 | 
						|
		{
 | 
						|
			/* Just a data-unit; no real instructions */
 | 
						|
			putlines(l->l_next, lfile2);
 | 
						|
			oldline(l);
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
		if (IS_DISPENSABLE(curproc))
 | 
						|
		{
 | 
						|
			liquidate(curproc, l->l_next);
 | 
						|
		}
 | 
						|
		else
 | 
						|
		{
 | 
						|
			startscan = l->l_next;
 | 
						|
			lastcid = 0;
 | 
						|
			for (c = curproc->P_CALS; c != (call_p)0; c = next)
 | 
						|
			{
 | 
						|
				next = c->cl_cdr;
 | 
						|
				cal = scan_to_cal(startscan, c->cl_id - lastcid);
 | 
						|
				assert(cal != (line_p)0);
 | 
						|
				startscan = scan_to_cal(cal->l_next, 1);
 | 
						|
				/* next CAL */
 | 
						|
				lastcid = c->cl_id;
 | 
						|
				/* next CAL after current one */
 | 
						|
				substitute(lfilerand, c, cal, l->l_next);
 | 
						|
				if (verbose)
 | 
						|
				{
 | 
						|
					putcall(c, sfile, 0);
 | 
						|
				}
 | 
						|
				else
 | 
						|
				{
 | 
						|
					rem_call(c);
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
		putlines(l->l_next, lfile2);
 | 
						|
		Ldeleteset(mesregs);
 | 
						|
		mesregs = Lempty_set();
 | 
						|
		oldline(l);
 | 
						|
	}
 | 
						|
	fclose(lfile);
 | 
						|
	fclose(lfile2);
 | 
						|
	if (verbose)
 | 
						|
	{
 | 
						|
		fclose(sfile);
 | 
						|
		if (!kp_temps)
 | 
						|
			unlink(sname);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
STATIC il_extptab(ptab)
 | 
						|
    proc_p ptab;
 | 
						|
{
 | 
						|
	/* Allocate space for extension of proctable entries.
 | 
						|
	 * Also, initialise some of the fields just allocated.
 | 
						|
	 */
 | 
						|
 | 
						|
	register proc_p p;
 | 
						|
 | 
						|
	for (p = ptab; p != (proc_p)0; p = p->p_next)
 | 
						|
	{
 | 
						|
		p->p_extend = newilpx();
 | 
						|
		p->P_ORGLABELS = p->p_nrlabels;
 | 
						|
		p->P_ORGLOCALS = p->p_localbytes;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
STATIC il_cleanptab(ptab)
 | 
						|
    proc_p ptab;
 | 
						|
{
 | 
						|
	/* De-allocate space for extensions */
 | 
						|
 | 
						|
	register proc_p p;
 | 
						|
 | 
						|
	for (p = ptab; p != (proc_p)0; p = p->p_next)
 | 
						|
	{
 | 
						|
		oldilpx(p->p_extend);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
#ifdef VERBOSE
 | 
						|
Sdiagnostics()
 | 
						|
{
 | 
						|
	/* print statictical information */
 | 
						|
 | 
						|
	fprintf(stderr, "STATISTICS:\n");
 | 
						|
	fprintf(stderr, "Info about procedures:\n");
 | 
						|
	fprintf(stderr, "environment accessed: %d\n", Senv);
 | 
						|
	fprintf(stderr, "recursive: %d\n", Srecursive);
 | 
						|
	fprintf(stderr, "too many locals: %d\n", Slocals);
 | 
						|
	fprintf(stderr, "instr. lab in data block: %d\n", Sinstrlab);
 | 
						|
	fprintf(stderr, "procedures removed: %d\n", Spremoved);
 | 
						|
	fprintf(stderr, "\nInfo about calls:\n");
 | 
						|
	fprintf(stderr, "total number of calls: %d\n", Scals);
 | 
						|
	fprintf(stderr, "total number of calls substituted: %d\n", Ssubst);
 | 
						|
	fprintf(stderr, "parser failed: %d\n", Sparsefails);
 | 
						|
	fprintf(stderr, "caller too big: %d\n", Sbig_caller);
 | 
						|
	fprintf(stderr, "caller dispensable: %d\n", Sdispensable);
 | 
						|
	fprintf(stderr, "callee is changed: %d\n", Schangedcallee);
 | 
						|
	fprintf(stderr, "callee too big: %d\n", Sbigcallee);
 | 
						|
	fprintf(stderr, "no space available: %d\n", Sspace);
 | 
						|
	fprintf(stderr, "zero ratio: %d\n", Szeroratio);
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
il_flags(p) char* p;
 | 
						|
{
 | 
						|
	switch (*p++)
 | 
						|
	{
 | 
						|
		case 's':
 | 
						|
			while (*p != '\0')
 | 
						|
			{
 | 
						|
				space = 10 * space + *p++ - '0';
 | 
						|
			}
 | 
						|
			break;
 | 
						|
		case 'a':
 | 
						|
			complete_program = 1;
 | 
						|
			break;
 | 
						|
		case 't':
 | 
						|
			strcpy(cname, ".");
 | 
						|
			strcpy(ccname, ".");
 | 
						|
			strcpy(sname, ".");
 | 
						|
			strcpy(cname2, ".");
 | 
						|
			kp_temps = 1;
 | 
						|
			break;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
main(argc, argv) int argc;
 | 
						|
char* argv[];
 | 
						|
{
 | 
						|
	struct files* files = findfiles(argc, argv);
 | 
						|
	FILE* f;
 | 
						|
 | 
						|
	go(argc, argv, no_action, no_action, no_action, il_flags);
 | 
						|
	il_extptab(fproc); /* add extended data structures */
 | 
						|
	strcat(cname, "/ego.i1.XXXXXX");
 | 
						|
	strcat(ccname, "/ego.i2.XXXXXX");
 | 
						|
	strcat(sname, "/ego.i3.XXXXXX");
 | 
						|
	strcat(cname2, "/ego.i4.XXXXXX");
 | 
						|
	mktemp(cname);
 | 
						|
	mktemp(ccname);
 | 
						|
	mktemp(sname);
 | 
						|
	mktemp(cname2);
 | 
						|
	pass1(files->lname_in, files->bname_in, cname); /* grep calls, analyse procedures */
 | 
						|
	space = total_size * space / 100;
 | 
						|
	pass2(cname, space); /* select calls to be expanded */
 | 
						|
	pass3(files->lname_in, files->lname_out); /* do substitutions */
 | 
						|
	f = openfile(files->dname_out, "w");
 | 
						|
	il_cleanptab(fproc); /* remove extended data structures */
 | 
						|
	putdtable(fdblock, f);
 | 
						|
	f = openfile(files->pname_out, "w");
 | 
						|
	putptable(fproc, f, FALSE);
 | 
						|
	report("inline substitutions", Ssubst);
 | 
						|
#ifdef VERBOSE
 | 
						|
	if (verbose_flag)
 | 
						|
	{
 | 
						|
		Sdiagnostics();
 | 
						|
	}
 | 
						|
#endif
 | 
						|
#ifdef DEBUG
 | 
						|
	core_usage();
 | 
						|
#endif
 | 
						|
	exit(0);
 | 
						|
}
 |