Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add benchmark for simple transaction #577

Merged
merged 28 commits into from
Apr 25, 2024
Merged
Show file tree
Hide file tree
Changes from 2 commits
Commits
Show all changes
28 commits
Select commit Hold shift + click to select a range
0667acd
feat: add benchmark for default transaction
Fumuran Apr 8, 2024
b63fbf7
refactor: use trace instead of emit, create wrapper for host
Fumuran Apr 10, 2024
d3943e9
refactor: improve cycles print
Fumuran Apr 10, 2024
c823d82
refactor: add inline comments, change trace ids
Fumuran Apr 10, 2024
7e79a5b
chore: ignore the test untill it become a binary
Fumuran Apr 10, 2024
74e99fd
refactor: organise cycles storing, add traces for each note
Fumuran Apr 11, 2024
b7c22b1
feat: create a simple cli for benchmark binary package
Fumuran Apr 11, 2024
4e33ee8
feat: create cargo-make script for benchmarking, move benchmarks back…
Fumuran Apr 12, 2024
cdf40f9
Merge branch 'next' into andrew-tx-execution-bench
Fumuran Apr 12, 2024
b919a14
refactor: move TransactionProgress to the TransactionHost
Fumuran Apr 12, 2024
dd97cb5
feat: implement bench for P2ID note, add NoteId to the output
Fumuran Apr 13, 2024
ff5d2fd
refactor: move last note id obtaining to the separate function
Fumuran Apr 15, 2024
e3bb648
refactor: fix no-std build
Fumuran Apr 15, 2024
0323b67
refactor: update comments in main.masm file to be consistent with api…
Fumuran Apr 17, 2024
118f371
feat: write benchmark results to the json file
Fumuran Apr 18, 2024
f7d446b
Merge branch 'next' into andrew-tx-execution-bench
Fumuran Apr 18, 2024
58e0324
refactor: move benchmarks to their own crate
Fumuran Apr 18, 2024
ea8e5ba
refactor: add ability to run all benches
Fumuran Apr 18, 2024
d2bfbf3
refactor: update bin name, dependencies; remove benches except all
Fumuran Apr 22, 2024
d6a0d9b
Merge branch 'next' into andrew-tx-execution-bench
Fumuran Apr 22, 2024
b9d6473
chore: fix no-std build
Fumuran Apr 22, 2024
1fe5ad2
refactor: rework TransactionProgress serialization
Fumuran Apr 24, 2024
4e284aa
refactor: exclude bench-tx from no-std build, update dependencies
Fumuran Apr 24, 2024
a06ca80
refactor: improve imports formatting
Fumuran Apr 24, 2024
fad7743
refactor: rework writing to json
Fumuran Apr 25, 2024
fc99ee2
chore: change visibility of miden-tx MockDataStore
Fumuran Apr 25, 2024
52d614b
Merge branch 'next' into andrew-tx-execution-bench
Fumuran Apr 25, 2024
4016957
chore: update CHANGELOG, create README for bench-tx
Fumuran Apr 25, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
12 changes: 12 additions & 0 deletions miden-lib/asm/kernels/transaction/main.masm
Original file line number Diff line number Diff line change
Expand Up @@ -68,6 +68,9 @@ proc.main.1
exec.prologue::prepare_transaction
# => []

push.0 drop
bobbinth marked this conversation as resolved.
Show resolved Hide resolved
trace.0

# Note Processing
# ---------------------------------------------------------------------------------------------

Expand Down Expand Up @@ -109,6 +112,9 @@ proc.main.1
exec.note::note_processing_teardown
# => []

push.0 drop
trace.1

# Transaction Script Processing
# ---------------------------------------------------------------------------------------------

Expand All @@ -133,12 +139,18 @@ proc.main.1
# => []
end

push.0 drop
trace.2

# Epilogue
# ---------------------------------------------------------------------------------------------

# execute the transaction epilogue
exec.epilogue::finalize_transaction
# => [TX_SCRIPT_ROOT, CREATED_NOTES_COMMITMENT, FINAL_ACCOUNT_HASH]

push.0 drop
trace.3
end

