206 lines
4 KiB
C
206 lines
4 KiB
C
|
/* S T R E N G T H R E D U C T I O N */
|
||
|
|
||
|
|
||
|
#include <stdio.h>
|
||
|
#include "../share/types.h"
|
||
|
#include "sr.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/go.h"
|
||
|
#include "sr_aux.h"
|
||
|
#include "sr_iv.h"
|
||
|
|
||
|
/* Strength reduction tries to change expensive operators occurring
|
||
|
* in a loop into cheaper operators. The expensive operators considered
|
||
|
* are multiplication and array referencing.
|
||
|
* The transformations can be expressed in C as:
|
||
|
*
|
||
|
* [1]: for (i = e1; i <= e2; i++)
|
||
|
* print(118*i);
|
||
|
* becomes:
|
||
|
* for (i = e1, t = 118*e1; i <= e2; i++, t += 118)
|
||
|
* print(t);
|
||
|
*
|
||
|
* [2]: for (i = e1; i <= e2; i++)
|
||
|
* print(a[i]);
|
||
|
* becomes:
|
||
|
* for (i = e1, p = &a[i]; i <= e2; i++, p++)
|
||
|
* print(*p);
|
||
|
* The latter optimization is suppressed if array bound checking
|
||
|
* is required.
|
||
|
*/
|
||
|
|
||
|
/* Machine and/or language dependent parameters: */
|
||
|
|
||
|
bool ovfl_harmful;
|
||
|
bool arrbound_harmful;
|
||
|
|
||
|
int Ssr; /* #optimizations found */
|
||
|
|
||
|
sr_machinit(f)
|
||
|
FILE *f;
|
||
|
{
|
||
|
/* Read target machine dependent information */
|
||
|
char s[100];
|
||
|
|
||
|
|
||
|
for (;;) {
|
||
|
while(getc(f) != '\n');
|
||
|
fscanf(f,"%s",s);
|
||
|
if (strcmp(s,"%%SR") == 0)break;
|
||
|
}
|
||
|
fscanf(f,"%d",&ovfl_harmful);
|
||
|
fscanf(f,"%d",&arrbound_harmful);
|
||
|
}
|
||
|
|
||
|
STATIC del_ivs(ivs)
|
||
|
lset ivs;
|
||
|
{
|
||
|
/* Delete the set of iv structs */
|
||
|
|
||
|
Lindex i;
|
||
|
|
||
|
for (i = Lfirst(ivs); i != (Lindex) 0; i = Lnext(i,ivs)) {
|
||
|
oldiv(Lelem(i));
|
||
|
}
|
||
|
Ldeleteset(ivs);
|
||
|
}
|
||
|
|
||
|
|
||
|
STATIC do_loop(loop)
|
||
|
loop_p loop;
|
||
|
{
|
||
|
lset ivs, vars;
|
||
|
|
||
|
OUTTRACE("going to process loop %d",loop->lp_id);
|
||
|
induc_vars(loop,&ivs, &vars);
|
||
|
/* Build a set of iv_structs, one for every induction
|
||
|
* variable of the loop, i.e. a variable i that
|
||
|
* is changed only by i := i + c, where c is a loop constant.
|
||
|
* Also detects variables that are changed (including induction
|
||
|
* variables!).
|
||
|
*/
|
||
|
OUTTRACE("loop has %d induction variables",Lnrelems(ivs));
|
||
|
if (Lnrelems(ivs) > 0) {
|
||
|
strength_reduction(loop,ivs,vars);
|
||
|
/* Perform strength reduction. Reduce:
|
||
|
* iv * c to addition
|
||
|
* a[iv] to indirection (*p)
|
||
|
* (unless array bound checking is required)
|
||
|
*/
|
||
|
}
|
||
|
del_ivs(ivs);
|
||
|
Ldeleteset(vars);
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
STATIC loopblocks(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 opt_proc(p)
|
||
|
proc_p p;
|
||
|
{
|
||
|
/* Optimize all loops of one procedure. We first do all
|
||
|
* outer loops at the lowest nesting level and proceed
|
||
|
* in the inwards direction.
|
||
|
*/
|
||
|
|
||
|
Lindex i;
|
||
|
loop_p lp,outermost;
|
||
|
int min_level;
|
||
|
|
||
|
for (;;) {
|
||
|
min_level = 1000;
|
||
|
for (i = Lfirst(p->p_loops); i != (Lindex) 0;
|
||
|
i = Lnext(i,p->p_loops)) {
|
||
|
lp = (loop_p) Lelem(i);
|
||
|
if (!lp->LP_DONE && lp->lp_level < min_level) {
|
||
|
min_level = lp->lp_level;
|
||
|
outermost = lp;
|
||
|
}
|
||
|
}
|
||
|
if (min_level == 1000) break;
|
||
|
do_loop(outermost);
|
||
|
outermost->LP_DONE = TRUE;
|
||
|
OUTTRACE("loop %d processed",outermost->lp_id);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
STATIC sr_extproc(p)
|
||
|
proc_p p;
|
||
|
{
|
||
|
/* Allocate the extended data structures for procedure p */
|
||
|
|
||
|
register loop_p lp;
|
||
|
register Lindex pi;
|
||
|
|
||
|
for (pi = Lfirst(p->p_loops); pi != (Lindex) 0;
|
||
|
pi = Lnext(pi,p->p_loops)) {
|
||
|
lp = (loop_p) Lelem(pi);
|
||
|
lp->lp_extend = newsrlpx();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
STATIC sr_cleanproc(p)
|
||
|
proc_p p;
|
||
|
{
|
||
|
/* Remove the extended data structures for procedure p */
|
||
|
|
||
|
register loop_p lp;
|
||
|
register Lindex pi;
|
||
|
|
||
|
|
||
|
for (pi = Lfirst(p->p_loops); pi != (Lindex) 0;
|
||
|
pi = Lnext(pi,p->p_loops)) {
|
||
|
lp = (loop_p) Lelem(pi);
|
||
|
oldsrlpx(lp->lp_extend);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
sr_optimize(p)
|
||
|
proc_p p;
|
||
|
{
|
||
|
sr_extproc(p);
|
||
|
loopblocks(p);
|
||
|
opt_proc(p);
|
||
|
sr_cleanproc(p);
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
main(argc,argv)
|
||
|
int argc;
|
||
|
char *argv[];
|
||
|
{
|
||
|
go(argc,argv,no_action,sr_optimize,sr_machinit,no_action);
|
||
|
report("strength reductions",Ssr);
|
||
|
exit(0);
|
||
|
}
|