Files
hakorune/src/mir/optimizer_passes/boxfield.rs

86 lines
3.5 KiB
Rust

use crate::ast::Span;
use crate::mir::optimizer::MirOptimizer;
use crate::mir::optimizer_stats::OptimizationStats;
use crate::mir::{MirInstruction as I, MirModule};
/// Optimize BoxField operations (scaffolding)
pub fn optimize_boxfield_operations(
opt: &mut MirOptimizer,
module: &mut MirModule,
) -> OptimizationStats {
let mut stats = OptimizationStats::new();
for (func_name, function) in &mut module.functions {
if opt.debug_enabled() {
println!(" 📦 BoxField optimization in function: {}", func_name);
}
for (_bb_id, block) in &mut function.blocks {
let mut changed = 0usize;
let mut out: Vec<I> = Vec::with_capacity(block.instructions.len());
let mut out_spans: Vec<Span> = Vec::with_capacity(block.instruction_spans.len());
let mut i = 0usize;
while i < block.instructions.len() {
// Get span for current instruction (or unknown if missing)
let span_i = block
.instruction_spans
.get(i)
.copied()
.unwrap_or_else(Span::unknown);
// Look ahead for simple store-followed-by-load on same box/index
if i + 1 < block.instructions.len() {
let span_i1 = block
.instruction_spans
.get(i + 1)
.copied()
.unwrap_or_else(Span::unknown);
match (&block.instructions[i], &block.instructions[i + 1]) {
(
I::BoxCall {
box_val: b1,
method: m1,
args: a1,
..
},
I::BoxCall {
dst: Some(dst2),
box_val: b2,
method: m2,
args: a2,
..
},
) if (m1 == "set" || m1 == "setField")
&& (m2 == "get" || m2 == "getField") =>
{
// set(arg0=index/key, arg1=value), then get(arg0=index/key)
if b1 == b2 && a1.len() == 2 && a2.len() == 1 && a1[0] == a2[0] {
// Replace the second with Copy from just-stored value
let src_val = a1[1];
out.push(block.instructions[i].clone());
out_spans.push(span_i);
out.push(I::Copy {
dst: *dst2,
src: src_val,
});
out_spans.push(span_i1);
changed += 1;
i += 2;
continue;
}
}
_ => {}
}
}
out.push(block.instructions[i].clone());
out_spans.push(span_i);
i += 1;
}
if changed > 0 {
block.instructions = out;
block.instruction_spans = out_spans;
stats.boxfield_optimizations += changed;
}
}
}
stats
}