226 lines
4.6 KiB
C
226 lines
4.6 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".
|
|
*/
|
|
/* S T A C K P O L L U T I O N
|
|
*
|
|
* S P . C
|
|
*/
|
|
|
|
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <em_mnem.h>
|
|
#include <em_spec.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/lset.h"
|
|
#include "../share/map.h"
|
|
#include "../share/alloc.h"
|
|
#include "../share/utils.h"
|
|
#include "../share/go.h"
|
|
#include "../share/stack_chg.h"
|
|
|
|
|
|
/* Stack pollution throws away the ASP instructions after a procedure call.
|
|
* This saves a lot of code, at the cost of some extra stack space.
|
|
* ASPs that are part of a loop are not removed.
|
|
*/
|
|
|
|
#define BF_MARK 04
|
|
#define MARK(b) b->b_flags |= BF_MARK
|
|
#define NOT_MARKED(b) (!(b->b_flags&BF_MARK))
|
|
#define IN_LOOP(b) (Lnrelems(b->b_loops) > 0)
|
|
|
|
STATIC int Ssp; /* number of optimizations */
|
|
|
|
/* According to the EM definition, the stack must be cleaned up
|
|
* before any return. However, for some backends it causes no harm
|
|
* if the stack is not cleaned up. If so, we can do Stack Pollution
|
|
* more globally.
|
|
*/
|
|
|
|
STATIC int globl_sp_allowed;
|
|
|
|
|
|
#define IS_ASP(l) (INSTR(l) == op_asp && TYPE(l) == OPSHORT && SHORT(l) > 0)
|
|
|
|
|
|
STATIC void sp_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,"%%SP") == 0)break;
|
|
}
|
|
fscanf(f,"%d",&globl_sp_allowed);
|
|
}
|
|
|
|
STATIC void comb_asps(line_p l1, line_p l2, bblock_p b)
|
|
{
|
|
assert(INSTR(l1) == op_asp);
|
|
assert(INSTR(l2) == op_asp);
|
|
assert(TYPE(l1) == OPSHORT);
|
|
assert(TYPE(l2) == OPSHORT);
|
|
|
|
SHORT(l2) += SHORT(l1);
|
|
rm_line(l1,b);
|
|
}
|
|
|
|
STATIC void stack_pollution(bblock_p b)
|
|
{
|
|
/* For every pair of successive ASP instructions in basic
|
|
* block b, try to combine the two into one ASP.
|
|
*/
|
|
|
|
register line_p l;
|
|
line_p asp,next = b->b_start;
|
|
bool asp_seen = FALSE;
|
|
int stack_diff,pop,push;
|
|
bool ok;
|
|
|
|
do {
|
|
stack_diff = 0;
|
|
for (l = next; l != (line_p) 0; l = next) {
|
|
next = l->l_next;
|
|
if (IS_ASP(l)) break;
|
|
if (asp_seen) {
|
|
if (INSTR(l) == op_ret) {
|
|
stack_diff -= SHORT(l);
|
|
} else {
|
|
line_change(l,&ok,&pop,&push);
|
|
if (!ok || (stack_diff -= pop) < 0) {
|
|
/* can't eliminate last ASP */
|
|
asp_seen = FALSE;
|
|
} else {
|
|
stack_diff += push;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (asp_seen) {
|
|
if (l == (line_p) 0) {
|
|
/* last asp of basic block */
|
|
if (globl_sp_allowed &&
|
|
NOT_MARKED(b) && !IN_LOOP(b)) {
|
|
Ssp++;
|
|
rm_line(asp,b);
|
|
}
|
|
} else {
|
|
/* try to combine with previous asp */
|
|
if (SHORT(l) == stack_diff) {
|
|
Ssp++;
|
|
comb_asps(asp,l,b);
|
|
}
|
|
}
|
|
}
|
|
asp = l;
|
|
asp_seen = TRUE; /* use new ASP for next try! */
|
|
} while (asp != (line_p) 0);
|
|
}
|
|
|
|
STATIC bool block_save(bblock_p b)
|
|
{
|
|
|
|
register line_p l;
|
|
int stack_diff,pop,push;
|
|
bool ok;
|
|
|
|
stack_diff = 0;
|
|
for (l = b->b_start; l != (line_p) 0; l = l->l_next) {
|
|
if (INSTR(l) == op_ret) {
|
|
stack_diff -= SHORT(l);
|
|
break;
|
|
}
|
|
line_change(l,&ok,&pop,&push);
|
|
/* printf("instr %d, pop %d,push %d,ok %d\n",INSTR(l),pop,push,ok); */
|
|
if (!ok || (stack_diff -= pop) < 0) {
|
|
return FALSE;
|
|
} else {
|
|
stack_diff += push;
|
|
}
|
|
}
|
|
return stack_diff >= 0;
|
|
}
|
|
|
|
STATIC void mark_pred(bblock_p b)
|
|
{
|
|
Lindex i;
|
|
bblock_p x;
|
|
|
|
for (i = Lfirst(b->b_pred); i != (Lindex) 0; i = Lnext(i,b->b_pred)) {
|
|
x = (bblock_p) Lelem(i);
|
|
if (NOT_MARKED(x)) {
|
|
MARK(x);
|
|
mark_pred(x);
|
|
}
|
|
}
|
|
}
|
|
|
|
STATIC void mark_unsave_blocks(proc_p p)
|
|
{
|
|
register bblock_p b;
|
|
|
|
for (b = p->p_start; b != (bblock_p) 0; b = b->b_next) {
|
|
if (NOT_MARKED(b) && !block_save(b)) {
|
|
MARK(b);
|
|
mark_pred(b);
|
|
}
|
|
}
|
|
}
|
|
|
|
STATIC void sp_optimize(void *vp)
|
|
{
|
|
proc_p p = vp;
|
|
register bblock_p b;
|
|
|
|
if (IS_ENTERED_WITH_GTO(p)) return;
|
|
mark_unsave_blocks(p);
|
|
for (b = p->p_start; b != 0; b = b->b_next) {
|
|
stack_pollution(b);
|
|
}
|
|
}
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
go(argc,argv,no_action,sp_optimize,sp_machinit,no_action);
|
|
report("stack adjustments deleted",Ssp);
|
|
exit(0);
|
|
}
|
|
|
|
/***** DEBUGGING:
|
|
|
|
debug_stack_pollution(p)
|
|
proc_p p;
|
|
{
|
|
register bblock_p b;
|
|
register line_p l;
|
|
int lcnt,aspcnt,instr;
|
|
|
|
for (b = p->p_start; b != 0; b = b->b_next) {
|
|
lcnt = 0; aspcnt = 0;
|
|
for (l = b->b_start; l != 0; l= l->l_next) {
|
|
instr = INSTR(l);
|
|
if (instr >= sp_fmnem && instr <= sp_lmnem) {
|
|
lcnt++;
|
|
if (instr == op_asp && off_set(l) > 0) {
|
|
aspcnt++;
|
|
}
|
|
}
|
|
}
|
|
printf("%d\t%d\n",aspcnt,lcnt);
|
|
}
|
|
}
|
|
|
|
*/
|