roto 0.10.0

a statically-typed, compiled, embedded scripting language
Documentation
use super::{Block, Function, Instruction, Lir, Operand, Var, VarKind};
use crate::{
    ir_printer::{IrPrinter, Printable},
    lir::ValueOrSlot,
    typechecker::scoped_display::TypeDisplay,
};

impl Printable for Lir {
    fn print(&self, printer: &IrPrinter) -> String {
        let strings: Vec<_> =
            self.functions.iter().map(|f| f.print(printer)).collect();
        strings.join("\n")
    }
}

impl Printable for Function {
    fn print(&self, printer: &IrPrinter) -> String {
        use std::fmt::Write;
        let mut s = String::new();

        let printer = IrPrinter {
            scope: Some(self.scope),
            type_info: printer.type_info,
            label_store: printer.label_store,
        };

        let _ = write!(
            &mut s,
            "fn {}({}) {}{{",
            self.name,
            self.ir_signature
                .parameters
                .iter()
                .map(|(a, t)| {
                    let a = a.print(&printer);
                    let t = t.display(printer.type_info);
                    format!("{a}: {t}")
                })
                .collect::<Vec<_>>()
                .join(", "),
            self.ir_signature
                .return_type
                .map(|t| {
                    let t = t.display(printer.type_info);
                    format!("-> {t} ")
                })
                .unwrap_or_default(),
        );

        s.push('\n');
        for (var, ty) in &self.variables {
            let var = var.print(&printer);
            let ty = match ty {
                ValueOrSlot::Val(ir_type) => ir_type.to_string(),
                ValueOrSlot::StackSlot(layout) => {
                    format!(
                        "Pointer = Slot(size={}, align={})",
                        layout.size(),
                        layout.align()
                    )
                }
            };
            let _ = writeln!(&mut s, "  {var}: {ty}");
        }

        for b in &self.blocks {
            writeln!(s).unwrap();
            for line in b.print(&printer).lines() {
                let _ = writeln!(&mut s, "  {line}");
            }
        }

        s.push_str("}\n\n");

        s
    }
}

impl Printable for Block {
    fn print(&self, printer: &IrPrinter) -> String {
        use std::fmt::Write;
        let mut s = String::new();
        writeln!(s, ".{}", &self.label.print(printer)).unwrap();
        for i in &self.instructions {
            writeln!(s, "  {}", i.print(printer)).unwrap();
        }
        s
    }
}

impl Printable for Var {
    fn print(&self, printer: &IrPrinter) -> String {
        let name = match &self.kind {
            VarKind::Explicit(name) => name.print(printer).to_string(),
            VarKind::Tmp(idx) => format!("${idx}"),
            VarKind::Return => "$return".to_string(),
            VarKind::Context => "$context".to_string(),
        };
        if Some(self.scope) != printer.scope {
            let f = self.scope.print(printer);
            format!("{}.{name}", f,)
        } else {
            name
        }
    }
}

