Skip to content

Commit

Permalink
refactor: improve analyze tool
Browse files Browse the repository at this point in the history
  • Loading branch information
Fumuran committed Oct 10, 2023
1 parent 6d8cb96 commit 4afbbe8
Show file tree
Hide file tree
Showing 4 changed files with 138 additions and 27 deletions.
87 changes: 64 additions & 23 deletions miden/src/tools/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ use super::{cli::InputFile, ProgramError};
use clap::Parser;
use core::fmt;
use miden::{utils::collections::Vec, Assembler, DefaultHost, Host, Operation, StackInputs};
use processor::AsmOpInfo;
use processor::{AsmOpInfo, TraceLenSummary};
use std::{fs, path::PathBuf};
use stdlib::StdLibrary;

Expand Down Expand Up @@ -49,20 +49,15 @@ impl Analyze {
/// Contains details of executing a program, used for program analysis.
#[derive(Debug, Default, Eq, PartialEq)]
pub struct ExecutionDetails {
/// Number of VM cycles it took to execute the entire program.
total_vm_cycles: u32,
/// Number of noops executed as part of a program.
total_noops: usize,
/// Statistics about individual assembly operations executed by the VM, see [AsmOpStats].
asm_op_stats: Vec<AsmOpStats>,
/// Information about VM components trace lengths.
trace_len_summary: TraceLenSummary,
}

impl ExecutionDetails {
/// Returns total vm cycles to execute a program
pub fn total_vm_cycles(&self) -> u32 {
self.total_vm_cycles
}

/// Returns total noops executed as part of a program
pub fn total_noops(&self) -> usize {
self.total_noops
Expand All @@ -74,6 +69,11 @@ impl ExecutionDetails {
&self.asm_op_stats
}

/// Returns [TraceLenSummary] that contains the data about lengths of the trace parts.
pub fn trace_len_summary(&self) -> TraceLenSummary {
self.trace_len_summary
}

// STATE MUTATORS
// --------------------------------------------------------------------------------------------

Expand All @@ -82,11 +82,6 @@ impl ExecutionDetails {
self.total_noops += 1;
}

/// Sets the total vm cycles to the provided value
pub fn set_total_vm_cycles(&mut self, total_vm_cycles: u32) {
self.total_vm_cycles = total_vm_cycles;
}

/// Records a new occurrence of asmop in the sorted asmop stats vector of this program info.
/// If the asmop is already in the list, increments its frequency by one.
/// If the asmop is not already in the list, add it at the appropriate index to keep the
Expand Down Expand Up @@ -115,28 +110,72 @@ impl ExecutionDetails {
}
}
}

/// Sets the information about lengths of the trace parts.
pub fn set_trace_len_summary(&mut self, extended_cycles_info: &TraceLenSummary) {
self.trace_len_summary = *extended_cycles_info;
}
}

