shithub: mc

Download patch

ref: 48996257e5e74a72258db974a03ba8e40ce9287f
parent: ad438bbebc8bc26ec441150577792dde4e729558
author: Ori Bernstein <[email protected]>
date: Sun Aug 23 13:31:40 EDT 2015

Clean up remapping code.

--- a/6/ra.c
+++ b/6/ra.c
@@ -999,19 +999,12 @@
     return spilled;
 }
 
-typedef struct Remapping Remapping;
-struct Remapping {
-    regid oldreg;
-    Loc *newreg;
-};
-
-static Loc *mapfind(Isel *s, Loc *old, Remapping *r, size_t nr)
+static Loc *mapfind(Isel *s, Htab *map, Loc *old)
 {
     Loc *new;
     Loc *base;
     Loc *idx;
     regid id;
-    size_t i;
 
     if (!old)
         return NULL;
@@ -1019,20 +1012,16 @@
     new = NULL;
     if (old->type == Locreg) {
         id = getalias(s, old->reg.id);
-        for (i = 0; i < nr; i++) {
-            if (id == r[i].oldreg) {
-                return r[i].newreg;
-            }
-        }
+        new = htget(map, locmap[id]);
     } else if (old->type == Locmem || old->type == Locmeml) {
-        base = NULL;
-        idx = NULL;
-        if (old->mem.base)
-            base = locmap[getalias(s, old->mem.base->reg.id)];
-        if (old->mem.idx)
-            idx = locmap[getalias(s, old->mem.idx->reg.id)];
-        base = mapfind(s, base, r, nr);
-        idx = mapfind(s, idx, r, nr);
+        base = old->mem.base;
+        idx = old->mem.idx;
+        if (base)
+            base = locmap[getalias(s, base->reg.id)];
+        if (idx)
+            idx = locmap[getalias(s, idx->reg.id)];
+        base = mapfind(s, map, base);
+        idx = mapfind(s, map, idx);
         if (base != old->mem.base || idx != old->mem.idx) {
             if (old->type == Locmem)
                 new = locmems(old->mem.constdisp, base, idx, old->mem.scale, old->mode);
@@ -1039,9 +1028,9 @@
             else
                 new = locmemls(old->mem.lbldisp, base, idx, old->mem.scale, old->mode);
         }
-        if (new)
-            return new;
     }
+    if (new)
+        return new;
     return old;
 }
 
@@ -1053,13 +1042,13 @@
     return locmem(-stkoff, locphysreg(Rrbp), NULL, locmap[reg]->mode);
 }
 
-static void updatelocs(Isel *s, Insn *insn, Remapping *use, size_t nuse, Remapping *def, size_t ndef)
+static void updatelocs(Isel *s, Htab *map, Insn *insn)
 {
     size_t i;
 
     for (i = 0; i < insn->nargs; i++) {
-        insn->args[i] = mapfind(s, insn->args[i], use, nuse);
-        insn->args[i] = mapfind(s, insn->args[i], def, ndef);
+        insn->args[i] = mapfind(s, map, insn->args[i]);
+        insn->args[i] = mapfind(s, map, insn->args[i]);
     }
 }
 
@@ -1068,58 +1057,37 @@
  * and fills them, storign the number of uses or defs. Returns
  * whether there are any remappings at all.
  */
-static int remap(Isel *s, Insn *insn, Remapping *use, size_t *nuse, Remapping *def, size_t *ndef)
+static int remap(Isel *s, Htab *map, Insn *insn, regid *use, size_t nuse, regid *def, size_t ndef)
 {
-    regid u[Nreg], d[Nreg];
     regid ruse, rdef;
-    size_t nu, nd;
-    size_t useidx, defidx;
-    size_t i, j, k;
-    int found;
+    int remapped;
+    Loc *tmp;
+    size_t i;
 
-    useidx = 0;
-    nu = uses(insn, u);
-    nd = defs(insn, d);
-    for (i = 0; i < nu; i++) {
-        ruse = getalias(s, u[i]);
+    remapped = 0;
+    for (i = 0; i < nuse; i++) {
+        ruse = getalias(s, use[i]);
         if (!bshas(s->spilled, ruse))
             continue;
-        use[useidx].oldreg = ruse;
-        use[useidx].newreg = locreg(locmap[ruse]->mode);
-        bsput(s->neverspill, use[useidx].newreg->reg.id);
-        useidx++;
+        tmp = locreg(locmap[ruse]->mode);
+        htput(map, locmap[ruse], tmp);
+        bsput(s->neverspill, tmp->reg.id);
+        remapped = 1;
     }
 
-    defidx = 0; 
-    for (i = 0; i < nd; i++) {
-        rdef = getalias(s, d[i]);
+    for (i = 0; i < ndef; i++) {
+        rdef = getalias(s, def[i]);
         if (!bshas(s->spilled, rdef))
             continue;
-        def[defidx].oldreg = rdef;
-
-        /* if we already have remapped a use for this register, we want to
-         * store the same register from the def. */
-        found = 0;
-        for (j = 0; j <= defidx; j++) {
-            for (k = 0; k < useidx; k++) {
-                if (use[k].oldreg == getalias(s, d[j])) {
-                    def[defidx].newreg = use[j].newreg;
-                    bsput(s->neverspill, def[defidx].newreg->reg.id);
-                    found = 1;
-                }
-            }
-        }
-        if (!found) {
-            def[defidx].newreg = locreg(locmap[rdef]->mode);
-            bsput(s->neverspill, def[defidx].newreg->reg.id);
-        }
-
-        defidx++;
+        if (hthas(map, locmap[rdef]))
+            continue;
+        tmp = locreg(locmap[rdef]->mode);
+        htput(map, locmap[rdef], tmp);
+        bsput(s->neverspill, tmp->reg.id);
+        remapped = 1;
     }
 
-    *nuse = useidx;
-    *ndef = defidx;
-    return useidx > 0 || defidx > 0;
+    return remapped;
 }
 
 static int nopmov(Insn *insn)
@@ -1151,60 +1119,92 @@
     }
 }
 
