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.
236 lines
5.1 KiB
C
236 lines
5.1 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
|
|
*
|
|
* R A _ P R O F I T S . C
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <em_reg.h>
|
|
#include "../share/types.h"
|
|
#include "../share/debug.h"
|
|
#include "../share/lset.h"
|
|
#include "../share/global.h"
|
|
#include "ra.h"
|
|
#include "ra_aux.h"
|
|
#include "ra_profits.h"
|
|
|
|
STATIC bool test_cond(short cond, offset val)
|
|
{
|
|
switch(cond) {
|
|
case DEFAULT:
|
|
return TRUE;
|
|
case FITBYTE:
|
|
return val >= -128 && val < 128;
|
|
case IN_0_63:
|
|
return val >= 0 && val <= 63;
|
|
case IN_0_8:
|
|
return val >= 0 && val <= 8;
|
|
}
|
|
}
|
|
|
|
STATIC short map_value(struct cond_tab tab[], offset val, bool time)
|
|
{
|
|
cond_p p;
|
|
|
|
for (p = &tab[0]; ; p++) {
|
|
if (test_cond(p->mc_cond,val)) {
|
|
return (time ? p->mc_tval : p->mc_sval);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
STATIC short index_value(struct cond_tab tab[], short n, bool time)
|
|
{
|
|
cond_p p;
|
|
|
|
p = &tab[n];
|
|
return (time ? p->mc_tval : p->mc_sval);
|
|
}
|
|
|
|
|
|
STATIC void
|
|
allocscore(short itemtyp, short localtyp, short size, offset off,
|
|
short totyp, short *time_out, short *space_out)
|
|
{
|
|
cond_p m = (cond_p) 0;
|
|
|
|
if (localtyp == reg_loop) localtyp = reg_any;
|
|
if (size == ws || (size == ps && totyp == reg_pointer) ||
|
|
(size == 2 * ws && totyp == reg_float)) {
|
|
switch(itemtyp) {
|
|
case LOCALVAR:
|
|
m = alocaltab[localtyp][totyp];
|
|
break;
|
|
case LOCAL_ADDR:
|
|
if (use_any_as_pointer || totyp == reg_pointer)
|
|
m = alocaddrtab[localtyp][totyp];
|
|
break;
|
|
case CONST:
|
|
m = aconsttab;
|
|
break;
|
|
case DCONST:
|
|
m = aconsttab;
|
|
break;
|
|
case GLOBL_ADDR:
|
|
if (use_any_as_pointer || totyp == reg_pointer)
|
|
m = aglobaltab;
|
|
break;
|
|
case PROC_ADDR:
|
|
if (use_any_as_pointer || totyp == reg_pointer)
|
|
m = aproctab;
|
|
break;
|
|
}
|
|
}
|
|
*time_out = (m == (cond_p) 0 ? -1 : map_value(m,off,TRUE));
|
|
*space_out = (m == (cond_p) 0 ? -1 : map_value(m,off,FALSE));
|
|
/*
|
|
fprintf(stderr,"itemtyp = %d, localtyp = %d off = %ld\n",itemtyp,localtyp,off);
|
|
fprintf(stderr,"ALLOCSCORE = (%d,%d)\n",*time_out,*space_out);
|
|
*/
|
|
}
|
|
|
|
STATIC void
|
|
opening_cost(short itemtyp, short localtyp, offset off,
|
|
short *time_out, short *space_out)
|
|
{
|
|
cond_p m;
|
|
|
|
if (localtyp == reg_loop) localtyp = reg_any;
|
|
switch(itemtyp) {
|
|
case LOCALVAR:
|
|
m = olocaltab[localtyp];
|
|
break;
|
|
case LOCAL_ADDR:
|
|
m = olocaddrtab[localtyp];
|
|
break;
|
|
case CONST:
|
|
m = oconsttab;
|
|
break;
|
|
case DCONST:
|
|
m = oconsttab;
|
|
break;
|
|
case GLOBL_ADDR:
|
|
m = oglobaltab;
|
|
break;
|
|
case PROC_ADDR:
|
|
m = oproctab;
|
|
break;
|
|
}
|
|
*time_out = (m == (cond_p) 0 ? 1000 : map_value(m,off,TRUE));
|
|
*space_out = (m == (cond_p) 0 ? 1000 : map_value(m,off,FALSE));
|
|
/*
|
|
fprintf(stderr,"itemtyp = %d, localtyp = %d off = %ld\n",itemtyp,localtyp,off);
|
|
fprintf(stderr,"OPEN_COST = (%d,%d)\n",*time_out,*space_out);
|
|
*/
|
|
}
|
|
|
|
|
|
|
|
|
|
void regsave_cost(short regs[], short *time_out, short *space_out)
|
|
{
|
|
/* Estimate the costs of saving and restoring the registers
|
|
* The array regs contains the number of registers of every
|
|
* possible type.
|
|
*/
|
|
|
|
short n = regs[reg_any] + regs[reg_pointer] + regs[reg_float];
|
|
/* #registers */
|
|
|
|
*time_out = index_value(regsav_cost,n,TRUE);
|
|
*space_out = index_value(regsav_cost,n,FALSE);
|
|
/*
|
|
fprintf(stderr,"REGSAVE COST, n=%d, (%d,%d)\n",n,*time_out,*space_out);
|
|
*/
|
|
}
|
|
|
|
|
|
|
|
STATIC short dyn_inits(inits)
|
|
lset inits;
|
|
{
|
|
Lindex i;
|
|
short sum = 0;
|
|
bblock_p b;
|
|
|
|
for (i = Lfirst(inits); i != (Lindex) 0; i = Lnext(i,inits)) {
|
|
b = (bblock_p) Lelem(i);
|
|
sum += loop_scale(Lnrelems(b->b_loops));
|
|
}
|
|
return sum;
|
|
}
|
|
|
|
|
|
|
|
void compute_profits(alloc_p alloclist, bool time_opt)
|
|
{
|
|
/* Compute the profits attribute of every allocation.
|
|
* If the item of an allocation may be put in several types
|
|
* of register, we choose only the most advanteagous one.
|
|
*/
|
|
|
|
register alloc_p alloc;
|
|
short s,t,rtyp,maxsc;
|
|
item_p item;
|
|
short time,space,sc;
|
|
short otime,ospace;
|
|
offset off;
|
|
short cnt,nr_inits;
|
|
|
|
for (alloc = alloclist; alloc != (alloc_p) 0; alloc = alloc->al_next) {
|
|
maxsc = 0;
|
|
item = alloc->al_item;
|
|
switch(item->it_type) {
|
|
case LOCALVAR:
|
|
case LOCAL_ADDR:
|
|
case CONST:
|
|
case DCONST:
|
|
off = item->i_t.it_off;
|
|
break;
|
|
default:
|
|
off = 0;
|
|
}
|
|
for (rtyp = item->it_regtype; ; rtyp = reg_any) {
|
|
allocscore( item->it_type,
|
|
item->it_regtype,
|
|
item->it_size,
|
|
off,
|
|
rtyp,
|
|
&time,
|
|
&space);
|
|
opening_cost( item->it_type,
|
|
item->it_regtype,
|
|
off,
|
|
&otime,
|
|
&ospace);
|
|
nr_inits = Lnrelems(alloc->al_inits);
|
|
s = alloc->al_susecount * space -
|
|
nr_inits*ospace;
|
|
#ifdef __STRANGE__
|
|
if (!alloc->al_isloop && nr_inits > 0) {
|
|
/* might lead to increase of execution time */
|
|
cnt = 0;
|
|
} else
|
|
#endif
|
|
{
|
|
cnt = alloc->al_dusecount;
|
|
}
|
|
t = cnt * time - dyn_inits(alloc->al_inits) * otime;
|
|
sc = (time_opt ? t : s);
|
|
/*
|
|
fprintf(stderr, "cnt: %d time: %d otime: %d t: %d s: %d score: %d\n", cnt, time, otime, t, s, sc);
|
|
*/
|
|
if (sc > maxsc) {
|
|
maxsc = sc;
|
|
alloc->al_regtype = rtyp;
|
|
alloc->al_profits = sc;
|
|
}
|
|
if (rtyp == reg_any) break;
|
|
}
|
|
}
|
|
}
|