mirror of
https://github.com/janet-lang/janet
synced 2025-01-25 06:36:52 +00:00
More work on sysir.
This commit is contained in:
parent
25b7c74089
commit
3e273ce03a
@ -27,6 +27,10 @@
|
|||||||
#include "vector.h"
|
#include "vector.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Wrap stuff up in a context struct
|
||||||
|
*/
|
||||||
|
|
||||||
static uint32_t v2reg(JanetTable *assignments, uint32_t var) {
|
static uint32_t v2reg(JanetTable *assignments, uint32_t var) {
|
||||||
return (uint32_t) janet_unwrap_number(janet_table_get(assignments, janet_wrap_number(var)));
|
return (uint32_t) janet_unwrap_number(janet_table_get(assignments, janet_wrap_number(var)));
|
||||||
}
|
}
|
||||||
@ -167,23 +171,8 @@ JanetSysSpill *assign_registers(JanetSysIR *ir, JanetTable *assignments,
|
|||||||
return spills;
|
return spills;
|
||||||
}
|
}
|
||||||
|
|
||||||
void janet_sys_ir_lower_to_x64(JanetSysIRLinkage *linkage, JanetBuffer *buffer) {
|
static void do_spills(JanetBuffer *buffer, JanetSysSpill *spills, uint32_t index) {
|
||||||
|
JanetSysSpill spill = spills[index];
|
||||||
/* Do register allocation */
|
|
||||||
for (int32_t i = 0; i < linkage->ir_ordered->count; i++) {
|
|
||||||
JanetSysIR *ir = janet_unwrap_pointer(linkage->ir_ordered->data[i]);
|
|
||||||
JanetTable *assignments = janet_table(0);
|
|
||||||
JanetSysSpill *spills = assign_registers(ir, assignments, 15);
|
|
||||||
|
|
||||||
/* Emit prelude */
|
|
||||||
if (ir->link_name != NULL) {
|
|
||||||
janet_formatb(buffer, ".%s\n", ir->link_name);
|
|
||||||
} else {
|
|
||||||
janet_formatb(buffer, "._section_%d\n", i);
|
|
||||||
}
|
|
||||||
for (uint32_t j = 0; j < ir->instruction_count; j++) {
|
|
||||||
JanetSysInstruction instruction = ir->instructions[j];
|
|
||||||
JanetSysSpill spill = spills[j];
|
|
||||||
for (int spi = 0; spi < 3; spi++) {
|
for (int spi = 0; spi < 3; spi++) {
|
||||||
if (spill.spills[spi] == JANET_SYS_SPILL_READ || spill.spills[spi] == JANET_SYS_SPILL_BOTH) {
|
if (spill.spills[spi] == JANET_SYS_SPILL_READ || spill.spills[spi] == JANET_SYS_SPILL_BOTH) {
|
||||||
// emit load
|
// emit load
|
||||||
@ -198,6 +187,25 @@ void janet_sys_ir_lower_to_x64(JanetSysIRLinkage *linkage, JanetBuffer *buffer)
|
|||||||
janet_formatb(buffer, "store r%u to %v ; SPILL\n", reg, janet_wrap_pointer(x));
|
janet_formatb(buffer, "store r%u to %v ; SPILL\n", reg, janet_wrap_pointer(x));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void janet_sys_ir_lower_to_x64(JanetSysIRLinkage *linkage, JanetBuffer *buffer) {
|
||||||
|
|
||||||
|
/* Do register allocation */
|
||||||
|
for (int32_t i = 0; i < linkage->ir_ordered->count; i++) {
|
||||||
|
JanetSysIR *ir = janet_unwrap_pointer(linkage->ir_ordered->data[i]);
|
||||||
|
JanetTable *assignments = janet_table(0);
|
||||||
|
/* 16 total 64 bit registers - 3 temp */
|
||||||
|
JanetSysSpill *spills = assign_registers(ir, assignments, 13);
|
||||||
|
|
||||||
|
/* Emit prelude */
|
||||||
|
if (ir->link_name != NULL) {
|
||||||
|
janet_formatb(buffer, ".%s\n", ir->link_name);
|
||||||
|
} else {
|
||||||
|
janet_formatb(buffer, "._section_%d\n", i);
|
||||||
|
}
|
||||||
|
for (uint32_t j = 0; j < ir->instruction_count; j++) {
|
||||||
|
JanetSysInstruction instruction = ir->instructions[j];
|
||||||
switch (instruction.opcode) {
|
switch (instruction.opcode) {
|
||||||
default:
|
default:
|
||||||
janet_formatb(buffer, "; nyi: %s\n", janet_sysop_names[instruction.opcode]);
|
janet_formatb(buffer, "; nyi: %s\n", janet_sysop_names[instruction.opcode]);
|
||||||
@ -216,6 +224,7 @@ void janet_sys_ir_lower_to_x64(JanetSysIRLinkage *linkage, JanetBuffer *buffer)
|
|||||||
case JANET_SYSOP_SUBTRACT:
|
case JANET_SYSOP_SUBTRACT:
|
||||||
case JANET_SYSOP_MULTIPLY:
|
case JANET_SYSOP_MULTIPLY:
|
||||||
case JANET_SYSOP_DIVIDE:
|
case JANET_SYSOP_DIVIDE:
|
||||||
|
do_spills(buffer, spills, j);
|
||||||
janet_formatb(buffer, "r%u = %s r%u, r%u\n",
|
janet_formatb(buffer, "r%u = %s r%u, r%u\n",
|
||||||
v2reg(assignments, instruction.three.dest),
|
v2reg(assignments, instruction.three.dest),
|
||||||
janet_sysop_names[instruction.opcode],
|
janet_sysop_names[instruction.opcode],
|
||||||
@ -223,24 +232,50 @@ void janet_sys_ir_lower_to_x64(JanetSysIRLinkage *linkage, JanetBuffer *buffer)
|
|||||||
v2reg(assignments, instruction.three.rhs));
|
v2reg(assignments, instruction.three.rhs));
|
||||||
break;
|
break;
|
||||||
case JANET_SYSOP_MOVE:
|
case JANET_SYSOP_MOVE:
|
||||||
|
do_spills(buffer, spills, j);
|
||||||
janet_formatb(buffer, "r%u = r%u\n",
|
janet_formatb(buffer, "r%u = r%u\n",
|
||||||
v2reg(assignments, instruction.two.dest),
|
v2reg(assignments, instruction.two.dest),
|
||||||
v2reg(assignments, instruction.two.src));
|
v2reg(assignments, instruction.two.src));
|
||||||
break;
|
break;
|
||||||
case JANET_SYSOP_RETURN:
|
case JANET_SYSOP_RETURN:
|
||||||
|
do_spills(buffer, spills, j);
|
||||||
janet_formatb(buffer, "return r%u\n",
|
janet_formatb(buffer, "return r%u\n",
|
||||||
v2reg(assignments, instruction.one.src));
|
v2reg(assignments, instruction.one.src));
|
||||||
break;
|
break;
|
||||||
case JANET_SYSOP_CONSTANT:
|
case JANET_SYSOP_CONSTANT:
|
||||||
|
do_spills(buffer, spills, j);
|
||||||
janet_formatb(buffer, "r%u = constant $%v\n",
|
janet_formatb(buffer, "r%u = constant $%v\n",
|
||||||
v2reg(assignments, instruction.constant.dest),
|
v2reg(assignments, instruction.constant.dest),
|
||||||
ir->constants[instruction.constant.constant]);
|
ir->constants[instruction.constant.constant]);
|
||||||
break;
|
break;
|
||||||
case JANET_SYSOP_LABEL:
|
case JANET_SYSOP_LABEL:
|
||||||
janet_formatb(buffer, ":label_%u\n",
|
do_spills(buffer, spills, j);
|
||||||
|
janet_formatb(buffer, "label_%u:\n",
|
||||||
v2reg(assignments, instruction.label.id));
|
v2reg(assignments, instruction.label.id));
|
||||||
break;
|
break;
|
||||||
|
case JANET_SYSOP_EQ:
|
||||||
|
case JANET_SYSOP_NEQ:
|
||||||
|
case JANET_SYSOP_LT:
|
||||||
|
case JANET_SYSOP_LTE:
|
||||||
|
case JANET_SYSOP_GT:
|
||||||
|
case JANET_SYSOP_GTE:
|
||||||
|
do_spills(buffer, spills, j);
|
||||||
|
janet_formatb(buffer, "r%u = %s r%u, r%u\n",
|
||||||
|
v2reg(assignments, instruction.three.dest),
|
||||||
|
janet_sysop_names[instruction.opcode],
|
||||||
|
v2reg(assignments, instruction.three.lhs),
|
||||||
|
v2reg(assignments, instruction.three.rhs));
|
||||||
|
break;
|
||||||
|
case JANET_SYSOP_BRANCH:
|
||||||
|
case JANET_SYSOP_BRANCH_NOT:
|
||||||
|
do_spills(buffer, spills, j);
|
||||||
|
janet_formatb(buffer, "branch label_%u if r%u\n",
|
||||||
|
instruction.branch.to,
|
||||||
|
v2reg(assignments, instruction.branch.cond));
|
||||||
|
break;
|
||||||
|
case JANET_SYSOP_CALLK:
|
||||||
|
do_spills(buffer, spills, j);
|
||||||
|
break;
|
||||||
// On a comparison, if next instruction is branch that reads from dest, combine into a single op.
|
// On a comparison, if next instruction is branch that reads from dest, combine into a single op.
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user