ack/mach/proto/mcg/pass_instructionselection.c
David Given a3cfe6047f More rigorous dealing of IR groups; no need for is_generated and is_root any
more (but now passes are required to set IR roots properly when changing
instructions).
2016-10-01 22:58:29 +02:00

147 lines
3.2 KiB
C

#include "mcg.h"
static struct basicblock* current_bb;
static struct hop* current_hop;
static struct ir* current_ir;
static const struct burm_emitter_data emitter_data;
void burm_trace(struct ir* p, int ruleno, int cost, int bestcost) {
const struct burm_instruction_data* insndata = &burm_instruction_data[ruleno];
//tracef('I', "I: 0x%p matched %s with cost %d vs. %d\n", p,
// insndata->name, cost, bestcost);
}
void burm_panic_cannot_match(struct ir* ir)
{
fprintf(stderr, "could not find any patterns to match:\n");
ir_print(0, ir);
fprintf(stderr, "aborting!\n");
exit(1);
}
int burm_calculate_label(struct ir* ir)
{
if (ir->root != current_ir)
return ir_to_esn(IR_REG, ir->size);
return ir_to_esn(ir->opcode, ir->size);
}
static void emit_reg(struct ir* ir, int goal)
{
hop_add_reg_insel(current_hop, ir, goal);
}
static void emit_string(const char* data)
{
hop_add_string_insel(current_hop, data);
}
static void emit_fragment(struct ir* ir, int goal)
{
int insn_no = burm_rule(ir->state_label, goal);
const struct burm_instruction_data* insndata = &burm_instruction_data[insn_no];
if (insndata->emitter)
insndata->emitter(ir, &emitter_data);
}
static void emit_value(struct ir* ir)
{
hop_add_value_insel(current_hop, ir);
}
static void emit_resultreg(void)
{
}
static void emit_eoi(void)
{
hop_add_eoi_insel(current_hop);
}
static const struct burm_emitter_data emitter_data =
{
&emit_string,
&emit_fragment,
&emit_reg,
&emit_value,
&emit_resultreg,
&emit_eoi
};
static void walk_instructions(struct ir* ir, int goal)
{
struct ir* children[10];
int insn_no = burm_rule(ir->state_label, goal);
const struct burm_instruction_data* insndata = &burm_instruction_data[insn_no];
const short* nts = burm_nts[insn_no];
struct hop* parent_hop = NULL;
int i;
if (!insndata->is_fragment)
{
parent_hop = current_hop;
current_hop = new_hop(insn_no, ir);
}
burm_kids(ir, insn_no, children);
for (i=0; nts[i]; i++)
walk_instructions(children[i], nts[i]);
ir->insn_no = insn_no;
if (goal != 1)
ir->goal_no = goal;
tracef('I', "I: $%d goal %d selected %s %d: %s\n",
ir->id,
ir->goal_no,
insndata->is_fragment ? "fragment" : "instruction",
insn_no,
insndata->name);
if (!insndata->is_fragment)
{
if (insndata->emitter)
insndata->emitter(ir, &emitter_data);
hop_print('I', current_hop);
array_append(&ir->hops, current_hop);
current_hop = parent_hop;
}
}
static void select_instructions(void)
{
int i;
tracef('I', "I: BLOCK: %s\n", current_bb->name);
for (i=0; i<current_bb->irs.count; i++)
{
int insnno;
current_ir = current_bb->irs.item[i];
burm_label(current_ir);
insnno = burm_rule(current_ir->state_label, 1);
if (!insnno)
burm_panic_cannot_match(current_ir);
ir_print('I', current_ir);
walk_instructions(current_ir, 1);
}
}
void pass_instruction_selector(struct procedure* proc)
{
int i;
for (i=0; i<proc->blocks.count; i++)
{
current_bb = proc->blocks.item[i];
select_instructions();
}
}
/* vim: set sw=4 ts=4 expandtab : */