17bc9cdef7
Most warnings are for functions implicitly returning int. Change most
of these functions to return void. (Traditional K&R C had no void
type, but C89 has it.)
Add prototypes to most function declarations in headers. This is
easy, because ego declares most of its extern functions, and the
comments listed most parameters. There were a few outdated or missing
declarations, and a few .c files that failed to include an .h with the
declarations.
Add prototypes to a few function definitions in .c files. Most
functions still have traditional K&R definitions. Most STATIC
functions still don't have prototypes, because they have no earlier
declaration where I would have added the prototype.
Change some prototypes in util/ego/share/alloc.h. Functions newmap()
and oldmap() handle an array of pointers to something; change the
array's type from `short **` to `void **`. Callers use casts to go
between `void **` and the correct type, like `line_p *`. Function
oldtable() takes a `short *`, not a `short **`; I added the wrong type
in 5bbbaf4
.
Make a few other changes to silence warnings. There are a few places
where clang wants extra parentheses in the code.
Edit util/ego/ra/build.lua to add the missing dependency on ra*.h; I
needed this to prevent crashes from ra.
573 lines
12 KiB
C
573 lines
12 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".
|
|
*/
|
|
/*
|
|
* R E G I S T E R A L L O C A T I O N
|
|
*
|
|
*/
|
|
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <em_reg.h>
|
|
#include "../share/types.h"
|
|
#include "../share/debug.h"
|
|
#include "../share/global.h"
|
|
#include "../share/files.h"
|
|
#include "../share/get.h"
|
|
#include "../share/put.h"
|
|
#include "../share/cset.h"
|
|
#include "../share/lset.h"
|
|
#include "../share/map.h"
|
|
#include "../share/alloc.h"
|
|
#include "../share/go.h"
|
|
#include "ra.h"
|
|
#include "ra_items.h"
|
|
#include "ra_allocl.h"
|
|
#include "ra_lifet.h"
|
|
#include "ra_profits.h"
|
|
#include "ra_pack.h"
|
|
#include "ra_xform.h"
|
|
|
|
|
|
#define newrabx() (bext_p) newstruct(bext_ra)
|
|
#define newralpx() (lpext_p) newstruct(lpext_ra)
|
|
#define oldrabx(x) oldstruct(bext_ra,x)
|
|
#define oldralpx(x) oldstruct(lpext_ra,x)
|
|
|
|
STATIC void stat_regusage(alloc_p list);
|
|
|
|
short alloc_id;
|
|
static item_p items[NRITEMTYPES];
|
|
int nrinstrs;
|
|
line_p *instrmap;
|
|
|
|
cond_p alocaltab[NRREGTYPES][NRREGTYPES],alocaddrtab[NRREGTYPES][NRREGTYPES],
|
|
aconsttab,adconsttab,aglobaltab,aproctab;
|
|
cond_p olocaltab[NRREGTYPES],olocaddrtab[NRREGTYPES],
|
|
oconsttab,odconsttab,oglobaltab,oproctab;
|
|
cond_p regsav_cost;
|
|
|
|
short regs_available[] = {
|
|
/* Actually machine dependent; this is for vax2 */
|
|
3, /* reg_any i.e. data regs */
|
|
0, /* reg_loop */
|
|
3, /* reg_pointer i.e. address reg. */
|
|
0 /* reg_float */
|
|
} ;
|
|
|
|
short use_any_as_pointer = 0;
|
|
|
|
STATIC cond_p getcondtab(f)
|
|
FILE *f;
|
|
{
|
|
int l,i;
|
|
cond_p tab;
|
|
|
|
fscanf(f,"%d",&l);
|
|
tab = newcondtab(l);
|
|
for (i = 0; i < l; i++) {
|
|
fscanf(f,"%hd %hd %hd",&tab[i].mc_cond,&tab[i].mc_tval,
|
|
&tab[i].mc_sval);
|
|
}
|
|
assert(tab[l-1].mc_cond == DEFAULT);
|
|
return tab;
|
|
}
|
|
|
|
STATIC void get_atab(f,tab)
|
|
FILE *f;
|
|
cond_p tab[NRREGTYPES][NRREGTYPES];
|
|
{
|
|
int i,cnt,totyp,regtyp;
|
|
|
|
fscanf(f,"%d",&cnt);
|
|
for (i = 0; i < cnt; i++) {
|
|
fscanf(f,"%d %d",®typ,&totyp);
|
|
assert(regtyp >= 0 && regtyp < NRREGTYPES);
|
|
assert(totyp >= 0 && totyp < NRREGTYPES);
|
|
tab[regtyp][totyp] = getcondtab(f);
|
|
}
|
|
}
|
|
|
|
|
|
STATIC void get_otab(f,tab)
|
|
FILE *f;
|
|
cond_p tab[NRREGTYPES];
|
|
{
|
|
int i,cnt,regtyp;
|
|
|
|
fscanf(f,"%d",&cnt);
|
|
for (i = 0; i < cnt; i++) {
|
|
fscanf(f,"%d",®typ);
|
|
assert(regtyp >= 0 && regtyp < NRREGTYPES);
|
|
tab[regtyp] = getcondtab(f);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
STATIC void ra_machinit(void *vp)
|
|
{
|
|
/* Read target machine dependent information for this phase */
|
|
FILE *f = vp;
|
|
char s[100];
|
|
|
|
for (;;) {
|
|
while(getc(f) != '\n');
|
|
fscanf(f,"%99s",s);
|
|
if (strcmp(s,"%%RA") == 0)break;
|
|
}
|
|
fscanf(f,"%hd",®s_available[reg_any]);
|
|
fscanf(f,"%hd",®s_available[reg_pointer]);
|
|
fscanf(f,"%hd",®s_available[reg_float]);
|
|
fscanf(f,"%hd",&use_any_as_pointer);
|
|
get_atab(f,alocaltab);
|
|
get_atab(f,alocaddrtab);
|
|
aconsttab = getcondtab(f);
|
|
adconsttab = getcondtab(f);
|
|
aglobaltab = getcondtab(f);
|
|
aproctab = getcondtab(f);
|
|
get_otab(f,olocaltab);
|
|
get_otab(f,olocaddrtab);
|
|
oconsttab = getcondtab(f);
|
|
odconsttab = getcondtab(f);
|
|
oglobaltab = getcondtab(f);
|
|
oproctab = getcondtab(f);
|
|
regsav_cost = getcondtab(f);
|
|
}
|
|
|
|
|
|
STATIC bblock_p header(lp)
|
|
loop_p lp;
|
|
{
|
|
/* Try to determine the 'header' block of loop lp.
|
|
* If 'e' is the entry block of loop L, then block 'b' is
|
|
* called the header block of L, iff:
|
|
* SUCC(b) = {e} & PRED(e) = {b}
|
|
* If lp has no header block, 0 is returned.
|
|
*/
|
|
|
|
bblock_p x = lp->lp_entry->b_idom;
|
|
|
|
if (x != (bblock_p) 0 && Lnrelems(x->b_succ) == 1 &&
|
|
(bblock_p) Lelem(Lfirst(x->b_succ)) == lp->lp_entry) {
|
|
return x;
|
|
}
|
|
return (bblock_p) 0;
|
|
}
|
|
|
|
|
|
STATIC void ra_extproc(p)
|
|
proc_p p;
|
|
{
|
|
/* Allocate the extended data structures for procedure p */
|
|
|
|
register loop_p lp;
|
|
register Lindex pi;
|
|
register bblock_p b;
|
|
|
|
for (pi = Lfirst(p->p_loops); pi != (Lindex) 0;
|
|
pi = Lnext(pi,p->p_loops)) {
|
|
lp = (loop_p) Lelem(pi);
|
|
lp->lp_extend = newralpx();
|
|
lp->LP_HEADER = header(lp);
|
|
}
|
|
for (b = p->p_start; b != (bblock_p) 0; b = b->b_next) {
|
|
b->b_extend = newrabx();
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
STATIC void ra_cleanproc(p)
|
|
proc_p p;
|
|
{
|
|
/* Allocate the extended data structures for procedure p */
|
|
|
|
register loop_p lp;
|
|
register Lindex pi;
|
|
register bblock_p b;
|
|
|
|
for (pi = Lfirst(p->p_loops); pi != (Lindex) 0;
|
|
pi = Lnext(pi,p->p_loops)) {
|
|
lp = (loop_p) Lelem(pi);
|
|
oldralpx(lp->lp_extend);
|
|
}
|
|
for (b = p->p_start; b != (bblock_p) 0; b = b->b_next) {
|
|
oldrabx(b->b_extend);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
STATIC void loop_blocks(p)
|
|
proc_p p;
|
|
{
|
|
/* Compute the LP_BLOCKS sets for all loops of p */
|
|
|
|
register bblock_p b;
|
|
register Lindex i;
|
|
|
|
for (b = p->p_start; b != (bblock_p) 0; b = b->b_next) {
|
|
for (i = Lfirst(b->b_loops); i != (Lindex) 0;
|
|
i = Lnext(i,b->b_loops)) {
|
|
Ladd(b,&(((loop_p) Lelem(i))->LP_BLOCKS));
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
STATIC void make_instrmap(p,map)
|
|
proc_p p;
|
|
line_p map[];
|
|
{
|
|
/* make the instructions map of procedure p */
|
|
|
|
register bblock_p b;
|
|
register line_p l;
|
|
register int i = 0;
|
|
|
|
for (b = p->p_start; b != (bblock_p) 0; b = b->b_next) {
|
|
b->B_BEGIN = i; /* number of first instruction */
|
|
for (l = b->b_start; l != (line_p) 0; l = l->l_next) {
|
|
map[i++] = l;
|
|
}
|
|
b->B_END = i-1; /* number of last instruction */
|
|
}
|
|
}
|
|
|
|
|
|
|
|
STATIC bool useful_item(item)
|
|
item_p item;
|
|
{
|
|
/* See if it may be useful to put the item in a register.
|
|
* A local variable may always be put in a register.
|
|
* Other items must be used at least twice.
|
|
*/
|
|
|
|
int nruses = Lnrelems(item->it_usage);
|
|
assert (nruses > 0); /* otherwise it would not be an item! */
|
|
return nruses > 1 || item->it_type == LOCALVAR;
|
|
}
|
|
|
|
|
|
STATIC void cleantimeset(s)
|
|
lset s;
|
|
{
|
|
register Lindex i;
|
|
register time_p t;
|
|
|
|
for (i = Lfirst(s); i != (Lindex) 0; i = Lnext(i,s)) {
|
|
t = (time_p) Lelem(i);
|
|
oldtime(t);
|
|
}
|
|
Ldeleteset(s);
|
|
}
|
|
|
|
|
|
|
|
STATIC item_p cat_items(items)
|
|
item_p items[];
|
|
{
|
|
/* Make one item list out of an array of itemlists.
|
|
* Remove items that are used only once.
|
|
*/
|
|
|
|
register item_p it;
|
|
item_p *ip,head,next;
|
|
int t;
|
|
|
|
|
|
ip = &head;
|
|
for (t = 0; t < NRITEMTYPES;t++) {
|
|
for ( it = items[t]; it != (item_p) 0; it = next) {
|
|
next = it->it_next;
|
|
if (!it->it_desirable || !useful_item(it)) {
|
|
cleantimeset(it->it_usage);
|
|
olditem(it);
|
|
} else {
|
|
*ip = it;
|
|
ip = &it->it_next;
|
|
}
|
|
}
|
|
}
|
|
*ip = (item_p) 0;
|
|
return head;
|
|
}
|
|
|
|
|
|
|
|
|
|
STATIC void clean_interval(list)
|
|
interv_p list;
|
|
{
|
|
register interv_p x,next;
|
|
|
|
for (x = list; x != (interv_p) 0; x = next) {
|
|
next = x->i_next;
|
|
oldinterval(x);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
STATIC void clean_allocs(list)
|
|
alloc_p list;
|
|
{
|
|
register alloc_p x,next;
|
|
|
|
for (x = list; x != (alloc_p) 0; x = next) {
|
|
next = x->al_next;
|
|
clean_interval(x->al_timespan);
|
|
Cdeleteset(x->al_rivals);
|
|
Ldeleteset(x->al_inits);
|
|
clean_interval(x->al_busy);
|
|
clean_allocs(x->al_mates);
|
|
oldalloc(x);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
STATIC void cleanitems(list)
|
|
item_p list;
|
|
{
|
|
register item_p x,next;
|
|
|
|
for (x = list; x != (item_p) 0; x = next ) {
|
|
next = x->it_next;
|
|
cleantimeset(x->it_usage);
|
|
olditem(x);
|
|
}
|
|
}
|
|
|
|
|
|
/* ARGSUSED */
|
|
void ra_initialize(void *null)
|
|
{
|
|
init_replacements(ps,ws);
|
|
}
|
|
|
|
|
|
void ra_optimize(void *vp)
|
|
{
|
|
proc_p p = vp;
|
|
item_p itemlist;
|
|
alloc_p alloclist,packed,unpacked;
|
|
offset locls;
|
|
bool time_opt = (time_space_ratio == 100);
|
|
|
|
if (IS_ENTERED_WITH_GTO(p)) return;
|
|
ra_extproc(p);
|
|
loop_blocks(p);
|
|
alloc_id =0;
|
|
locls = p->p_localbytes;
|
|
build_itemlist(p,items,&nrinstrs);
|
|
instrmap = (line_p *) newmap(nrinstrs-1); /* map starts counting at 0 */
|
|
make_instrmap(p,instrmap);
|
|
build_lifetimes(items);
|
|
/* print_items(items,p); */
|
|
/* statistics(items); */
|
|
itemlist = cat_items(items); /* make one list */
|
|
alloclist = build_alloc_list(p,Lnrelems(p->p_loops),
|
|
itemlist);
|
|
build_rivals_graph(alloclist);
|
|
compute_profits(alloclist,time_opt);
|
|
/* print_allocs(alloclist); */
|
|
pack(alloclist,time_opt,&packed,&unpacked,p);
|
|
stat_regusage(packed);
|
|
xform_proc(p,packed,nrinstrs,instrmap);
|
|
/* print_allocs(packed); */
|
|
p->p_localbytes = locls;
|
|
/* don't really allocate dummy local variables! */
|
|
rem_locals(p,packed);
|
|
rem_formals(p,packed);
|
|
/* remove storage for real locals that
|
|
*are always put in register .
|
|
*/
|
|
clean_allocs(unpacked);
|
|
clean_allocs(packed);
|
|
cleanitems(itemlist);
|
|
oldmap((void **) instrmap,nrinstrs-1);
|
|
ra_cleanproc(p);
|
|
}
|
|
|
|
|
|
|
|
int main(argc,argv)
|
|
int argc;
|
|
char *argv[];
|
|
{
|
|
go(argc,argv,ra_initialize,ra_optimize,ra_machinit,no_action);
|
|
exit(0);
|
|
}
|
|
|
|
|
|
/***************************************************************************/
|
|
/***************************************************************************/
|
|
/***************************************************************************/
|
|
|
|
/* debugging stuff */
|
|
|
|
|
|
|
|
char *str_types[] = {
|
|
"local variable",
|
|
"addr. of local",
|
|
"addr. of external",
|
|
"addr. of procedure",
|
|
"constant",
|
|
"double constant"
|
|
};
|
|
|
|
char *str_regtypes[] = {
|
|
"any",
|
|
"loop",
|
|
"pointer",
|
|
"float"
|
|
};
|
|
|
|
|
|
/*
|
|
* All calls to print_items() and print_allocs() are in comments!
|
|
*/
|
|
#if 0
|
|
print_items(items,p)
|
|
item_p items[];
|
|
proc_p p;
|
|
{
|
|
int t;
|
|
item_p item;
|
|
interv_p iv;
|
|
|
|
fprintf(stderr, "BEGIN PROCEDURE %d\n",p->p_id);
|
|
for (t = 0; t < NRITEMTYPES;t++) {
|
|
for (item = items[t]; item != (item_p) 0;item = item->it_next) {
|
|
fprintf(stderr, "\nitemtype = %s\n",str_types[t]);
|
|
if (t == GLOBL_ADDR) {
|
|
fprintf(stderr, "id of external = %d\n",
|
|
item->i_t.it_obj->o_id);
|
|
} else {
|
|
fprintf(stderr, "offset = %ld\n",
|
|
item->i_t.it_off);
|
|
}
|
|
fprintf(stderr, "regtype = %s\n",str_regtypes[item->it_regtype]);
|
|
fprintf(stderr, "size = %d\n",item->it_size);
|
|
fprintf(stderr, "#usages = %d\n", Lnrelems(item->it_usage));
|
|
fprintf(stderr, "lifetime = {");
|
|
for (iv = item->it_lives; iv != (interv_p) 0;
|
|
iv = iv->i_next) {
|
|
fprintf(stderr, "(%d,%d) ",iv->i_start,iv->i_stop);
|
|
}
|
|
fprintf(stderr, "} \n");
|
|
}
|
|
}
|
|
fprintf(stderr, "END PROCEDURE %d\n\n",p->p_id);
|
|
}
|
|
|
|
|
|
print_allocs(list)
|
|
alloc_p list;
|
|
{
|
|
alloc_p al,m;
|
|
item_p item;
|
|
short t;
|
|
interv_p iv;
|
|
|
|
fprintf(stderr,"BEGIN ALLOCLIST of proc %d\n",curproc->p_id);
|
|
for (m = list ; m != (alloc_p) 0; m = m->al_next) {
|
|
for (al = m; al != (alloc_p) 0; al = al->al_mates) {
|
|
item = al->al_item;
|
|
t = item->it_type;
|
|
fprintf(stderr,"\nitem: [type = %s, ",str_types[t]);
|
|
switch(t) {
|
|
case GLOBL_ADDR:
|
|
fprintf(stderr,"id = %d]\n", item->i_t.it_obj->o_id);
|
|
break;
|
|
case PROC_ADDR:
|
|
fprintf(stderr,"id = %d]\n", item->i_t.it_proc->p_id);
|
|
break;
|
|
default:
|
|
fprintf(stderr,"offset = %ld]\n", item->i_t.it_off);
|
|
}
|
|
fprintf(stderr,"#usages(static) = %d\n",al->al_susecount);
|
|
fprintf(stderr,"#usages(dyn) = %d\n",al->al_dusecount);
|
|
fprintf(stderr,"#inits = %d\n",Lnrelems(al->al_inits));
|
|
fprintf(stderr,"isloop = %d\n",al->al_isloop);
|
|
fprintf(stderr,"timespan = {");
|
|
for (iv = al->al_timespan; iv != (interv_p) 0;
|
|
iv = iv->i_next) {
|
|
fprintf(stderr,"(%d,%d) ",iv->i_start,iv->i_stop);
|
|
}
|
|
fprintf(stderr,"} \n");
|
|
fprintf(stderr,"busy = {");
|
|
for (iv = al->al_busy; iv != (interv_p) 0;
|
|
iv = iv->i_next) {
|
|
fprintf(stderr,"(%d,%d) ",iv->i_start,iv->i_stop);
|
|
}
|
|
fprintf(stderr,"} \n");
|
|
fprintf(stderr,"profits = %d\n",al->al_profits);
|
|
fprintf(stderr,"dummy local = %ld\n",al->al_dummy);
|
|
fprintf(stderr,"regnr = %d\n",al->al_regnr);
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
|
|
STATIC short regs_needed[4];
|
|
STATIC void stat_regusage(list)
|
|
alloc_p list;
|
|
{
|
|
int i;
|
|
alloc_p x;
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
regs_needed[i] = 0;
|
|
}
|
|
for (x = list; x != (alloc_p) 0; x = x->al_next) {
|
|
regs_needed[x->al_regtype]++;
|
|
}
|
|
/* fprintf(stderr, "data regs:%d\n",regs_needed[reg_any]); */
|
|
/* fprintf(stderr, "address regs:%d\n",regs_needed[reg_pointer]); */
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* All calls to statistics() are in comments!
|
|
*/
|
|
#if 0
|
|
int cnt_regtypes[reg_float+1];
|
|
|
|
statistics(items)
|
|
item_p items[];
|
|
{
|
|
register item_p item,next;
|
|
int t,r;
|
|
int cnt;
|
|
|
|
fprintf(stderr, "\nSTATISTICS\n");
|
|
for (r = 0; r <= reg_float; r++) cnt_regtypes[r] = 0;
|
|
for (t = 0; t < NRITEMTYPES;t++) {
|
|
cnt = 0;
|
|
for (item = items[t]; item != (item_p) 0;item = next) {
|
|
if (useful_item(item)) {
|
|
cnt++;
|
|
cnt_regtypes[item->it_regtype]++;
|
|
}
|
|
next = item->it_next;
|
|
}
|
|
fprintf(stderr, "#%s = %d\n",str_types[t],cnt);
|
|
}
|
|
for (r = 0; r <= reg_float; r++) {
|
|
fprintf(stderr, "#%s = %d\n",str_regtypes[r],cnt_regtypes[r]);
|
|
}
|
|
}
|
|
#endif
|