ref: e7ba66a1d2698910fcc0ba4aeadff491fb0ead94
dir: /mi/reaching.c/
#include <stdlib.h> #include <stdio.h> #include <inttypes.h> #include <stdarg.h> #include <ctype.h> #include <string.h> #include <assert.h> #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> #include <unistd.h> #include "util.h" #include "parse.h" #include "mi.h" Node *assignee(Node *n) { Node *a; switch (exprop(n)) { case Oundef: case Odef: case Oset: case Oasn: case Oaddeq: case Osubeq: case Omuleq: case Odiveq: case Omodeq: case Oboreq: case Obandeq: case Obxoreq: case Obsleq: case Obsreq: return n->expr.args[0]; break; case Oblit: case Oclear: a = n->expr.args[0]; if (exprop(a) != Oaddr) break; a = a->expr.args[0]; if (exprop(a) != Ovar) break; return a; default: break; } return NULL; } static void collectdefs(Cfg *cfg, size_t **defs, size_t *ndefs) { size_t i, j, did; Node *n; Bb *bb; for (i = 0; i < cfg->nbb; i++) { bb = cfg->bb[i]; if (!bb) continue; for (j = 0; j < bb->nnl; j++) { n = assignee(bb->nl[j]); if (n && exprop(n) == Ovar) { did = n->expr.did; ndefs[did]++; defs[did] = xrealloc(defs[did], ndefs[did] * sizeof(size_t)); defs[did][ndefs[did] - 1] = bb->nl[j]->nid; } } } } static void genkill(Bb *bb, size_t **defs, size_t *ndefs, Bitset *gen, Bitset *kill) { size_t i, j, did; Node *n; for (i = 0; i < bb->nnl; i++) { n = assignee(bb->nl[i]); if (!n) continue; did = n->expr.did; for (j = 0; j < ndefs[did]; j++) { bsput(kill, defs[did][j]); bsdel(gen, defs[did][j]); } bsput(gen, bb->nl[i]->nid); bsdel(kill, bb->nl[i]->nid); } } void bsdump(Bitset *bs) { size_t i; for (i = 0; bsiter(bs, &i); i++) printf("%zd ", i); printf("\n"); } Reaching *reaching(Cfg *cfg) { Bitset **in, **out; Bitset **gen, **kill; Bitset *bbin, *bbout; Reaching *reaching; size_t **defs; /* mapping from did => [def,list] */ size_t *ndefs; size_t i, j; int changed; in = zalloc(cfg->nbb * sizeof(Bb*)); out = zalloc(cfg->nbb * sizeof(Bb*)); gen = zalloc(cfg->nbb * sizeof(Bb*)); kill = zalloc(cfg->nbb * sizeof(Bb*)); defs = zalloc(ndecls * sizeof(size_t*)); ndefs = zalloc(ndecls * sizeof(size_t)); collectdefs(cfg, defs, ndefs); for (i = 0; i < cfg->nbb; i++) { in[i] = mkbs(); out[i] = mkbs(); gen[i] = mkbs(); kill[i] = mkbs(); if (cfg->bb[i]) genkill(cfg->bb[i], defs, ndefs, gen[i], kill[i]); } do { changed = 0; for (i = 0; i < cfg->nbb; i++) { if (!cfg->bb[i]) continue; bbin = mkbs(); for (j = 0; bsiter(cfg->bb[i]->pred, &j); j++) bsunion(bbin, out[j]); bbout = bsdup(bbin); bsdiff(bbout, kill[i]); bsunion(bbout, gen[i]); if (!bseq(out[i], bbout) || !bseq(in[i], bbin)) { changed = 1; bsfree(in[i]); bsfree(out[i]); in[i] = bbin; out[i] = bbout; } else { bsfree(bbin); bsfree(bbout); } } } while (changed); for (i = 0; i < cfg->nbb; i++) { bsfree(gen[i]); bsfree(kill[i]); } free(gen); free(kill); reaching = xalloc(sizeof(Reaching)); reaching->in = in; reaching->out = out; reaching->defs = defs; reaching->ndefs = ndefs; reaching->nbb = cfg->nbb; return reaching; } void reachingfree(Reaching *r) { size_t i; for (i = 0; i < r->nbb; i++) { bsfree(r->in[i]); bsfree(r->out[i]); } for (i = 0; i < ndecls; i++) free(r->defs[i]); free(r->in); free(r->out); free(r->defs); free(r->ndefs); free(r); }