impl fmt::Display for ExecutionDetails {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
let total_vm_cycles = self.total_vm_cycles();
// calculate the percentage of padded rows
let padding_percentage = (self.trace_len_summary().padded_trace_len()
- self.trace_len_summary().trace_len())
* 100
/ self.trace_len_summary().padded_trace_len();

writeln!(
f,
"\nVM cycles: {} extended to {} steps ({}% padding).
├── Stack rows: {}
├── Range checker rows: {}
└── Chiplets rows: {}
├── Hash chiplet rows: {}
├── Bitwise chiplet rows: {}
├── Memory chiplet rows: {}
└── Kernel ROM rows: {}",
self.trace_len_summary().trace_len(),
self.trace_len_summary().padded_trace_len(),
padding_percentage,
self.trace_len_summary().main_trace_len(),
self.trace_len_summary().range_trace_len(),
self.trace_len_summary().chiplets_trace_len().trace_len(),
self.trace_len_summary().chiplets_trace_len().hash_chiplet_len(),
self.trace_len_summary().chiplets_trace_len().bitwise_chiplet_len(),
self.trace_len_summary().chiplets_trace_len().memory_chiplet_len(),
self.trace_len_summary().chiplets_trace_len().kernel_rom_len(),
)?;
let total_noops = self.total_noops();
let asm_op_stats = self.asm_op_stats();
writeln!(f, "Total Number of VM Cycles: {}\n", total_vm_cycles)?;
writeln!(f, "Total Number of NOOPs executed: {}\n", total_noops)?;

// calculate the total length of pading for the `AsmOp` column
let padding =
asm_op_stats.iter().try_fold(20, |max, value| Ok(value.op().len().max(max)))?;

writeln!(
f,
"{0: <20} | {1: <20} | {2: <20} | {3: <20}",
"AsmOp", "Frequency", "Total Cycles", "Avg Instruction Cycles"
"{0: <width$} | {1: <20} | {2: <20} | {3:}",
"AsmOp",
"Frequency",
"Total Cycles",
"Avg Instruction Cycles",
width = padding,
)?;

let delimeter = "-".repeat(padding + 71);
writeln!(f, "{delimeter}")?;

for op_info in asm_op_stats {
writeln!(
f,
"{0: <20} | {1: <20} | {2: <20} | {3: <20.2}",
"{0: <width$} | {1: <20} | {2: <20} | {3:.2}",
op_info.op(),
op_info.frequency(),
op_info.total_vm_cycles(),
op_info.total_vm_cycles() as f64 / op_info.frequency() as f64
op_info.total_vm_cycles() as f64 / op_info.frequency() as f64,
width = padding,
)?;
}
Ok(())
Expand All @@ -158,9 +197,12 @@ where
.map_err(ProgramError::AssemblyError)?
.compile(program)
.map_err(ProgramError::AssemblyError)?;
let vm_state_iterator = processor::execute_iter(&program, stack_inputs, host);
let mut execution_details = ExecutionDetails::default();

let (vm_state_iterator, trace_len_summary) =
processor::execute_iter_with_trace_len(&program, stack_inputs, host);
execution_details.set_trace_len_summary(&trace_len_summary);

for state in vm_state_iterator {
let vm_state = state.map_err(ProgramError::ExecutionError)?;
if matches!(vm_state.op, Some(Operation::Noop)) {
Expand All @@ -169,7 +211,6 @@ where
if let Some(asmop_info) = vm_state.asmop {
execution_details.record_asmop(asmop_info);
}
execution_details.set_total_vm_cycles(vm_state.clk);
}

Ok(execution_details)
Expand Down Expand Up @@ -233,7 +274,7 @@ impl AsmOpStats {
mod tests {
use super::{AsmOpStats, ExecutionDetails, StackInputs};
use miden::MemAdviceProvider;
use processor::DefaultHost;
use processor::{ChipletsLengths, DefaultHost, TraceLenSummary};

#[test]
fn analyze_test() {
Expand All @@ -244,7 +285,6 @@ mod tests {
let execution_details =
super::analyze(source, stack_inputs, host).expect("analyze_test: Unexpected Error");
let expected_details = ExecutionDetails {
total_vm_cycles: 23,
total_noops: 2,
asm_op_stats: vec![
AsmOpStats::new("dropw".to_string(), 1, 4),
Expand All @@ -253,6 +293,7 @@ mod tests {
AsmOpStats::new("movdn.2".to_string(), 1, 1),
AsmOpStats::new("push".to_string(), 2, 3),
],
trace_len_summary: TraceLenSummary::new(23, 39, ChipletsLengths::new_debug(8, 0, 2, 0)),
};
assert_eq!(execution_details, expected_details);
}
Expand Down
34 changes: 33 additions & 1 deletion processor/src/debug.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
use crate::{
Chiplets, Decoder, ExecutionError, Felt, Host, Process, Stack, StarkField, System, Vec,
Chiplets, ChipletsLengths, Decoder, ExecutionError, Felt, Host, Process, Stack, StarkField,
System, TraceLenSummary, Vec,
};
use core::fmt;
use vm_core::{
Expand Down Expand Up @@ -75,6 +76,37 @@ impl VmStateIterator {
}
}

/// Returns a new instance of [VmStateIterator] together with [TraceLenSummary].
pub(super) fn new_with_trace_len<H>(
process: Process<H>,
result: Result<StackOutputs, ExecutionError>,
) -> (Self, TraceLenSummary)
where
H: Host,
{
let (system, decoder, stack, mut range, chiplets, _) = process.into_parts();

let clk = system.clk();
let range_table_len = range.get_number_range_checker_rows();
chiplets.append_range_checks(&mut range);

let trace_len_summary =
TraceLenSummary::new(clk as usize, range_table_len, ChipletsLengths::new(&chiplets));

let state_iterator = Self {
chiplets,
decoder,
stack,
system,
error: result.err(),
clk: 0,
asmop_idx: 0,
forward: true,
};

(state_iterator, trace_len_summary)
}

/// Returns the asm op info corresponding to this vm state and whether this is the start of
/// operation sequence corresponding to current assembly instruction.
fn get_asmop(&self) -> (Option<AsmOpInfo>, bool) {
Expand Down
26 changes: 25 additions & 1 deletion processor/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -51,8 +51,8 @@ mod chiplets;
use chiplets::Chiplets;

mod trace;
pub use trace::ExecutionTrace;
use trace::TraceFragment;
pub use trace::{ChipletsLengths, ExecutionTrace, TraceLenSummary};

mod errors;
pub use errors::{ExecutionError, Ext2InttError};
Expand Down Expand Up @@ -147,6 +147,30 @@ where
VmStateIterator::new(process, result)
}

/// Returns a tuple with VM state iterator and trace lengths summary.
pub fn execute_iter_with_trace_len<H>(
program: &Program,
stack_inputs: StackInputs,
host: H,
) -> (VmStateIterator, TraceLenSummary)
where
H: Host,
{
let mut process = Process::new_debug(program.kernel().clone(), stack_inputs, host);
let stack_outputs = process.execute(program);
if stack_outputs.is_ok() {
assert_eq!(
program.hash(),
process.decoder.program_hash().into(),
"inconsistent program hash"
);
}
let (state_iterator, trace_len_summary) =
VmStateIterator::new_with_trace_len(process, stack_outputs);

(state_iterator, trace_len_summary)
}

// PROCESS
// ================================================================================================

Expand Down
18 changes: 16 additions & 2 deletions processor/src/trace/utils.rs
Original file line number Diff line number Diff line change
Expand Up @@ -247,7 +247,7 @@ impl HintCycle for u64 {
/// - `range_trace_len` contains the length of the range checker trace.
/// - `chiplets_trace_len` contains the trace lengths of the all chiplets (hash, bitwise, memory,
/// kernel ROM)
#[derive(Debug)]
#[derive(Debug, Default, Eq, PartialEq, Clone, Copy)]
pub struct TraceLenSummary {
main_trace_len: usize,
range_trace_len: usize,
Expand Down Expand Up @@ -297,7 +297,7 @@ impl TraceLenSummary {

/// Contains trace lengths of all chilplets: hash, bitwise, memory and kernel ROM trace
/// lengths.
#[derive(Clone, Copy, Debug)]
#[derive(Default, Clone, Copy, Debug, PartialEq, Eq)]
pub struct ChipletsLengths {
hash_chiplet_len: usize,
bitwise_chiplet_len: usize,
Expand All @@ -315,6 +315,20 @@ impl ChipletsLengths {
}
}

pub fn new_debug(
hash_len: usize,
bitwise_len: usize,
memory_len: usize,
kernel_len: usize,
) -> Self {
ChipletsLengths {
hash_chiplet_len: hash_len,
bitwise_chiplet_len: bitwise_len,
memory_chiplet_len: memory_len,
kernel_rom_len: kernel_len,
}
}

/// Returns the length of the hash chiplet trace
pub fn hash_chiplet_len(&self) -> usize {
self.hash_chiplet_len
Expand Down

0 comments on commit 4afbbe8

Please sign in to comment.