begin
Expand Down
9 changes: 8 additions & 1 deletion miden-tx/src/executor/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,7 @@ pub struct TransactionExecutor<D: DataStore> {
impl<D: DataStore> TransactionExecutor<D> {
// CONSTRUCTOR
// --------------------------------------------------------------------------------------------

/// Creates a new [TransactionExecutor] instance with the specified [DataStore].
pub fn new(data_store: D) -> Self {
Self {
Expand All @@ -52,6 +53,12 @@ impl<D: DataStore> TransactionExecutor<D> {
}
}

/// Enables tracing for the created instance of [TransactionExecutor].
pub fn with_tracing(mut self) -> Self {
self.exec_options = self.exec_options.with_tracing();
self
}

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

Expand Down Expand Up @@ -180,7 +187,7 @@ impl<D: DataStore> TransactionExecutor<D> {
/// Returns an error if:
/// - If required data can not be fetched from the [DataStore].
/// - If the transaction can not be compiled.
fn prepare_transaction(
pub fn prepare_transaction(
&self,
account_id: AccountId,
block_ref: u32,
Expand Down
2 changes: 1 addition & 1 deletion miden-tx/src/tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -355,7 +355,7 @@ fn test_tx_script() {
// ================================================================================================

#[derive(Clone)]
struct MockDataStore {
pub struct MockDataStore {
pub account: Account,
pub block_header: BlockHeader,
pub block_chain: ChainMmr,
Expand Down
1 change: 1 addition & 0 deletions miden-tx/tests/integration/bench/mod.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
mod tx_benchmark;
133 changes: 133 additions & 0 deletions miden-tx/tests/integration/bench/tx_benchmark.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,133 @@
use crate::MockDataStore;
use miden_lib::transaction::ToTransactionKernelInputs;
use miden_objects::{accounts::AccountStub, transaction::TransactionArgs};
use miden_tx::{TransactionExecutor, TransactionHost};
use vm_processor::{
AdviceExtractor, AdviceInjector, AdviceProvider, ExecutionError, ExecutionOptions, Host,
HostResponse, ProcessState, RecAdviceProvider,
};

// CONSTANTS
// ================================================================================================

const END_OF_PROLOGUE: u8 = 0;
const END_OF_NODE_PROCESSING: u8 = 1;
const END_OF_TX_SCRIPT_PROCESSING: u8 = 2;
const END_OF_EPILOGUE: u8 = 3;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would use the same scheme here as we do for events and error codes - i.e., the first 16 bits are set to 2 and the remaining 16 bits is a counter that increments (0, 1, 2 etc.).


// BENCHMARK HOST
// ================================================================================================

/// Wrapper around [TransactionHost] used for benchmarking
pub struct BenchHost<A: AdviceProvider> {
host: TransactionHost<A>,
prologue: u32,
note_processing: u32,
tx_script_processing: u32,
epilogue: u32,
}

impl<A: AdviceProvider> BenchHost<A> {
pub fn new(account: AccountStub, adv_provider: A) -> Self {
Self {
host: TransactionHost::new(account, adv_provider),
prologue: 0,
note_processing: 0,
tx_script_processing: 0,
epilogue: 0,
}
}

/// Calculate the final amount of cycles spent for each stage
fn print_stages_cycle(&self) {
std::println!(
"Number of cycles it takes to execule:
- Prologue: {},
- Note processing: {},
- Transaction script processing: {},
- Epilogue: {}
",
self.prologue - 2,
self.note_processing - self.prologue - 2,
self.tx_script_processing - self.note_processing - 2,
self.epilogue - self.tx_script_processing - 2
)
}
}

impl<A: AdviceProvider> Host for BenchHost<A> {
fn get_advice<S: ProcessState>(
&mut self,
process: &S,
extractor: AdviceExtractor,
) -> Result<HostResponse, ExecutionError> {
self.host.get_advice(process, extractor)
}

fn set_advice<S: ProcessState>(
&mut self,
process: &S,
injector: AdviceInjector,
) -> Result<HostResponse, ExecutionError> {
self.host.set_advice(process, injector)
}

fn on_trace<S: ProcessState>(
&mut self,
process: &S,
trace_id: u32,
) -> Result<HostResponse, ExecutionError> {
#[cfg(feature = "std")]
match trace_id as u8 {
END_OF_PROLOGUE => self.prologue = process.clk(),
END_OF_NODE_PROCESSING => self.note_processing = process.clk(),
END_OF_TX_SCRIPT_PROCESSING => self.tx_script_processing = process.clk(),
END_OF_EPILOGUE => {
self.epilogue = process.clk();
self.print_stages_cycle()
},
_ => println!("Invalid trace id was used: {}", trace_id),
}

Ok(HostResponse::None)
}

fn on_event<S: ProcessState>(
&mut self,
process: &S,
event_id: u32,
) -> Result<HostResponse, ExecutionError> {
self.host.on_event(process, event_id)
}
}

// BENCHMARKS
// ================================================================================================

#[test]
fn benchmark_default_tx() {
let data_store = MockDataStore::default();
let mut executor = TransactionExecutor::new(data_store.clone()).with_tracing();

let account_id = data_store.account.id();
executor.load_account(account_id).unwrap();

let block_ref = data_store.block_header.block_num();
let note_ids = data_store.notes.iter().map(|note| note.id()).collect::<Vec<_>>();

let transaction = executor
.prepare_transaction(account_id, block_ref, &note_ids, TransactionArgs::default())
.unwrap();

let (stack_inputs, advice_inputs) = transaction.get_kernel_inputs();
let advice_recorder: RecAdviceProvider = advice_inputs.into();
let mut host = BenchHost::new(transaction.account().into(), advice_recorder);

vm_processor::execute(
transaction.program(),
stack_inputs,
&mut host,
ExecutionOptions::default().with_tracing(),
)
.unwrap();
}
1 change: 1 addition & 0 deletions miden-tx/tests/integration/main.rs
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
mod bench;
mod scripts;
mod wallet;

Expand Down
Loading