summaryrefslogtreecommitdiff
path: root/util/romcc/romcc.c
diff options
context:
space:
mode:
Diffstat (limited to 'util/romcc/romcc.c')
-rw-r--r--util/romcc/romcc.c195
1 files changed, 160 insertions, 35 deletions
diff --git a/util/romcc/romcc.c b/util/romcc/romcc.c
index 5c784fc5e6..1f5c91af82 100644
--- a/util/romcc/romcc.c
+++ b/util/romcc/romcc.c
@@ -14,7 +14,7 @@
#define DEBUG_ERROR_MESSAGES 0
#define DEBUG_COLOR_GRAPH 0
#define DEBUG_SCC 0
-#define DEBUG_CONSISTENCY 1
+#define DEBUG_CONSISTENCY 2
#warning "FIXME boundary cases with small types in larger registers"
#warning "FIXME give clear error messages about unused variables"
@@ -858,11 +858,7 @@ struct type {
#define MAX_REGISTERS 75
#define MAX_REG_EQUIVS 16
-#if 1
#define REGISTER_BITS 16
-#else
-#define REGISTER_BITS 28
-#endif
#define MAX_VIRT_REGISTERS (1<<REGISTER_BITS)
#define TEMPLATE_BITS 6
#define MAX_TEMPLATES (1<<TEMPLATE_BITS)
@@ -881,7 +877,6 @@ struct type {
#define REG_VIRT9 (MAX_REGISTERS + 5)
/* Provision for 8 register classes */
-#if 1
#define REG_SHIFT 0
#define REGC_SHIFT REGISTER_BITS
#define REGC_MASK (((1 << MAX_REGC) - 1) << REGISTER_BITS)
@@ -892,11 +887,6 @@ struct type {
#define SET_REGCM(ID, REGCM) ((ID) = (((ID) & ~REGC_MASK) | (((REGCM) << REGC_SHIFT) & REGC_MASK)))
#define SET_INFO(ID, INFO) ((ID) = (((ID) & ~(REG_MASK | REGC_MASK)) | \
(((INFO).reg) & REG_MASK) | ((((INFO).regcm) << REGC_SHIFT) & REGC_MASK)))
-#else
-#define REG_MASK (MAX_VIRT_REGISTERS -1)
-#define ID_REG(ID) ((ID) & REG_MASK)
-#define SET_REG(ID, REG) ((ID) = (((ID) & ~REG_MASK) | ((REG) & REG_MASK)))
-#endif
static unsigned arch_reg_regcm(struct compile_state *state, int reg);
static unsigned arch_regcm_normalize(struct compile_state *state, unsigned regcm);
@@ -933,7 +923,8 @@ static struct triple *transform_to_arch_instruction(
#define DEBUG_CODE_ELIMINATION 0x0200
#define DEBUG_INSERTED_COPIES 0x0400
-#define GLOBAL_SCOPE_DEPTH 1
+#define GLOBAL_SCOPE_DEPTH 1
+#define FUNCTION_SCOPE_DEPTH (GLOBAL_SCOPE_DEPTH + 1)
static void compile_file(struct compile_state *old_state, const char *filename, int local);
@@ -2169,6 +2160,22 @@ static void symbol(
*chain = sym;
}
+static void label_symbol(struct compile_state *state,
+ struct hash_entry *ident, struct triple *label)
+{
+ struct symbol *sym;
+ if (ident->sym_label) {
+ error(state, 0, "label %s already defined", ident->name);
+ }
+ sym = xcmalloc(sizeof(*sym), "label");
+ sym->ident = ident;
+ sym->def = label;
+ sym->type = &void_type;
+ sym->scope_depth = FUNCTION_SCOPE_DEPTH;
+ sym->next = 0;
+ ident->sym_label = sym;
+}
+
static void start_scope(struct compile_state *state)
{
state->scope_depth++;
@@ -7793,22 +7800,46 @@ static void continue_statement(struct compile_state *state, struct triple *first
static void goto_statement(struct compile_state *state, struct triple *first)
{
- FINISHME();
+ struct hash_entry *ident;
eat(state, TOK_GOTO);
eat(state, TOK_IDENT);
+ ident = state->token[0].ident;
+ if (!ident->sym_label) {
+ /* If this is a forward branch allocate the label now,
+ * it will be flattend in the appropriate location later.
+ */
+ struct triple *ins;
+ ins = label(state);
+ label_symbol(state, ident, ins);
+ }
eat(state, TOK_SEMI);
- error(state, 0, "goto is not implemeted");
- FINISHME();
+
+ flatten(state, first, branch(state, ident->sym_label->def, 0));
}
static void labeled_statement(struct compile_state *state, struct triple *first)
{
- FINISHME();
+ struct triple *ins;
+ struct hash_entry *ident;
eat(state, TOK_IDENT);
+
+ ident = state->token[0].ident;
+ if (ident->sym_label && ident->sym_label->def) {
+ ins = ident->sym_label->def;
+ put_occurance(ins->occurance);
+ ins->occurance = new_occurance(state);
+ }
+ else {
+ ins = label(state);
+ label_symbol(state, ident, ins);
+ }
+ if (ins->id & TRIPLE_FLAG_FLATTENED) {
+ error(state, 0, "label %s already defined", ident->name);
+ }
+ flatten(state, first, ins);
+
eat(state, TOK_COLON);
statement(state, first);
- error(state, 0, "labeled statements are not implemented");
- FINISHME();
}
static void switch_statement(struct compile_state *state, struct triple *first)
@@ -8845,6 +8876,30 @@ static struct triple *initializer(
return result;
}
+static void resolve_branches(struct compile_state *state)
+{
+ /* Make a second pass and finish anything outstanding
+ * with respect to branches. The only outstanding item
+ * is to see if there are goto to labels that have not
+ * been defined and to error about them.
+ */
+ int i;
+ for(i = 0; i < HASH_TABLE_SIZE; i++) {
+ struct hash_entry *entry;
+ for(entry = state->hash_table[i]; entry; entry = entry->next) {
+ struct triple *ins;
+ if (!entry->sym_label) {
+ continue;
+ }
+ ins = entry->sym_label->def;
+ if (!(ins->id & TRIPLE_FLAG_FLATTENED)) {
+ error(state, ins, "label `%s' used but not defined",
+ entry->name);
+ }
+ }
+ }
+}
+
static struct triple *function_definition(
struct compile_state *state, struct type *type)
{
@@ -8926,8 +8981,12 @@ static struct triple *function_definition(
/* Now get the actual function definition */
compound_statement(state, end);
+ /* Finish anything unfinished with branches */
+ resolve_branches(state);
+
/* Remove the parameter scope */
end_scope(state);
+
#if 0
fprintf(stdout, "\n");
loc(stdout, state, 0);
@@ -10781,6 +10840,9 @@ static struct triple *pre_copy(
*/
struct triple *in;
struct triple **expr;
+ if (ins->op == OP_PHI) {
+ internal_error(state, ins, "pre_copy on a phi?");
+ }
expr = &RHS(ins, index);
in = pre_triple(state, ins, OP_COPY, (*expr)->type, *expr, 0);
unuse_triple(*expr, ins);
@@ -11772,6 +11834,19 @@ static void remove_live_edges(struct reg_state *rstate, struct live_range *range
}
}
+static void transfer_live_edges(struct reg_state *rstate,
+ struct live_range *dest, struct live_range *src)
+{
+ struct live_range_edge *edge, *next;
+ for(edge = src->edges; edge; edge = next) {
+ struct live_range *other;
+ next = edge->next;
+ other = edge->node;
+ remove_live_edge(rstate, src, other);
+ add_live_edge(rstate, dest, other);
+ }
+}
+
/* Interference graph...
*
@@ -11887,7 +11962,7 @@ static struct live_range *coalesce_ranges(
fprintf(stderr, "lr2 pre\n");
}
#endif
-#if 1
+#if 0
fprintf(stderr, "coalesce color1(%p): %3d color2(%p) %3d\n",
lr1->defs->def,
lr1->color,
@@ -11898,6 +11973,12 @@ static struct live_range *coalesce_ranges(
lr1->classes = classes;
/* Append lr2 onto lr1 */
#warning "FIXME should this be a merge instead of a splice?"
+ /* This FIXME item applies to the correctness of live_range_end
+ * and to the necessity of making multiple passes of coalesce_live_ranges.
+ * A failure to find some coalesce opportunities in coaleace_live_ranges
+ * does not impact the correct of the compiler just the efficiency with
+ * which registers are allocated.
+ */
head = lr1->defs;
mid1 = lr1->defs->prev;
mid2 = lr2->defs;
@@ -11928,6 +12009,9 @@ static struct live_range *coalesce_ranges(
lr1->color = color;
lr1->classes = classes;
+ /* Keep the graph in sync by transfering the edges from lr2 to lr1 */
+ transfer_live_edges(rstate, lr1, lr2);
+
return lr1;
}
@@ -12505,6 +12589,7 @@ static void fix_tangles(struct compile_state *state,
struct reg_block *blocks, struct triple_reg_set *live,
struct reg_block *rb, struct triple *ins, void *arg)
{
+ int *tangles = arg;
struct triple *tangle;
do {
char used[MAX_REGISTERS];
@@ -12531,6 +12616,13 @@ static void fix_tangles(struct compile_state *state,
if (used[info.reg] < 2) {
continue;
}
+ /* Changing copies that feed into phi functions
+ * is incorrect.
+ */
+ if (set->member->use &&
+ (set->member->use->member->op == OP_PHI)) {
+ continue;
+ }
if (!tangle || tdominates(state, set->member, tangle)) {
tangle = set->member;
}
@@ -12538,6 +12630,7 @@ static void fix_tangles(struct compile_state *state,
/* If I have found a tangle resolve it */
if (tangle) {
struct triple *post_copy;
+ (*tangles)++;
post_copy = resolve_tangle(state, tangle);
if (post_copy) {
replace_block_use(state, blocks, tangle, post_copy);
@@ -12550,11 +12643,14 @@ static void fix_tangles(struct compile_state *state,
return;
}
-static void correct_tangles(
+static int correct_tangles(
struct compile_state *state, struct reg_block *blocks)
{
+ int tangles;
+ tangles = 0;
color_instructions(state);
- walk_variable_lifetimes(state, blocks, fix_tangles, 0);
+ walk_variable_lifetimes(state, blocks, fix_tangles, &tangles);
+ return tangles;
}
struct least_conflict {
@@ -13536,6 +13632,7 @@ static void cleanup_rstate(struct compile_state *state, struct reg_state *rstate
rstate->blocks = 0;
}
+static void verify_consistency(struct compile_state *state);
static void allocate_registers(struct compile_state *state)
{
struct reg_state rstate;
@@ -13547,8 +13644,13 @@ static void allocate_registers(struct compile_state *state)
do {
struct live_range **point, **next;
+ int tangles;
int coalesced;
+#if 0
+ fprintf(stderr, "pass: %d\n", rstate.passes);
+#endif
+
/* Restore ids */
ids_from_rstate(state, &rstate);
@@ -13562,30 +13664,42 @@ static void allocate_registers(struct compile_state *state)
walk_variable_lifetimes(
state, rstate.blocks, fix_coalesce_conflicts, 0);
- /* Fix two simultaneous uses of the same register */
- correct_tangles(state, rstate.blocks);
+ /* Fix two simultaneous uses of the same register.
+ * In a few pathlogical cases a partial untangle moves
+ * the tangle to a part of the graph we won't revisit.
+ * So we keep looping until we have no more tangle fixes
+ * to apply.
+ */
+ do {
+ tangles = correct_tangles(state, rstate.blocks);
+ } while(tangles);
if (state->debug & DEBUG_INSERTED_COPIES) {
printf("After resolve_tangles\n");
print_blocks(state, stdout);
print_control_flow(state);
}
-
+ verify_consistency(state);
/* Allocate and initialize the live ranges */
initialize_live_ranges(state, &rstate);
-
- do {
- /* Forget previous live range edge calculations */
- cleanup_live_edges(&rstate);
+ /* Note current doing coalescing in a loop appears to
+ * buys me nothing. The code is left this way in case
+ * there is some value in it. Or if a future bugfix
+ * yields some benefit.
+ */
+ do {
#if 0
fprintf(stderr, "coalescing\n");
#endif
+ /* Remove any previous live edge calculations */
+ cleanup_live_edges(&rstate);
+
/* Compute the interference graph */
walk_variable_lifetimes(
state, rstate.blocks, graph_ins, &rstate);
-
+
/* Display the interference graph if desired */
if (state->debug & DEBUG_INTERFERENCE) {
printf("\nlive variables by block\n");
@@ -13595,14 +13709,25 @@ static void allocate_registers(struct compile_state *state)
state, rstate.blocks,
print_interference_ins, &rstate);
}
-#if DEBUG_CONSISTENCY
- /* Verify the interference graph */
- walk_variable_lifetimes(
- state, rstate.blocks, verify_graph_ins, &rstate);
-#endif
coalesced = coalesce_live_ranges(state, &rstate);
+
+#if 0
+ fprintf(stderr, "coalesced: %d\n", coalesced);
+#endif
} while(coalesced);
+
+#if DEBUG_CONSISTENCY > 1
+# if 0
+ fprintf(stderr, "verify_graph_ins...\n");
+# endif
+ /* Verify the interference graph */
+ walk_variable_lifetimes(
+ state, rstate.blocks, verify_graph_ins, &rstate);
+# if 0
+ fprintf(stderr, "verify_graph_ins done\n");
+#endif
+#endif
/* Build the groups low and high. But with the nodes
* first sorted by degree order.
@@ -14545,7 +14670,7 @@ static void verify_consistency(struct compile_state *state)
verify_ins_colors(state);
}
#else
-#define verify_consistency(state) do {} while(0)
+static void verify_consistency(struct compile_state *state) {}
#endif /* DEBUG_USES */
static void optimize(struct compile_state *state)