miden-assembly 0.22.3

Miden VM assembly language
Documentation
use miden_assembly_syntax::{
    debuginfo::{SourceSpan, Spanned},
    diagnostics::{RelatedLabel, Report},
};
use miden_core::{Felt, WORD_SIZE, operations::Operation::*};

use super::{BasicBlockBuilder, push_u32_value};
use crate::{ProcedureContext, fmp::push_offset_fmp_sequence};

// INSTRUCTION PARSERS
// ================================================================================================

/// Appends operations to the span needed to execute a memory read instruction. This includes
/// reading a single element or an entire word from either local or global memory. Specifically,
/// this handles mem_load, mem_loadw_be, mem_loadw_le, loc_load, loc_loadw_be, and loc_loadw_le
/// instructions.
///
/// VM cycles per operation:
/// - mem_load(w): 1 cycle
/// - mem_load(w).b: 2 cycles
/// - loc_load(w).b:
///    - 4 cycles if b = 1
///    - 3 cycles if b != 1
///
/// # Errors
/// Returns an error if we are reading from local memory and local memory index is greater than
/// the number of procedure locals.
pub fn mem_read(
    block_builder: &mut BasicBlockBuilder,
    proc_ctx: &ProcedureContext,
    addr: Option<u32>,
    is_local: bool,
    is_single: bool,
    instr_span: SourceSpan,
) -> Result<(), Report> {
    // if the address was provided as an immediate value, put it onto the stack
    if let Some(addr) = addr {
        if is_local {
            let num_locals = proc_ctx.num_locals();
            local_to_absolute_addr(
                block_builder,
                proc_ctx,
                addr as u16,
                num_locals,
                is_single,
                instr_span,
            )?;
        } else {
            push_u32_value(block_builder, addr);
        }
    } else {
        assert!(!is_local, "local always contains addr value");
    }

    // load from the memory address on top of the stack
    if is_single {
        block_builder.push_op(MLoad);
    } else {
        block_builder.push_op(MLoadW);
    }

    Ok(())
}

/// Appends operations to the span needed to execute a memory write instruction with an immediate
/// address. This includes writing a single element or an entire word into either local or global
/// memory. Specifically, this handles mem_store, mem_storew_be, mem_storew_le, loc_store,
/// loc_storew_be, and loc_storew_le instructions.
///
/// VM cycles per operation:
/// - mem_store.b:
///   - 4 cycles if b = 1
///   - 3 cycles if b != 1
/// - mem_storew_be.b:
///   - 3 cycles if b = 1
///   - 2 cycles if b != 1
/// - loc_store.b:
///   - 5 cycles if b = 1
///   - 4 cycles if b != 1
/// - loc_storew_be.b:
///   - 4 cycles if b = 1
///   - 3 cycles if b != 1
///
/// # Errors
/// Returns an error if we are writing to local memory and local memory index is greater than
/// the number of procedure locals.
pub fn mem_write_imm(
    block_builder: &mut BasicBlockBuilder,
    proc_ctx: &ProcedureContext,
    addr: u32,
    is_local: bool,
    is_single: bool,
    instr_span: SourceSpan,
) -> Result<(), Report> {
    if is_local {
        local_to_absolute_addr(
            block_builder,
            proc_ctx,
            addr as u16,
            proc_ctx.num_locals(),
            is_single,
            instr_span,
        )?;
    } else {
        push_u32_value(block_builder, addr);
    }

    if is_single {
        block_builder.push_op(MStore);
        block_builder.push_op(Drop);
    } else {
        block_builder.push_op(MStoreW);
    }

    Ok(())
}

// HELPER FUNCTIONS
// ================================================================================================

/// Appends a sequence of operations to the basic block needed for converting procedure local index
/// to absolute memory address. This consists in calculating the offset of the local value from the
/// frame pointer and pushing the result onto the stack.
///
/// This operation takes:
/// - 3 VM cycles if index == 1
/// - 2 VM cycles if index != 1
///
/// # Errors
/// Returns an error if index is greater than the number of procedure locals.
pub fn local_to_absolute_addr(
    block_builder: &mut BasicBlockBuilder,
    proc_ctx: &ProcedureContext,
    index_of_local: u16,
    num_proc_locals: u16,
    is_single: bool,
    instr_span: SourceSpan,
) -> Result<(), Report> {
    if num_proc_locals == 0 {
        return Err(RelatedLabel::error("invalid procedure local reference")
            .with_labeled_span(
                proc_ctx.span(),
                "this procedure definition does not allocate any locals",
            )
            .with_labeled_span(instr_span, "the procedure local index referenced here is invalid")
            .with_source_file(proc_ctx.source_manager().get(instr_span.source_id()).ok())
            .into());
    }

    // If a single local value is being accessed, then the index can take the full range
    // [0, num_proc_locals - 1]. Otherwise, the index can take the range [0, num_proc_locals - 4]
    // to account for the fact that a full word is being accessed.
    let max = if is_single {
        num_proc_locals - 1
    } else {
        // If a word local value is used, then the procedure needs at least 4 local values.
        num_proc_locals.checked_sub(4).ok_or_else(|| {
            RelatedLabel::error("invalid procedure local reference")
                .with_labeled_span(
                    proc_ctx.span(),
                    format!("this procedure only allocates {num_proc_locals} locals"),
                )
                .with_labeled_span(instr_span, "but this instruction expects at least 4")
                .with_source_file(proc_ctx.source_manager().get(instr_span.source_id()).ok())
        })?
    };

    // Local values are placed under the frame pointer, so we need to calculate the offset of the
    // local value from the frame pointer.
    // The offset is in the range [1, num_proc_locals], which is then subtracted from `fmp`.
    if index_of_local > max {
        return Err(RelatedLabel::error("invalid procedure local index")
            .with_help(
                if is_single {
                    "the index is greater than the number of allocated locals"
                } else {
                    "this instruction expects a word-sized value, so at least 4 locals must be addressable at the given index"
                }
            )
            .with_labeled_span(proc_ctx.span(),  format!("this procedure only allocates {num_proc_locals} locals"))
            .with_labeled_span(instr_span, "but this local index would reach out of bounds")
            .with_source_file(proc_ctx.source_manager().get(instr_span.source_id()).ok())
            .into()
        );
    }

    // The frame pointer is always incremented by a word-aligned number of locals to ensure
    // word-aligned memory access. We need to use the same alignment when calculating the offset.
    // See fmp_start_frame_sequence() and fmp_end_frame_sequence() in fmp.rs.
    let aligned_num_locals = num_proc_locals.next_multiple_of(WORD_SIZE as u16);
    let fmp_offset_of_local = -Felt::from_u16(aligned_num_locals - index_of_local);
    block_builder.push_ops(push_offset_fmp_sequence(fmp_offset_of_local));

    Ok(())
}