+static ulong reglochash(void *p)
+{
+    Loc *l;
+    
+    l = p;
+    return inthash(l->reg.id);
+}
+
+
+static int regloceq(void *pa, void *pb)
+{
+    Loc *a, *b;
+    
+    a = pa;
+    b = pb;
+    return a->reg.id == b->reg.id;
+}
 /*
  * Rewrite instructions using spilled registers, inserting
  * appropriate loads and stores into the BB
  */
-static void rewritebb(Isel *s, Asmbb *bb, Loc **map)
+static void rewritebb(Isel *s, Asmbb *bb, Loc **aliasmap)
 {
-    Remapping use[Nreg], def[Nreg];
-    Insn *insn, *mov;
+    regid use[Nreg], def[Nreg];
     size_t nuse, ndef;
+    Insn *insn, *mov;
     size_t i, j;
     Insn **new;
     size_t nnew;
+    Htab *map;
+    Loc *tmp;
 
     new = NULL;
     nnew = 0;
     if (!bb)
         return;
+    map = mkht(reglochash, regloceq);
     for (j = 0; j < bb->ni; j++) {
         insn = bb->il[j];
-        replacealias(s, map, s->nreg, insn);
+        replacealias(s, aliasmap, s->nreg, insn);
         if (nopmov(insn))
             continue;
+        nuse = uses(insn, use);
+        ndef = defs(insn, def);
         /* if there is a remapping, insert the loads and stores as needed */
-        if (remap(s, insn, use, &nuse, def, &ndef)) {
+        if (remap(s, map, insn, use, nuse, def, ndef)) {
             for (i = 0; i < nuse; i++) {
-                if (isfloatmode(use[i].newreg->mode))
-                    mov = mkinsn(Imovs, spillslot(s, use[i].oldreg), use[i].newreg, NULL);
+                tmp = htget(map, locmap[use[i]]);
+                if (!tmp)
+                    continue;
+                if (isfloatmode(tmp->mode))
+                    mov = mkinsn(Imovs, spillslot(s, use[i]), tmp, NULL);
                 else
-                    mov = mkinsn(Imov, spillslot(s, use[i].oldreg), use[i].newreg, NULL);
+                    mov = mkinsn(Imov, spillslot(s, use[i]), tmp, NULL);
                 lappend(&new, &nnew, mov);
                 if (debugopt['r']) {
                     printf("loading ");
-                    dbglocprint(stdout, locmap[use[i].oldreg], 'x');
+                    dbglocprint(stdout, locmap[use[i]], 'x');
                     printf(" -> ");
-                    dbglocprint(stdout, use[i].newreg, 'x');
+                    dbglocprint(stdout, tmp, 'x');
                     printf("\n");
                 }
             }
-            updatelocs(s, insn, use, nuse, def, ndef);
+            updatelocs(s, map, insn);
             lappend(&new, &nnew, insn);
             for (i = 0; i < ndef; i++) {
-                if (isfloatmode(def[i].newreg->mode))
-                    mov = mkinsn(Imovs, def[i].newreg, spillslot(s, def[i].oldreg), NULL);
+                tmp = htget(map, locmap[def[i]]);
+                if (!tmp)
+                    continue;
+                if (isfloatmode(tmp->mode))
+                    mov = mkinsn(Imovs, tmp, spillslot(s, def[i]), NULL);
                 else
-                    mov = mkinsn(Imov, def[i].newreg, spillslot(s, def[i].oldreg), NULL);
+                    mov = mkinsn(Imov, tmp, spillslot(s, def[i]), NULL);
                 lappend(&new, &nnew, mov);
                 if (debugopt['r']) {
                     printf("storing ");
-                    dbglocprint(stdout, locmap[def[i].oldreg], 'x');
+                    dbglocprint(stdout, locmap[def[i]], 'x');
                     printf(" -> ");
-                    dbglocprint(stdout, def[i].newreg, 'x');
+                    dbglocprint(stdout, tmp, 'x');
                     printf("\n");
                 }
             }
+            for (i = 0; i < nuse; i++)
+                htdel(map, locmap[use[i]]);
+            for (i = 0; i < ndef; i++)
+                htdel(map, locmap[def[i]]);
         } else {
             lappend(&new, &nnew, insn);
         }