impl Printable for Instruction {
    fn print(&self, printer: &IrPrinter) -> String {
        use Instruction::*;
        match self {
            Assign { to, val, ty } => {
                format!(
                    "{}: {ty} = {}",
                    to.print(printer),
                    val.print(printer),
                )
            }
            ConstantAddress { to, name } => {
                format!(
                    "{} = ConstantAddress(\"{}\")",
                    to.print(printer),
                    name.print(printer),
                )
            }
            FunctionAddress { to, name } => {
                format!(
                    "{} = FunctionAddress(\"{}\")",
                    to.print(printer),
                    name.print(printer),
                )
            }
            Call {
                to,
                ctx,
                func,
                args,
                return_ptr,
            } => format!(
                "{}{}({}{}{})",
                to.as_ref()
                    .map(|(to, ty)| format!(
                        "{}: {} = ",
                        to.print(printer),
                        ty
                    ))
                    .unwrap_or_default(),
                func.print(printer),
                return_ptr
                    .as_ref()
                    .map(|p| format!("{} ,", p.print(printer)))
                    .unwrap_or_default(),
                ctx.as_ref()
                    .map(|c| format!("{} ,", c.print(printer)))
                    .unwrap_or_default(),
                args.iter()
                    .map(|a| a.print(printer))
                    .collect::<Vec<_>>()
                    .join(", ")
            ),
            CallRuntime { func, args } => format!(
                "<rust function {:?}>({})",
                func,
                args.iter()
                    .map(|a| a.print(printer))
                    .collect::<Vec<_>>()
                    .join(", ")
            ),
            InitString {
                to,
                string,
                init_func: _,
            } => {
                format!(
                    "{}: String = InitString(\"{string}\")",
                    to.print(printer),
                )
            }
            Return(None) => "return".to_string(),
            Return(Some(v)) => {
                format!("return {}", v.print(printer))
            }
            IntCmp {
                to,
                cmp,
                left,
                right,
            } => {
                format!(
                    "{} = {cmp}({}, {})",
                    to.print(printer),
                    left.print(printer),
                    right.print(printer),
                )
            }
            FloatCmp {
                to,
                cmp,
                left,
                right,
            } => {
                format!(
                    "{} = {cmp}({}, {})",
                    to.print(printer),
                    left.print(printer),
                    right.print(printer),
                )
            }
            Not { to, val } => {
                format!("{} = not({})", to.print(printer), val.print(printer))
            }
            Negate { to, val } => {
                format!(
                    "{} = negate({})",
                    to.print(printer),
                    val.print(printer)
                )
            }
            Add { to, left, right } => {
                format!(
                    "{} = {} + {}",
                    to.print(printer),
                    left.print(printer),
                    right.print(printer),
                )
            }
            Sub { to, left, right } => {
                format!(
                    "{} = {} - {}",
                    to.print(printer),
                    left.print(printer),
                    right.print(printer),
                )
            }
            Mul { to, left, right } => {
                format!(
                    "{} = {} * {}",
                    to.print(printer),
                    left.print(printer),
                    right.print(printer),
                )
            }
            Div {
                to,
                signed,
                left,
                right,
            } => {
                format!(
                    "{}: = {} / {} ({})",
                    to.print(printer),
                    left.print(printer),
                    right.print(printer),
                    if *signed { "signed" } else { "unsigned" },
                )
            }
            FDiv { to, left, right } => {
                format!(
                    "{}: = {} / {} (float)",
                    to.print(printer),
                    left.print(printer),
                    right.print(printer),
                )
            }
            Jump(to) => {
                format!("jump {}", to.print(printer))
            }
            Switch {
                examinee,
                default,
                branches,
            } => {
                format!(
                    "switch {} [{}] else {}",
                    examinee.print(printer),
                    branches
                        .iter()
                        .map(|(i, b)| format!("{i} => {}", b.print(printer)))
                        .collect::<Vec<_>>()
                        .join(", "),
                    default.print(printer),
                )
            }
            Initialize { to, bytes, layout } => {
                format!(
                    "{} = mem::initialize([{}], size={}, align={})",
                    to.print(printer),
                    bytes
                        .iter()
                        .map(|b| b.to_string())
                        .collect::<Vec<_>>()
                        .join(", "),
                    layout.size(),
                    layout.align(),
                )
            }
            Offset { to, from, offset } => {
                format!(
                    "{} = ptr::offset({}, {offset})",
                    to.print(printer),
                    from.print(printer),
                )
            }
            Read { to, from, ty } => {
                format!(
                    "{}: {ty} = mem::read({})",
                    to.print(printer),
                    from.print(printer),
                )
            }
            Write { to, val } => {
                format!(
                    "mem::write({}, {})",
                    to.print(printer),
                    val.print(printer),
                )
            }
            Copy { to, from, size } => {
                format!(
                    "mem::copy({}, {}, {size})",
                    to.print(printer),
                    from.print(printer),
                )
            }
            Clone {
                to,
                from,
                clone_fn: clone,
            } => {
                format!(
                    "mem::clone({}, {}, with={clone:?})",
                    to.print(printer),
                    from.print(printer),
                )
            }
            Drop {
                var,
                drop: Some(drop),
            } => {
                format!("mem::drop({}, with={drop:?})", var.print(printer))
            }
            Drop { var, drop: None } => {
                format!("mem::drop({}, noop)", var.print(printer))
            }
        }
    }
}

impl Printable for Operand {
    fn print(&self, printer: &IrPrinter) -> String {
        match self {
            Operand::Place(x) => x.print(printer),
            Operand::Value(x) => x.to_string(),
        }
    }
}