From a8ae810e6d0565f802e9311c08fdd00fb2437669 Mon Sep 17 00:00:00 2001 From: Jonas Schneider-Bensch Date: Wed, 16 Oct 2024 11:51:51 +0200 Subject: [PATCH 01/12] Keep benchmarks and tests in their own crate to avoid duplication --- libcrux-testbench/Cargo.toml | 9 ++++ libcrux-testbench/src/lib.rs | 4 ++ libcrux-testbench/src/mldsa.rs | 81 ++++++++++++++++++++++++++++++++++ libcrux-testbench/src/mlkem.rs | 67 ++++++++++++++++++++++++++++ 4 files changed, 161 insertions(+) create mode 100644 libcrux-testbench/Cargo.toml create mode 100644 libcrux-testbench/src/lib.rs create mode 100644 libcrux-testbench/src/mldsa.rs create mode 100644 libcrux-testbench/src/mlkem.rs diff --git a/libcrux-testbench/Cargo.toml b/libcrux-testbench/Cargo.toml new file mode 100644 index 0000000..535cd8e --- /dev/null +++ b/libcrux-testbench/Cargo.toml @@ -0,0 +1,9 @@ +[package] +name = "libcrux-testbench" +version = "0.1.0" +edition = "2021" + +[dependencies] +libcrux-ml-dsa = { path = "../libcrux/libcrux-ml-dsa" } +libcrux-ml-kem = { path = "../libcrux/libcrux-ml-kem", default-features = false, features = [ "pre-verification", "mlkem512", "mlkem768", "mlkem1024",] } +libcrux-iot-testutil = { path = "../libcrux-iot-testutil" } diff --git a/libcrux-testbench/src/lib.rs b/libcrux-testbench/src/lib.rs new file mode 100644 index 0000000..6faa4be --- /dev/null +++ b/libcrux-testbench/src/lib.rs @@ -0,0 +1,4 @@ +#![no_std] + +pub mod mldsa; +pub mod mlkem; diff --git a/libcrux-testbench/src/mldsa.rs b/libcrux-testbench/src/mldsa.rs new file mode 100644 index 0000000..2d05b6d --- /dev/null +++ b/libcrux-testbench/src/mldsa.rs @@ -0,0 +1,81 @@ +use libcrux_iot_testutil::DefmtInfoLogger; +use libcrux_iot_testutil::*; +extern crate alloc; +use alloc::string::String; +use alloc::vec; + +use libcrux_ml_dsa::ml_dsa_65; + +struct MLDSABenchState { + randomness_gen: [u8; 32], + keypair: ml_dsa_65::MLDSA65KeyPair, + signing_randomness: [u8; 32], + message: [u8; 1024], + signature: ml_dsa_65::MLDSA65Signature, +} + +fn bench_keygen(_l: &mut L, state: &MLDSABenchState) -> Result<(), String> { + let _pair = ml_dsa_65::generate_key_pair(state.randomness_gen); + Ok(()) +} + +fn bench_sign(_l: &mut L, state: &MLDSABenchState) -> Result<(), String> { + let _signature = ml_dsa_65::sign( + &state.keypair.signing_key, + &state.message, + b"", + state.signing_randomness, + ); + Ok(()) +} + +fn bench_verify(_l: &mut L, state: &MLDSABenchState) -> Result<(), String> { + let _ = ml_dsa_65::verify( + &state.keypair.verification_key, + &state.message, + b"", + &state.signature, + ); + + Ok(()) +} + +pub fn run_benchmarks() { + // set up the test suite + let test_cases = [ + TestCase::new("bench_keygen", bench_keygen), + TestCase::new("bench_sign", bench_sign), + TestCase::new("bench_verify", bench_verify), + ]; + + let test_suite = TestSuite::new(&test_cases); + + // set up the test config + let test_config = TestConfig { + core_freq: 4_000_000, + only_names: vec!["bench_keygen", "bench_sign", "bench_verify"], + early_abort: false, + benchmark_runs: 1, + }; + + // prepare the state for the benchmarked functions + let randomness_gen = [1u8; 32]; + let keypair = ml_dsa_65::generate_key_pair(randomness_gen); + let signing_randomness = [4u8; 32]; + let message = [5u8; 1024]; + let signature = + ml_dsa_65::sign(&keypair.signing_key, &message, b"", signing_randomness).unwrap(); + + let state = MLDSABenchState { + randomness_gen, + keypair, + signing_randomness, + message, + signature, + }; + + // run the benchmark + let mut logger = DefmtInfoLogger; + let _ = test_suite.benchmark(&mut logger, &test_config, &state); +} + diff --git a/libcrux-testbench/src/mlkem.rs b/libcrux-testbench/src/mlkem.rs new file mode 100644 index 0000000..9c44fbd --- /dev/null +++ b/libcrux-testbench/src/mlkem.rs @@ -0,0 +1,67 @@ +use libcrux_iot_testutil::DefmtInfoLogger; +use libcrux_iot_testutil::*; +extern crate alloc; +use alloc::string::String; +use alloc::vec; + +use libcrux_ml_kem::mlkem768; + +struct MlKemBenchState<'a> { + randomness_gen: [u8; libcrux_ml_kem::KEY_GENERATION_SEED_SIZE], + public_key: &'a mlkem768::MlKem768PublicKey, + private_key: &'a mlkem768::MlKem768PrivateKey, + randomness_encaps: [u8; libcrux_ml_kem::ENCAPS_SEED_SIZE], + ciphertext: mlkem768::MlKem768Ciphertext, +} + +fn bench_keygen(_l: &mut L, state: &MlKemBenchState) -> Result<(), String> { + let _pair = mlkem768::generate_key_pair(state.randomness_gen); + Ok(()) +} + +fn bench_encaps(_l: &mut L, state: &MlKemBenchState) -> Result<(), String> { + let _ = mlkem768::encapsulate(&state.public_key, state.randomness_encaps); + Ok(()) +} + +fn bench_decaps(_l: &mut L, state: &MlKemBenchState) -> Result<(), String> { + let _ = mlkem768::decapsulate(&state.private_key, &state.ciphertext); + + Ok(()) +} + +pub fn run_benchmarks() { + // set up the test suite + let test_cases = [ + TestCase::new("bench_keygen", bench_keygen), + TestCase::new("bench_encaps", bench_encaps), + TestCase::new("bench_decaps", bench_decaps), + ]; + + let test_suite = TestSuite::new(&test_cases); + + // set up the test config + let test_config = TestConfig { + core_freq: 4_000_000, + only_names: vec!["bench_keygen", "bench_encaps", "bench_decaps"], + early_abort: false, + benchmark_runs: 1, + }; + + // prepare the state for the benchmarked functions + let randomness_gen = [1u8; libcrux_ml_kem::KEY_GENERATION_SEED_SIZE]; + let pair = mlkem768::generate_key_pair(randomness_gen); + let randomness_encaps = [2u8; libcrux_ml_kem::ENCAPS_SEED_SIZE]; + let (ciphertext, _shared_secret) = mlkem768::encapsulate(pair.public_key(), randomness_encaps); + let state = MlKemBenchState { + randomness_gen, + public_key: pair.public_key(), + private_key: pair.private_key(), + randomness_encaps, + ciphertext, + }; + + // run the benchmark + let mut logger = DefmtInfoLogger; + let _ = test_suite.benchmark(&mut logger, &test_config, &state); +} From 85e2b5c88a0ed9a4e7fe4c20f65612ddc5251a29 Mon Sep 17 00:00:00 2001 From: Jonas Schneider-Bensch Date: Wed, 16 Oct 2024 11:52:14 +0200 Subject: [PATCH 02/12] Reinstate missing import --- libcrux/libcrux-platform/src/lib.rs | 2 ++ 1 file changed, 2 insertions(+) diff --git a/libcrux/libcrux-platform/src/lib.rs b/libcrux/libcrux-platform/src/lib.rs index c83769b..380389c 100644 --- a/libcrux/libcrux-platform/src/lib.rs +++ b/libcrux/libcrux-platform/src/lib.rs @@ -54,6 +54,8 @@ mod platform { #[cfg(not(hax))] mod platform { + use super::*; + // TODO: Check for z14 or z15 pub fn simd128_support() -> bool { #[cfg(all(target_arch = "aarch64", target_os = "macos"))] From cb9342cb8b050501f9875b75921804283d87055b Mon Sep 17 00:00:00 2001 From: Jonas Schneider-Bensch Date: Wed, 16 Oct 2024 11:52:36 +0200 Subject: [PATCH 03/12] Use testbench crate --- libcrux-nucleo-l4r5zi/Cargo.toml | 6 +- libcrux-nucleo-l4r5zi/src/bin/mldsa_65.rs | 78 +---------------- libcrux-nucleo-l4r5zi/src/bin/mlkem.rs | 100 ++++------------------ 3 files changed, 20 insertions(+), 164 deletions(-) diff --git a/libcrux-nucleo-l4r5zi/Cargo.toml b/libcrux-nucleo-l4r5zi/Cargo.toml index 5ebc337..06c6c10 100644 --- a/libcrux-nucleo-l4r5zi/Cargo.toml +++ b/libcrux-nucleo-l4r5zi/Cargo.toml @@ -19,11 +19,9 @@ defmt = "0.3" defmt-rtt = "0.4" panic-probe = { version = "0.3", features = ["print-defmt"] } cortex-m-semihosting = "0.5.0" -libcrux-ml-kem = { path = "../libcrux/libcrux-ml-kem", default-features = false, features = [ "pre-verification", "mlkem512", "mlkem768", "mlkem1024", ] } -libcrux-ml-dsa = { path = "../libcrux/libcrux-ml-dsa" } libcrux-iot-testutil = { path = "../libcrux-iot-testutil" } -embassy-time = "0.3.2" -embassy-stm32 = { version = "0.1.0", features = [ "stm32l4r5zi", "defmt", "time", "time-driver-any", ] } +libcrux-testbench = { path = "../libcrux-testbench" } +embassy-stm32 = { version = "0.1.0", features = [ "stm32l4r5zi", "defmt", ] } embedded-alloc = "0.6.0" [dev-dependencies] diff --git a/libcrux-nucleo-l4r5zi/src/bin/mldsa_65.rs b/libcrux-nucleo-l4r5zi/src/bin/mldsa_65.rs index 4ac456a..d8ac40e 100644 --- a/libcrux-nucleo-l4r5zi/src/bin/mldsa_65.rs +++ b/libcrux-nucleo-l4r5zi/src/bin/mldsa_65.rs @@ -1,14 +1,7 @@ #![no_main] #![no_std] -use libcrux_iot_testutil::DefmtInfoLogger; -use libcrux_iot_testutil::*; -extern crate alloc; -use alloc::string::String; -use alloc::vec; - use cortex_m::peripheral::Peripherals; -use libcrux_ml_dsa::ml_dsa_65; use libcrux_nucleo_l4r5zi as _; // global logger + panicking-behavior + memory layout use core::ptr::addr_of_mut; @@ -17,40 +10,6 @@ use embedded_alloc::LlffHeap as Heap; #[global_allocator] static HEAP: Heap = Heap::empty(); -struct MLDSABenchState { - randomness_gen: [u8; 32], - keypair: ml_dsa_65::MLDSA65KeyPair, - signing_randomness: [u8; 32], - message: [u8; 1024], - signature: ml_dsa_65::MLDSA65Signature, -} - -fn bench_keygen(_l: &mut L, state: &MLDSABenchState) -> Result<(), String> { - let _pair = ml_dsa_65::generate_key_pair(state.randomness_gen); - Ok(()) -} - -fn bench_sign(_l: &mut L, state: &MLDSABenchState) -> Result<(), String> { - let _signature = ml_dsa_65::sign( - &state.keypair.signing_key, - &state.message, - b"", - state.signing_randomness, - ); - Ok(()) -} - -fn bench_verify(_l: &mut L, state: &MLDSABenchState) -> Result<(), String> { - let _ = ml_dsa_65::verify( - &state.keypair.verification_key, - &state.message, - b"", - &state.signature, - ); - - Ok(()) -} - #[cortex_m_rt::entry] fn main() -> ! { // Initialize the allocator BEFORE you use it @@ -65,42 +24,7 @@ fn main() -> ! { peripherals.DWT.enable_cycle_counter(); } - // set up the test suite - let test_cases = [ - TestCase::new("bench_keygen", bench_keygen), - TestCase::new("bench_sign", bench_sign), - TestCase::new("bench_verify", bench_verify), - ]; - - let test_suite = TestSuite::new(&test_cases); - - // set up the test config - let test_config = TestConfig { - core_freq: 4_000_000, - only_names: vec!["bench_keygen", "bench_sign", "bench_verify"], - early_abort: false, - benchmark_runs: 5, - }; - - // prepare the state for the benchmarked functions - let randomness_gen = [1u8; 32]; - let keypair = ml_dsa_65::generate_key_pair(randomness_gen); - let signing_randomness = [4u8; 32]; - let message = [5u8; 1024]; - let signature = - ml_dsa_65::sign(&keypair.signing_key, &message, b"", signing_randomness).unwrap(); - - let state = MLDSABenchState { - randomness_gen, - keypair, - signing_randomness, - message, - signature, - }; - - // run the benchmark - let mut logger = DefmtInfoLogger; - let _ = test_suite.benchmark(&mut logger, &test_config, &state); + libcrux_testbench::mldsa::run_benchmarks(); libcrux_nucleo_l4r5zi::exit() } diff --git a/libcrux-nucleo-l4r5zi/src/bin/mlkem.rs b/libcrux-nucleo-l4r5zi/src/bin/mlkem.rs index ed8c5ee..a560623 100644 --- a/libcrux-nucleo-l4r5zi/src/bin/mlkem.rs +++ b/libcrux-nucleo-l4r5zi/src/bin/mlkem.rs @@ -1,96 +1,30 @@ #![no_main] #![no_std] -use embassy_stm32::Config; -use embassy_time::{self, Instant}; -use libcrux_ml_kem::mlkem1024 as mlkem; +use cortex_m::peripheral::Peripherals; use libcrux_nucleo_l4r5zi as _; // global logger + panicking-behavior + memory layout -const KEYGEN_ITERATIONS: usize = 100; -const ENCAPS_ITERATIONS: usize = 100; -const DECAPS_ITERATIONS: usize = 100; +use core::ptr::addr_of_mut; +use embedded_alloc::LlffHeap as Heap; -fn time_operation( - description: &str, - setup: SetupF, - operation: OpF, - iterations: usize, -) where - SetupF: FnOnce() -> Input, - OpF: Fn(&Input) -> (), -{ - defmt::println!("{=str} ({=usize} times)", description, iterations); - let input = setup(); - let start_measuring = Instant::now(); - for _ in 0..iterations { - let _ = operation(&input); - } - let end_measuring = Instant::now(); - let time_avg = (end_measuring.as_micros() - start_measuring.as_micros()) / (iterations as u64); - defmt::println!("Took {=u64} µs on average", time_avg); -} +#[global_allocator] +static HEAP: Heap = Heap::empty(); #[cortex_m_rt::entry] fn main() -> ! { - // Need to initialize this, otherwise we have no timers - let _peripherals = embassy_stm32::init(Config::default()); - - defmt::println!("Testing that everything works"); - let randomness_gen = [1u8; libcrux_ml_kem::KEY_GENERATION_SEED_SIZE]; - let pair = mlkem::generate_key_pair(randomness_gen); - defmt::println!("\tKey Generation OK"); - - let randomness_encaps = [2u8; libcrux_ml_kem::ENCAPS_SEED_SIZE]; - let (ct, ss_enc) = mlkem::encapsulate(pair.public_key(), randomness_encaps); - defmt::println!("\tEncapsulation OK"); - - let ss_dec = mlkem::decapsulate(pair.private_key(), &ct); - defmt::println!("\tDecapsulation OK"); - - assert_eq!(ss_enc, ss_dec); - defmt::println!("\tSuccess!"); - - defmt::println!("Benchmarking"); - time_operation( - "\tKey Generation", - || { - let randomness_gen = [1u8; libcrux_ml_kem::KEY_GENERATION_SEED_SIZE]; - randomness_gen - }, - |randomness| { - let _ = mlkem::generate_key_pair(*randomness); - }, - KEYGEN_ITERATIONS, - ); - - time_operation( - "\tEncapsulation", - || { - let randomness_gen = [1u8; libcrux_ml_kem::KEY_GENERATION_SEED_SIZE]; - let pair = mlkem::generate_key_pair(randomness_gen); - let randomness_encaps = [2u8; libcrux_ml_kem::ENCAPS_SEED_SIZE]; - (randomness_encaps, pair) - }, - |(randomness, pair)| { - mlkem::encapsulate(pair.public_key(), *randomness); - }, - ENCAPS_ITERATIONS, - ); + // Initialize the allocator BEFORE you use it + { + use core::mem::MaybeUninit; + const HEAP_SIZE: usize = 1024; + static mut HEAP_MEM: [MaybeUninit; HEAP_SIZE] = [MaybeUninit::uninit(); HEAP_SIZE]; + unsafe { HEAP.init(addr_of_mut!(HEAP_MEM) as usize, HEAP_SIZE) } + + let mut peripherals = Peripherals::take().unwrap(); + peripherals.DCB.enable_trace(); + peripherals.DWT.enable_cycle_counter(); + } - time_operation( - "\tDecapsulation", - || { - let randomness_gen = [1u8; libcrux_ml_kem::KEY_GENERATION_SEED_SIZE]; - let pair = mlkem::generate_key_pair(randomness_gen); - let randomness_encaps = [2u8; libcrux_ml_kem::ENCAPS_SEED_SIZE]; - let (ct, _ss) = mlkem::encapsulate(pair.public_key(), randomness_encaps); - (ct, pair) - }, - |(ct, pair)| { - mlkem::decapsulate(pair.private_key(), ct); - }, - DECAPS_ITERATIONS, - ); + libcrux_testbench::mlkem::run_benchmarks(); libcrux_nucleo_l4r5zi::exit() } From 2480c855cffb47a9f839ed67cc96345fdcfb030f Mon Sep 17 00:00:00 2001 From: Jonas Schneider-Bensch Date: Wed, 16 Oct 2024 11:53:01 +0200 Subject: [PATCH 04/12] Use testbench crate for nRF52840, as well --- libcrux-nrf52840/.cargo/config.toml | 5 +- libcrux-nrf52840/Cargo.toml | 7 +- libcrux-nrf52840/memory.x | 23 ++++++ libcrux-nrf52840/src/bin/mldsa_65.rs | 113 ++++----------------------- libcrux-nrf52840/src/bin/mlkem.rs | 74 ++++-------------- libcrux-nrf52840/src/lib.rs | 2 +- 6 files changed, 65 insertions(+), 159 deletions(-) create mode 100644 libcrux-nrf52840/memory.x diff --git a/libcrux-nrf52840/.cargo/config.toml b/libcrux-nrf52840/.cargo/config.toml index 7da95f4..3ed569f 100644 --- a/libcrux-nrf52840/.cargo/config.toml +++ b/libcrux-nrf52840/.cargo/config.toml @@ -1,6 +1,7 @@ [target.'cfg(all(target_arch = "arm", target_os = "none"))'] -runner = "probe-rs run --chip nRF52840_xxAA" -# runner = ["probe-rs", "run", "--chip", "$CHIP", "--log-format", "{L} {s}"] +# runner = "probe-rs run --chip nRF52840_xxAA" +runner = ["probe-rs", "run", "--chip", "nRF52840_xxAA", "--log-format", "{s}"] + rustflags = [ "-C", "linker=flip-link", diff --git a/libcrux-nrf52840/Cargo.toml b/libcrux-nrf52840/Cargo.toml index e42d437..5738e73 100644 --- a/libcrux-nrf52840/Cargo.toml +++ b/libcrux-nrf52840/Cargo.toml @@ -19,9 +19,10 @@ defmt = "0.3" defmt-rtt = "0.4" panic-probe = { version = "0.3", features = ["print-defmt"] } cortex-m-semihosting = "0.5.0" -nrf52840-hal = "0.18.0" -libcrux-ml-kem = { path = "../libcrux/libcrux-ml-kem", default-features = false, features = ["pre-verification", "mlkem512", "mlkem768", "mlkem1024"] } -libcrux-ml-dsa = { path = "../libcrux/libcrux-ml-dsa" } +libcrux-iot-testutil = { path = "../libcrux-iot-testutil" } +libcrux-testbench = { path = "../libcrux-testbench" } +embassy-nrf = { version = "0.1.0", features = [ "nrf52840", "defmt", ] } +embedded-alloc = "0.6.0" [dev-dependencies] defmt-test = "0.3" diff --git a/libcrux-nrf52840/memory.x b/libcrux-nrf52840/memory.x new file mode 100644 index 0000000..70e5006 --- /dev/null +++ b/libcrux-nrf52840/memory.x @@ -0,0 +1,23 @@ +/* Linker script for the nRF52 - WITHOUT SOFT DEVICE */ +MEMORY +{ + /* NOTE K = KiBi = 1024 bytes */ + FLASH : ORIGIN = 0x00000000, LENGTH = 1024K + RAM : ORIGIN = 0x20000000, LENGTH = 256K +} + +/* This is where the call stack will be allocated. */ +/* The stack is of the full descending type. */ +/* You may want to use this variable to locate the call stack and static + variables in different memory regions. Below is shown the default value */ +/* _stack_start = ORIGIN(RAM) + LENGTH(RAM); */ + +/* You can use this symbol to customize the location of the .text section */ +/* If omitted the .text section will be placed right after the .vector_table + section */ +/* This is required only on microcontrollers that store some configuration right + after the vector table */ +/* _stext = ORIGIN(FLASH) + 0x400; */ + +/* Size of the heap (in bytes) */ +/* _heap_size = 1024; */ diff --git a/libcrux-nrf52840/src/bin/mldsa_65.rs b/libcrux-nrf52840/src/bin/mldsa_65.rs index f0299b2..0939be0 100644 --- a/libcrux-nrf52840/src/bin/mldsa_65.rs +++ b/libcrux-nrf52840/src/bin/mldsa_65.rs @@ -1,109 +1,30 @@ #![no_main] #![no_std] -use libcrux_ml_dsa::ml_dsa_65; +use cortex_m::peripheral::Peripherals; use libcrux_nrf52840 as _; // global logger + panicking-behavior + memory layout -use cortex_m::peripheral::{Peripherals, DWT}; +use core::ptr::addr_of_mut; +use embedded_alloc::LlffHeap as Heap; -const KEYGEN_ITERATIONS: usize = 10; -const SIGN_ITERATIONS: usize = 10; -const VERIFY_ITERATIONS: usize = 10; - -fn init_cycle_counter() -> Peripherals { - // Enable tracing - let mut peripherals = Peripherals::take().unwrap(); - peripherals.DCB.enable_trace(); - peripherals.DWT.enable_cycle_counter(); - - peripherals -} - -fn count_cycles( - description: &str, - setup: SetupF, - operation: OpF, - iterations: usize, -) where - SetupF: FnOnce() -> Input, - OpF: Fn(&Input) -> (), -{ - defmt::println!("{=str} ({=usize} times)", description, iterations); - let input = setup(); - let start_measuring = DWT::cycle_count(); - for _ in 0..iterations { - let _ = operation(&input); - } - let end_measuring = DWT::cycle_count(); - let time_avg = (end_measuring - start_measuring) / (iterations as u32); - defmt::println!("Took {=u32} cycles on average", time_avg); -} +#[global_allocator] +static HEAP: Heap = Heap::empty(); #[cortex_m_rt::entry] fn main() -> ! { - let _peripherals = init_cycle_counter(); - defmt::println!("Testing that everything works"); - let randomness_gen = [1u8; 32]; - let keypair = ml_dsa_65::generate_key_pair(randomness_gen); - defmt::println!("\tKey Generation OK"); - - let signing_randomness = [4u8; 32]; - let message = [5u8; 1024]; - - let signature = ml_dsa_65::sign(&keypair.signing_key, &message, b"", signing_randomness).unwrap(); - defmt::println!("\tSigning OK"); - - let result = ml_dsa_65::verify(&keypair.verification_key, &message, b"", &signature); - defmt::println!("\tVerification OK"); - - assert!(result.is_ok()); - defmt::println!("\tSuccess!"); - - defmt::println!("Benchmarking"); - count_cycles( - "\tKey Generation", - || { - let randomness_gen = [1u8; 32]; - randomness_gen - }, - |randomness| { - let _pair = ml_dsa_65::generate_key_pair(*randomness); - }, - KEYGEN_ITERATIONS, - ); - - count_cycles( - "\tSigning", - || { - let randomness_gen = [1u8; 32]; - let signing_randomness = [4u8; 32]; - let message = [5u8; 1024]; - - let keypair = ml_dsa_65::generate_key_pair(randomness_gen); - (message, keypair, signing_randomness) - }, - |(message, keypair, signing_randomness)| { - let _ = ml_dsa_65::sign(&keypair.signing_key, message, b"", *signing_randomness); - }, - SIGN_ITERATIONS, - ); - - count_cycles( - "\tVerification", - || { - let randomness_gen = [1u8; 32]; - let signing_randomness = [4u8; 32]; - let message = [5u8; 1024]; + // Initialize the allocator BEFORE you use it + { + use core::mem::MaybeUninit; + const HEAP_SIZE: usize = 1024; + static mut HEAP_MEM: [MaybeUninit; HEAP_SIZE] = [MaybeUninit::uninit(); HEAP_SIZE]; + unsafe { HEAP.init(addr_of_mut!(HEAP_MEM) as usize, HEAP_SIZE) } + + let mut peripherals = Peripherals::take().unwrap(); + peripherals.DCB.enable_trace(); + peripherals.DWT.enable_cycle_counter(); + } - let keypair = ml_dsa_65::generate_key_pair(randomness_gen); - let signature = ml_dsa_65::sign(&keypair.signing_key, &message, b"", signing_randomness).unwrap(); - (message, keypair, signature) - }, - |(message, keypair, signature)| { - ml_dsa_65::verify(&keypair.verification_key, message, b"", signature).unwrap(); - }, - VERIFY_ITERATIONS, - ); + libcrux_testbench::mldsa::run_benchmarks(); libcrux_nrf52840::exit() } diff --git a/libcrux-nrf52840/src/bin/mlkem.rs b/libcrux-nrf52840/src/bin/mlkem.rs index feadff3..3d4349b 100644 --- a/libcrux-nrf52840/src/bin/mlkem.rs +++ b/libcrux-nrf52840/src/bin/mlkem.rs @@ -1,70 +1,30 @@ #![no_main] #![no_std] -use libcrux_ml_kem::mlkem1024 as mlkem; +use cortex_m::peripheral::Peripherals; use libcrux_nrf52840 as _; // global logger + panicking-behavior + memory layout -use nrf52840_hal as hal; -// Resolution in µs -const RTC_RESOLUTION: f32 = 30.517; +use core::ptr::addr_of_mut; +use embedded_alloc::LlffHeap as Heap; + +#[global_allocator] +static HEAP: Heap = Heap::empty(); #[cortex_m_rt::entry] fn main() -> ! { - let board = hal::pac::Peripherals::take().unwrap(); - defmt::println!("Starting main!"); - - let randomness_gen = [1u8; libcrux_ml_kem::KEY_GENERATION_SEED_SIZE]; - - let clock = nrf52840_hal::clocks::Clocks::new(board.CLOCK); - clock.start_lfclk(); - - let rtc = nrf52840_hal::rtc::Rtc::new(board.RTC0, 0).unwrap(); - - defmt::println!("Generating 100 key pairs"); - let counter_start = rtc.get_counter(); - rtc.enable_counter(); - for _ in 0..100 { - let _ = mlkem::generate_key_pair(randomness_gen); + // Initialize the allocator BEFORE you use it + { + use core::mem::MaybeUninit; + const HEAP_SIZE: usize = 1024; + static mut HEAP_MEM: [MaybeUninit; HEAP_SIZE] = [MaybeUninit::uninit(); HEAP_SIZE]; + unsafe { HEAP.init(addr_of_mut!(HEAP_MEM) as usize, HEAP_SIZE) } + + let mut peripherals = Peripherals::take().unwrap(); + peripherals.DCB.enable_trace(); + peripherals.DWT.enable_cycle_counter(); } - rtc.disable_counter(); - let counter_end = rtc.get_counter(); - let counter_keygen_total = counter_end - counter_start; - let counter_keygen_avg = counter_keygen_total as f32/ 100.0; - let time_keygen_avg = counter_keygen_avg * RTC_RESOLUTION; - defmt::println!("Took {=f32} µs to generate key pair on average", time_keygen_avg); - - let pair = mlkem::generate_key_pair(randomness_gen); - let randomness_encaps = [2u8; libcrux_ml_kem::ENCAPS_SEED_SIZE]; - defmt::println!("Encapsulating 100 times."); - let counter_start = rtc.get_counter(); - rtc.enable_counter(); - for _ in 0..100 { - let _ = mlkem::encapsulate(pair.public_key(), randomness_encaps); - } - rtc.disable_counter(); - let counter_end = rtc.get_counter(); - let counter_encaps_total = counter_end - counter_start; - let counter_encaps_avg = counter_encaps_total as f32/ 100.0; - let time_encaps_avg = counter_encaps_avg * RTC_RESOLUTION; - defmt::println!("Took {=f32} µs to encapsulate on average", time_encaps_avg); - - let (ct, ss) = mlkem::encapsulate(pair.public_key(), randomness_encaps); - - defmt::println!("Decapsulating 100 times."); - let counter_start = rtc.get_counter(); - rtc.enable_counter(); - for _ in 0..100 { - let _ = mlkem::decapsulate(pair.private_key(), &ct); - } - let counter_end = rtc.get_counter(); - let counter_decaps_total = counter_end - counter_start; - let counter_decaps_avg = counter_decaps_total as f32/ 100.0; - let time_decaps_avg = counter_decaps_avg * RTC_RESOLUTION; - defmt::println!("Took {=f32} µs to decapsulate on average", time_decaps_avg); - - let ss_decaps = mlkem::decapsulate(pair.private_key(), &ct); - assert_eq!(ss, ss_decaps); + libcrux_testbench::mlkem::run_benchmarks(); libcrux_nrf52840::exit() } diff --git a/libcrux-nrf52840/src/lib.rs b/libcrux-nrf52840/src/lib.rs index fd3a0e5..d551cd1 100644 --- a/libcrux-nrf52840/src/lib.rs +++ b/libcrux-nrf52840/src/lib.rs @@ -5,7 +5,7 @@ use cortex_m_semihosting::debug; use defmt_rtt as _; // global logger -use nrf52840_hal as _; // memory layout +use embassy_nrf as _; // memory layout use panic_probe as _; From fc698556b0e3ff65e0a80aa290ca76bdb3317509 Mon Sep 17 00:00:00 2001 From: Jonas Schneider-Bensch Date: Wed, 16 Oct 2024 13:21:47 +0200 Subject: [PATCH 05/12] Add benchmark suite names --- libcrux-testbench/src/mldsa.rs | 2 +- libcrux-testbench/src/mlkem.rs | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/libcrux-testbench/src/mldsa.rs b/libcrux-testbench/src/mldsa.rs index 2d05b6d..646a075 100644 --- a/libcrux-testbench/src/mldsa.rs +++ b/libcrux-testbench/src/mldsa.rs @@ -48,7 +48,7 @@ pub fn run_benchmarks() { TestCase::new("bench_verify", bench_verify), ]; - let test_suite = TestSuite::new(&test_cases); + let test_suite = TestSuite::new("ML-DSA Benchmark", &test_cases); // set up the test config let test_config = TestConfig { diff --git a/libcrux-testbench/src/mlkem.rs b/libcrux-testbench/src/mlkem.rs index 9c44fbd..9ce71e4 100644 --- a/libcrux-testbench/src/mlkem.rs +++ b/libcrux-testbench/src/mlkem.rs @@ -38,7 +38,7 @@ pub fn run_benchmarks() { TestCase::new("bench_decaps", bench_decaps), ]; - let test_suite = TestSuite::new(&test_cases); + let test_suite = TestSuite::new("ML-KEM Benchmark", &test_cases); // set up the test config let test_config = TestConfig { From e0514ed88273f4263498161fbeeca9f055c91cd1 Mon Sep 17 00:00:00 2001 From: Jonas Schneider-Bensch Date: Wed, 16 Oct 2024 13:42:48 +0200 Subject: [PATCH 06/12] Introduce features to test/benchmark different parameter sets --- libcrux-nrf52840/Cargo.toml | 9 ++++ .../src/bin/{mldsa_65.rs => mldsa.rs} | 0 libcrux-nucleo-l4r5zi/Cargo.toml | 9 ++++ .../src/bin/{mldsa_65.rs => mldsa.rs} | 0 libcrux-testbench/Cargo.toml | 8 ++++ libcrux-testbench/src/mldsa.rs | 37 +++++++++++----- libcrux-testbench/src/mlkem.rs | 43 +++++++++++++++---- 7 files changed, 87 insertions(+), 19 deletions(-) rename libcrux-nrf52840/src/bin/{mldsa_65.rs => mldsa.rs} (100%) rename libcrux-nucleo-l4r5zi/src/bin/{mldsa_65.rs => mldsa.rs} (100%) diff --git a/libcrux-nrf52840/Cargo.toml b/libcrux-nrf52840/Cargo.toml index 5738e73..3957715 100644 --- a/libcrux-nrf52840/Cargo.toml +++ b/libcrux-nrf52840/Cargo.toml @@ -27,6 +27,15 @@ embedded-alloc = "0.6.0" [dev-dependencies] defmt-test = "0.3" +[features] +default = ["mldsa87", "mlkem1024"] # We want to benchmark the most expensive variants by default +mldsa44 = ["libcrux-testbench/mldsa44"] +mldsa65 = ["libcrux-testbench/mldsa65"] +mldsa87 = ["libcrux-testbench/mldsa87"] +mlkem512 = ["libcrux-testbench/mlkem512"] +mlkem768 = ["libcrux-testbench/mlkem768"] +mlkem1024 = ["libcrux-testbench/mlkem1024"] + # cargo build/run [profile.dev] codegen-units = 1 diff --git a/libcrux-nrf52840/src/bin/mldsa_65.rs b/libcrux-nrf52840/src/bin/mldsa.rs similarity index 100% rename from libcrux-nrf52840/src/bin/mldsa_65.rs rename to libcrux-nrf52840/src/bin/mldsa.rs diff --git a/libcrux-nucleo-l4r5zi/Cargo.toml b/libcrux-nucleo-l4r5zi/Cargo.toml index 06c6c10..eedd7fe 100644 --- a/libcrux-nucleo-l4r5zi/Cargo.toml +++ b/libcrux-nucleo-l4r5zi/Cargo.toml @@ -27,6 +27,15 @@ embedded-alloc = "0.6.0" [dev-dependencies] defmt-test = "0.3" +[features] +default = ["mldsa87", "mlkem1024"] # We want to benchmark the most expensive variants by default +mldsa44 = ["libcrux-testbench/mldsa44"] +mldsa65 = ["libcrux-testbench/mldsa65"] +mldsa87 = ["libcrux-testbench/mldsa87"] +mlkem512 = ["libcrux-testbench/mlkem512"] +mlkem768 = ["libcrux-testbench/mlkem768"] +mlkem1024 = ["libcrux-testbench/mlkem1024"] + # cargo build/run [profile.dev] codegen-units = 1 diff --git a/libcrux-nucleo-l4r5zi/src/bin/mldsa_65.rs b/libcrux-nucleo-l4r5zi/src/bin/mldsa.rs similarity index 100% rename from libcrux-nucleo-l4r5zi/src/bin/mldsa_65.rs rename to libcrux-nucleo-l4r5zi/src/bin/mldsa.rs diff --git a/libcrux-testbench/Cargo.toml b/libcrux-testbench/Cargo.toml index 535cd8e..f90354f 100644 --- a/libcrux-testbench/Cargo.toml +++ b/libcrux-testbench/Cargo.toml @@ -7,3 +7,11 @@ edition = "2021" libcrux-ml-dsa = { path = "../libcrux/libcrux-ml-dsa" } libcrux-ml-kem = { path = "../libcrux/libcrux-ml-kem", default-features = false, features = [ "pre-verification", "mlkem512", "mlkem768", "mlkem1024",] } libcrux-iot-testutil = { path = "../libcrux-iot-testutil" } + +[features] +mldsa44 = [] +mldsa65 = [] +mldsa87 = [] +mlkem512 = [] +mlkem768 = [] +mlkem1024 = [] diff --git a/libcrux-testbench/src/mldsa.rs b/libcrux-testbench/src/mldsa.rs index 646a075..8ba74da 100644 --- a/libcrux-testbench/src/mldsa.rs +++ b/libcrux-testbench/src/mldsa.rs @@ -4,23 +4,42 @@ extern crate alloc; use alloc::string::String; use alloc::vec; -use libcrux_ml_dsa::ml_dsa_65; +#[cfg(feature = "mldsa44")] +use libcrux_ml_dsa::ml_dsa_44 as mldsa; +#[cfg(feature = "mldsa44")] +type MLDSASignature = mldsa::MLDSA44Signature; +#[cfg(feature = "mldsa44")] +type MLDSAKeyPair = mldsa::MLDSA44KeyPair; + +#[cfg(feature = "mldsa65")] +use libcrux_ml_dsa::ml_dsa_65 as mldsa; +#[cfg(feature = "mldsa65")] +type MLDSASignature = mldsa::MLDSA65Signature; +#[cfg(feature = "mldsa65")] +type MLDSAKeyPair = mldsa::MLDSA65KeyPair; + +#[cfg(feature = "mldsa87")] +use libcrux_ml_dsa::ml_dsa_87 as mldsa; +#[cfg(feature = "mldsa87")] +type MLDSASignature = mldsa::MLDSA87Signature; +#[cfg(feature = "mldsa87")] +type MLDSAKeyPair = mldsa::MLDSA87KeyPair; struct MLDSABenchState { randomness_gen: [u8; 32], - keypair: ml_dsa_65::MLDSA65KeyPair, + keypair: MLDSAKeyPair, signing_randomness: [u8; 32], message: [u8; 1024], - signature: ml_dsa_65::MLDSA65Signature, + signature: MLDSASignature, } fn bench_keygen(_l: &mut L, state: &MLDSABenchState) -> Result<(), String> { - let _pair = ml_dsa_65::generate_key_pair(state.randomness_gen); + let _pair = mldsa::generate_key_pair(state.randomness_gen); Ok(()) } fn bench_sign(_l: &mut L, state: &MLDSABenchState) -> Result<(), String> { - let _signature = ml_dsa_65::sign( + let _signature = mldsa::sign( &state.keypair.signing_key, &state.message, b"", @@ -30,7 +49,7 @@ fn bench_sign(_l: &mut L, state: &MLDSABenchState) -> Result<(), } fn bench_verify(_l: &mut L, state: &MLDSABenchState) -> Result<(), String> { - let _ = ml_dsa_65::verify( + let _ = mldsa::verify( &state.keypair.verification_key, &state.message, b"", @@ -60,11 +79,10 @@ pub fn run_benchmarks() { // prepare the state for the benchmarked functions let randomness_gen = [1u8; 32]; - let keypair = ml_dsa_65::generate_key_pair(randomness_gen); + let keypair = mldsa::generate_key_pair(randomness_gen); let signing_randomness = [4u8; 32]; let message = [5u8; 1024]; - let signature = - ml_dsa_65::sign(&keypair.signing_key, &message, b"", signing_randomness).unwrap(); + let signature = mldsa::sign(&keypair.signing_key, &message, b"", signing_randomness).unwrap(); let state = MLDSABenchState { randomness_gen, @@ -78,4 +96,3 @@ pub fn run_benchmarks() { let mut logger = DefmtInfoLogger; let _ = test_suite.benchmark(&mut logger, &test_config, &state); } - diff --git a/libcrux-testbench/src/mlkem.rs b/libcrux-testbench/src/mlkem.rs index 9ce71e4..d797193 100644 --- a/libcrux-testbench/src/mlkem.rs +++ b/libcrux-testbench/src/mlkem.rs @@ -4,28 +4,53 @@ extern crate alloc; use alloc::string::String; use alloc::vec; -use libcrux_ml_kem::mlkem768; +#[cfg(feature = "mlkem512")] +use libcrux_ml_kem::mlkem512 as mlkem; +#[cfg(feature = "mlkem512")] +type MlKemPublicKey = mlkem::MlKem512PublicKey; +#[cfg(feature = "mlkem512")] +type MlKemPrivateKey = mlkem::MlKem512PrivateKey; +#[cfg(feature = "mlkem512")] +type MlKemCiphertext = mlkem::MlKem512Ciphertext; + +#[cfg(feature = "mlkem768")] +use libcrux_ml_kem::mlkem768 as mlkem; +#[cfg(feature = "mlkem768")] +type MlKemPublicKey = mlkem::MlKem768PublicKey; +#[cfg(feature = "mlkem768")] +type MlKemPrivateKey = mlkem::MlKem768PrivateKey; +#[cfg(feature = "mlkem768")] +type MlKemCiphertext = mlkem::MlKem768Ciphertext; + +#[cfg(feature = "mlkem1024")] +use libcrux_ml_kem::mlkem1024 as mlkem; +#[cfg(feature = "mlkem1024")] +type MlKemPublicKey = mlkem::MlKem1024PublicKey; +#[cfg(feature = "mlkem1024")] +type MlKemPrivateKey = mlkem::MlKem1024PrivateKey; +#[cfg(feature = "mlkem1024")] +type MlKemCiphertext = mlkem::MlKem1024Ciphertext; struct MlKemBenchState<'a> { randomness_gen: [u8; libcrux_ml_kem::KEY_GENERATION_SEED_SIZE], - public_key: &'a mlkem768::MlKem768PublicKey, - private_key: &'a mlkem768::MlKem768PrivateKey, + public_key: &'a MlKemPublicKey, + private_key: &'a MlKemPrivateKey, randomness_encaps: [u8; libcrux_ml_kem::ENCAPS_SEED_SIZE], - ciphertext: mlkem768::MlKem768Ciphertext, + ciphertext: MlKemCiphertext, } fn bench_keygen(_l: &mut L, state: &MlKemBenchState) -> Result<(), String> { - let _pair = mlkem768::generate_key_pair(state.randomness_gen); + let _pair = mlkem::generate_key_pair(state.randomness_gen); Ok(()) } fn bench_encaps(_l: &mut L, state: &MlKemBenchState) -> Result<(), String> { - let _ = mlkem768::encapsulate(&state.public_key, state.randomness_encaps); + let _ = mlkem::encapsulate(&state.public_key, state.randomness_encaps); Ok(()) } fn bench_decaps(_l: &mut L, state: &MlKemBenchState) -> Result<(), String> { - let _ = mlkem768::decapsulate(&state.private_key, &state.ciphertext); + let _ = mlkem::decapsulate(&state.private_key, &state.ciphertext); Ok(()) } @@ -50,9 +75,9 @@ pub fn run_benchmarks() { // prepare the state for the benchmarked functions let randomness_gen = [1u8; libcrux_ml_kem::KEY_GENERATION_SEED_SIZE]; - let pair = mlkem768::generate_key_pair(randomness_gen); + let pair = mlkem::generate_key_pair(randomness_gen); let randomness_encaps = [2u8; libcrux_ml_kem::ENCAPS_SEED_SIZE]; - let (ciphertext, _shared_secret) = mlkem768::encapsulate(pair.public_key(), randomness_encaps); + let (ciphertext, _shared_secret) = mlkem::encapsulate(pair.public_key(), randomness_encaps); let state = MlKemBenchState { randomness_gen, public_key: pair.public_key(), From 630efaeba0cf5bfcd92c456399a76e2fd64e7ad2 Mon Sep 17 00:00:00 2001 From: Jonas Schneider-Bensch Date: Wed, 16 Oct 2024 15:16:21 +0200 Subject: [PATCH 07/12] Initial results for more boards --- benchmarks.md | 115 ++++++++++++++++++++++++++ libcrux-nrf52810/.gitignore | 2 + libcrux-nrf52810/Cargo.toml | 76 +++++++++++++++++ libcrux-nrf52810/README.md | 32 +++++++ libcrux-nrf52810/memory.x | 23 ++++++ libcrux-nrf52810/src/bin/mldsa.rs | 30 +++++++ libcrux-nrf52810/src/bin/mlkem.rs | 30 +++++++ libcrux-nrf52810/src/lib.rs | 51 ++++++++++++ libcrux-nrf52810/tests/integration.rs | 16 ++++ libcrux-nrf52832/.gitignore | 2 + libcrux-nrf52832/Cargo.toml | 76 +++++++++++++++++ libcrux-nrf52832/README.md | 32 +++++++ libcrux-nrf52832/memory.x | 23 ++++++ libcrux-nrf52832/src/bin/mldsa.rs | 30 +++++++ libcrux-nrf52832/src/bin/mlkem.rs | 30 +++++++ libcrux-nrf52832/src/lib.rs | 51 ++++++++++++ libcrux-nrf52832/tests/integration.rs | 16 ++++ libcrux-nrf5340/.gitignore | 2 + libcrux-nrf5340/Cargo.toml | 76 +++++++++++++++++ libcrux-nrf5340/README.md | 32 +++++++ libcrux-nrf5340/memory.x | 34 ++++++++ libcrux-nrf5340/src/bin/mldsa.rs | 30 +++++++ libcrux-nrf5340/src/bin/mlkem.rs | 30 +++++++ libcrux-nrf5340/src/lib.rs | 51 ++++++++++++ libcrux-nrf5340/tests/integration.rs | 16 ++++ 25 files changed, 906 insertions(+) create mode 100644 benchmarks.md create mode 100644 libcrux-nrf52810/.gitignore create mode 100644 libcrux-nrf52810/Cargo.toml create mode 100644 libcrux-nrf52810/README.md create mode 100644 libcrux-nrf52810/memory.x create mode 100644 libcrux-nrf52810/src/bin/mldsa.rs create mode 100644 libcrux-nrf52810/src/bin/mlkem.rs create mode 100644 libcrux-nrf52810/src/lib.rs create mode 100644 libcrux-nrf52810/tests/integration.rs create mode 100644 libcrux-nrf52832/.gitignore create mode 100644 libcrux-nrf52832/Cargo.toml create mode 100644 libcrux-nrf52832/README.md create mode 100644 libcrux-nrf52832/memory.x create mode 100644 libcrux-nrf52832/src/bin/mldsa.rs create mode 100644 libcrux-nrf52832/src/bin/mlkem.rs create mode 100644 libcrux-nrf52832/src/lib.rs create mode 100644 libcrux-nrf52832/tests/integration.rs create mode 100644 libcrux-nrf5340/.gitignore create mode 100644 libcrux-nrf5340/Cargo.toml create mode 100644 libcrux-nrf5340/README.md create mode 100644 libcrux-nrf5340/memory.x create mode 100644 libcrux-nrf5340/src/bin/mldsa.rs create mode 100644 libcrux-nrf5340/src/bin/mlkem.rs create mode 100644 libcrux-nrf5340/src/lib.rs create mode 100644 libcrux-nrf5340/tests/integration.rs diff --git a/benchmarks.md b/benchmarks.md new file mode 100644 index 0000000..544ce4a --- /dev/null +++ b/benchmarks.md @@ -0,0 +1,115 @@ +# Benchmark Results + +All numbers reported here refer to cycle counts. + +## ML-KEM + +### ML-KEM 512 + +| Device | KeyGen[Debug] | KeyGen[Release] | Encaps[Debug] | Encaps[Release] | Decaps[Debug] | Decaps[Release] | +|---------------------------------|---------------|-----------------|---------------|-----------------|---------------|-----------------| +| Raspberry Pi 3 | | | | | | | +| Raspberry Pi 4 | | | | | | | +| [ESP32-S3] [^1] [^2] | | | | | | | +| [STM32-L4R5xx] (our Nucleo-144) | 4445020 | 771501 | 5044052 | 839800 | 6134089 | 839800 | +| [ESP32-C6] [^3] | | | | | | | +| [nRF52840-DK] | 5968523 | 764636 | 6806307 | 839812 | 8316331 | 1011669 | +| [nRF5340-DK] | | | | | | | +| [nRF52-DK] | | | | | | | +| - nRF52832 | 5789064 | 762595 | 6579357 | 838838 | 7998480 | 1010106 | +| - nRF52810 | ❌ [^5] | ❌ [^4] | ❌ [^5] | ❌ [^4] | ❌ [^5] | ❌ [^4] | + + +### ML-KEM 768 + +| Device | KeyGen[Debug] | KeyGen[Release] | Encaps[Debug] | Encaps[Release] | Decaps[Debug] | Decaps[Release] | +|---------------------------------|---------------|-----------------|---------------|-----------------|---------------|-----------------| +| Raspberry Pi 3 | | | | | | | +| Raspberry Pi 4 | | | | | | | +| [ESP32-S3] [^1] [^2] | | | | | | | +| [STM32-L4R5xx] (our Nucleo-144) | 7085443 | 1273471 | 7996675 | 1395000 | 9421249 | 1591459 | +| [ESP32-C6] [^3] | | | | | | | +| [nRF52840-DK] | 9315152 | 1257679 | 10418524 | 1381088 | 12319554 | 1607698 | +| [nRF5340-DK] | | | | | | | +| [nRF52-DK] | | | | | | | +| - nRF52832 | 9434994 | 1276466 | 10691113 | 1401091 | 12626256 | 1630202 | +| - nRF52810 | ❌ [^5] | ❌ [^4] | ❌ [^5] | ❌ [^4] | ❌ [^5] | ❌ [^4] | + + +### ML-KEM 1024 + +| Device | KeyGen[Debug] | KeyGen[Release] | Encaps[Debug] | Encaps[Release] | Decaps[Debug] | Decaps[Release] | +|---------------------------------|---------------|-----------------|---------------|-----------------|---------------|-----------------| +| Raspberry Pi 3 | | | | | | | +| Raspberry Pi 4 | | | | | | | +| [ESP32-S3] [^1] [^2] | | | | | | | +| [STM32-L4R5xx] (our Nucleo-144) | 10981246 | 1991691 | 11988292 | 2129442 | 13756274 | 2371826 | +| [ESP32-C6] [^3] | | | | | | | +| [nRF52840-DK] | 14_910_284 | 1_959_916 | 16_318_665 | 2_103_371 | 18_830_628 | 2_391_463 | +| [nRF5340-DK] | | | | | | | +| [nRF52-DK] | | | | | | | +| - nRF52832 | 14886756 | 1951395 | 16021505 | 2094650 | 18385162 | 2381655 | +| - nRF52810 | ❌ [^5] | ❌ [^4] | ❌ [^5] | ❌ [^4] | ❌ [^5] | ❌ [^4] | + + +## ML-DSA +### ML-DSA 44 + +| Device | KeyGen[Debug] | KeyGen[Release] | Sign[Debug] | Sign[Release] | Verify[Debug] | Verify[Release] | +|---------------------------------|---------------|-----------------|-------------|---------------|---------------|-----------------| +| Raspberry Pi 3 | | | | | | | +| Raspberry Pi 4 | | | | | | | +| [ESP32-S3] [^1] [^2] | | | | | | | +| [STM32-L4R5xx] (our Nucleo-144) | 19524681 | 3476142 | 28743483 | 4825636 | 20929483 | 3695919 | +| [ESP32-C6] [^3] | | | | | | | +| [nRF52840-DK] | 25565997 | 3565042 | 37809350 | 4987219 | 27441288 | 3788007 | +| [nRF5340-DK] | | | | | | | +| [nRF52-DK] | | | | | | | +| - nRF52832 | ❌ [^5] | ❌ [^4] | ❌ [^5] | ❌ [^4] | ❌ [^5] | ❌ [^4] | +| - nRF52810 | | | | | | | + +### ML-DSA 65 + +| Device | KeyGen[Debug] | KeyGen[Release] | Sign[Debug] | Sign[Release] | Verify[Debug] | Verify[Release] | +|---------------------------------|---------------|-----------------|-------------|---------------|---------------|-----------------| +| Raspberry Pi 3 | | | | | | | +| Raspberry Pi 4 | | | | | | | +| [ESP32-S3] [^1] [^2] | | | | | | | +| [STM32-L4R5xx] (our Nucleo-144) | 36271715 | 6382246 | 47606417 | 7981373 | 37005196 | 6501876 | +| [ESP32-C6] [^3] | | | | | | | +| [nRF52840-DK] | 47073665 | 6638977 | 62247859 | 8364565 | 48188918 | 6767475 | +| [nRF5340-DK] | | | | | | | +| [nRF52-DK] | | | | | | | +| - nRF52832 | ❌ [^5] | ❌ [^6] | ❌ [^5] | ❌ [^6] | ❌ [^5] | ❌ [^6] | +| - nRF52810 | | | | | | | + + +### ML-DSA 87 + +| Device | KeyGen[Debug] | KeyGen[Release] | Sign[Debug] | Sign[Release] | Verify[Debug] | Verify[Release] | +|---------------------------------|---------------|-----------------|-------------|---------------|---------------|-----------------| +| Raspberry Pi 3 | | | | | | | +| Raspberry Pi 4 | | | | | | | +| [ESP32-S3] [^1] [^2] | | | | | | | +| [STM32-L4R5xx] (our Nucleo-144) | 60053867 | 10633875 | 137335754 | 21909905 | 61676143 | 10794553 | +| [ESP32-C6] [^3] | | | | | | | +| [nRF52840-DK] | ❌ [^5] | ❌ [^4] | ❌ [^5] | ❌ [^4] | ❌ [^5] | ❌ [^4] | +| [nRF5340-DK] | | | | | | | +| [nRF52-DK] | | | | | | | +| - nRF52832 | ❌ [^5] | ❌ [^6] | ❌ [^5] | ❌ [^6] | ❌ [^5] | ❌ [^6] | +| - nRF52810 | | | | | | | + + +[STM32-L4R5xx]: https://www.st.com/en/microcontrollers-microprocessors/stm32l4r5zi.html?rt=db&id=DB3171 +[ESP32-C6]: https://www.espressif.com/en/products/socs/esp32-c6 +[STM32-L476RG]: https://www.st.com/en/microcontrollers-microprocessors/stm32l476rg.html?rt=db&id=DB2196 +[nRF52840-DK]: https://www.nordicsemi.com/Products/nRF52840 +[nRF5340-DK]: https://www.nordicsemi.com/Products/nRF5340 +[nRF52-DK]: https://www.nordicsemi.com/Products/Development-hardware/nRF52-DK +[esp32-s3]: https://www.espressif.com/en/products/socs/esp32-s3 +[^1]: https://docs.espressif.com/projects/esp-idf/en/latest/esp32s3/hw-reference/esp32s3/user-guide-devkitc-1.html +[^2]: https://www.espressif.com/sites/default/files/documentation/esp32-s3-wroom-1_wroom-1u_datasheet_en.pdf +[^3]: https://www.espressif.com/sites/default/files/documentation/esp32-c6-wroom-1_wroom-1u_datasheet_en.pdf +[^4]: The benchmark crashses due to insufficient stack space. +[^5]: Core locks up. +[^6]: Benchmark binary does not fit available flash space. diff --git a/libcrux-nrf52810/.gitignore b/libcrux-nrf52810/.gitignore new file mode 100644 index 0000000..96ef6c0 --- /dev/null +++ b/libcrux-nrf52810/.gitignore @@ -0,0 +1,2 @@ +/target +Cargo.lock diff --git a/libcrux-nrf52810/Cargo.toml b/libcrux-nrf52810/Cargo.toml new file mode 100644 index 0000000..49d8de3 --- /dev/null +++ b/libcrux-nrf52810/Cargo.toml @@ -0,0 +1,76 @@ +[package] +authors = ["Jonas Schneider-Bensch "] +name = "libcrux-nrf52810" +edition = "2021" +version = "0.1.0" + +[lib] +harness = false + +# needed for each integration test +[[test]] +name = "integration" +harness = false + +[dependencies] +cortex-m = { version = "0.7", features = ["critical-section-single-core"] } +cortex-m-rt = "0.7" +defmt = "0.3" +defmt-rtt = "0.4" +panic-probe = { version = "0.3", features = ["print-defmt"] } +cortex-m-semihosting = "0.5.0" +libcrux-iot-testutil = { path = "../libcrux-iot-testutil" } +libcrux-testbench = { path = "../libcrux-testbench" } +embassy-nrf = { version = "0.1.0", features = [ "nrf52810", "defmt", ] } +embedded-alloc = "0.6.0" + +[dev-dependencies] +defmt-test = "0.3" + +[features] +default = ["mldsa87", "mlkem1024"] # We want to benchmark the most expensive variants by default +mldsa44 = ["libcrux-testbench/mldsa44"] +mldsa65 = ["libcrux-testbench/mldsa65"] +mldsa87 = ["libcrux-testbench/mldsa87"] +mlkem512 = ["libcrux-testbench/mlkem512"] +mlkem768 = ["libcrux-testbench/mlkem768"] +mlkem1024 = ["libcrux-testbench/mlkem1024"] + +# cargo build/run +[profile.dev] +codegen-units = 1 +debug = 2 +debug-assertions = true # <- +incremental = false +opt-level = 'z' # <- +overflow-checks = true # <- + +# cargo test +[profile.test] +codegen-units = 1 +debug = 2 +debug-assertions = true # <- +incremental = false +opt-level = 3 # <- +overflow-checks = true # <- + +# cargo build/run --release +[profile.release] +codegen-units = 1 +debug = 2 +debug-assertions = false # <- +incremental = false +lto = 'fat' +opt-level = 3 # <- +overflow-checks = false # <- + +# cargo test --release +[profile.bench] +codegen-units = 1 +debug = 2 +debug-assertions = false # <- +incremental = false +lto = 'fat' +opt-level = 3 # <- +overflow-checks = false # <- + diff --git a/libcrux-nrf52810/README.md b/libcrux-nrf52810/README.md new file mode 100644 index 0000000..d86fd3e --- /dev/null +++ b/libcrux-nrf52810/README.md @@ -0,0 +1,32 @@ +# Libcrux ML-KEM / DSA on nRF52480 + +This setup is based on the app template found at [https://github.com/knurling-rs/app-template]. + +## Dependencies + +#### 1. `flip-link`: + +```console +$ cargo install flip-link +``` + +#### 2. `probe-rs`: + +``` console +$ # make sure to install v0.2.0 or later +$ cargo install probe-rs --features cli +``` + + +## Running Benchmarks + +With the device attached, run +```console +$ cargo rb mlkem --release +``` +for a crude benchmark of ML-KEM 1024 and + +```console +$ cargo rb mldsa_65 --release +``` +for a crude benchmark of ML-DSA 65. diff --git a/libcrux-nrf52810/memory.x b/libcrux-nrf52810/memory.x new file mode 100644 index 0000000..1be105b --- /dev/null +++ b/libcrux-nrf52810/memory.x @@ -0,0 +1,23 @@ +/* Linker script for the nRF52 - WITHOUT SOFT DEVICE */ +MEMORY +{ + /* NOTE K = KiBi = 1024 bytes */ + FLASH : ORIGIN = 0x00000000, LENGTH = 192K + RAM : ORIGIN = 0x20000000, LENGTH = 24K +} + +/* This is where the call stack will be allocated. */ +/* The stack is of the full descending type. */ +/* You may want to use this variable to locate the call stack and static + variables in different memory regions. Below is shown the default value */ +/* _stack_start = ORIGIN(RAM) + LENGTH(RAM); */ + +/* You can use this symbol to customize the location of the .text section */ +/* If omitted the .text section will be placed right after the .vector_table + section */ +/* This is required only on microcontrollers that store some configuration right + after the vector table */ +/* _stext = ORIGIN(FLASH) + 0x400; */ + +/* Size of the heap (in bytes) */ +/* _heap_size = 1024; */ diff --git a/libcrux-nrf52810/src/bin/mldsa.rs b/libcrux-nrf52810/src/bin/mldsa.rs new file mode 100644 index 0000000..fed07cd --- /dev/null +++ b/libcrux-nrf52810/src/bin/mldsa.rs @@ -0,0 +1,30 @@ +#![no_main] +#![no_std] + +use cortex_m::peripheral::Peripherals; +use libcrux_nrf52810 as _; // global logger + panicking-behavior + memory layout + +use core::ptr::addr_of_mut; +use embedded_alloc::LlffHeap as Heap; + +#[global_allocator] +static HEAP: Heap = Heap::empty(); + +#[cortex_m_rt::entry] +fn main() -> ! { + // Initialize the allocator BEFORE you use it + { + use core::mem::MaybeUninit; + const HEAP_SIZE: usize = 1024; + static mut HEAP_MEM: [MaybeUninit; HEAP_SIZE] = [MaybeUninit::uninit(); HEAP_SIZE]; + unsafe { HEAP.init(addr_of_mut!(HEAP_MEM) as usize, HEAP_SIZE) } + + let mut peripherals = Peripherals::take().unwrap(); + peripherals.DCB.enable_trace(); + peripherals.DWT.enable_cycle_counter(); + } + + libcrux_testbench::mldsa::run_benchmarks(); + + libcrux_nrf52810::exit() +} diff --git a/libcrux-nrf52810/src/bin/mlkem.rs b/libcrux-nrf52810/src/bin/mlkem.rs new file mode 100644 index 0000000..39ec1dd --- /dev/null +++ b/libcrux-nrf52810/src/bin/mlkem.rs @@ -0,0 +1,30 @@ +#![no_main] +#![no_std] + +use cortex_m::peripheral::Peripherals; +use libcrux_nrf52810 as _; // global logger + panicking-behavior + memory layout + +use core::ptr::addr_of_mut; +use embedded_alloc::LlffHeap as Heap; + +#[global_allocator] +static HEAP: Heap = Heap::empty(); + +#[cortex_m_rt::entry] +fn main() -> ! { + // Initialize the allocator BEFORE you use it + { + use core::mem::MaybeUninit; + const HEAP_SIZE: usize = 1024; + static mut HEAP_MEM: [MaybeUninit; HEAP_SIZE] = [MaybeUninit::uninit(); HEAP_SIZE]; + unsafe { HEAP.init(addr_of_mut!(HEAP_MEM) as usize, HEAP_SIZE) } + + let mut peripherals = Peripherals::take().unwrap(); + peripherals.DCB.enable_trace(); + peripherals.DWT.enable_cycle_counter(); + } + + libcrux_testbench::mlkem::run_benchmarks(); + + libcrux_nrf52810::exit() +} diff --git a/libcrux-nrf52810/src/lib.rs b/libcrux-nrf52810/src/lib.rs new file mode 100644 index 0000000..d551cd1 --- /dev/null +++ b/libcrux-nrf52810/src/lib.rs @@ -0,0 +1,51 @@ +#![no_main] +#![no_std] + +use cortex_m_semihosting::debug; + +use defmt_rtt as _; // global logger + +use embassy_nrf as _; // memory layout + +use panic_probe as _; + +// same panicking *behavior* as `panic-probe` but doesn't print a panic message +// this prevents the panic message being printed *twice* when `defmt::panic` is invoked +#[defmt::panic_handler] +fn panic() -> ! { + cortex_m::asm::udf() +} + +/// Terminates the application and makes a semihosting-capable debug tool exit +/// with status code 0. +pub fn exit() -> ! { + loop { + debug::exit(debug::EXIT_SUCCESS); + } +} + +/// Hardfault handler. +/// +/// Terminates the application and makes a semihosting-capable debug tool exit +/// with an error. This seems better than the default, which is to spin in a +/// loop. +#[cortex_m_rt::exception] +unsafe fn HardFault(_frame: &cortex_m_rt::ExceptionFrame) -> ! { + loop { + debug::exit(debug::EXIT_FAILURE); + } +} + +// defmt-test 0.3.0 has the limitation that this `#[tests]` attribute can only be used +// once within a crate. the module can be in any file but there can only be at most +// one `#[tests]` module in this library crate +#[cfg(test)] +#[defmt_test::tests] +mod unit_tests { + use defmt::assert; + + #[test] + fn it_works() { + assert!(true) + } +} diff --git a/libcrux-nrf52810/tests/integration.rs b/libcrux-nrf52810/tests/integration.rs new file mode 100644 index 0000000..013a8a4 --- /dev/null +++ b/libcrux-nrf52810/tests/integration.rs @@ -0,0 +1,16 @@ +#![no_std] +#![no_main] + +use libcrux_nrf52840 as _; // memory layout + panic handler + +// See https://crates.io/crates/defmt-test/0.3.0 for more documentation (e.g. about the 'state' +// feature) +#[defmt_test::tests] +mod tests { + use defmt::assert; + + #[test] + fn it_works() { + assert!(true) + } +} diff --git a/libcrux-nrf52832/.gitignore b/libcrux-nrf52832/.gitignore new file mode 100644 index 0000000..96ef6c0 --- /dev/null +++ b/libcrux-nrf52832/.gitignore @@ -0,0 +1,2 @@ +/target +Cargo.lock diff --git a/libcrux-nrf52832/Cargo.toml b/libcrux-nrf52832/Cargo.toml new file mode 100644 index 0000000..2813266 --- /dev/null +++ b/libcrux-nrf52832/Cargo.toml @@ -0,0 +1,76 @@ +[package] +authors = ["Jonas Schneider-Bensch "] +name = "libcrux-nrf52832" +edition = "2021" +version = "0.1.0" + +[lib] +harness = false + +# needed for each integration test +[[test]] +name = "integration" +harness = false + +[dependencies] +cortex-m = { version = "0.7", features = ["critical-section-single-core"] } +cortex-m-rt = "0.7" +defmt = "0.3" +defmt-rtt = "0.4" +panic-probe = { version = "0.3", features = ["print-defmt"] } +cortex-m-semihosting = "0.5.0" +libcrux-iot-testutil = { path = "../libcrux-iot-testutil" } +libcrux-testbench = { path = "../libcrux-testbench" } +embassy-nrf = { version = "0.1.0", features = [ "nrf52832", "defmt", ] } +embedded-alloc = "0.6.0" + +[dev-dependencies] +defmt-test = "0.3" + +[features] +default = ["mldsa87", "mlkem1024"] # We want to benchmark the most expensive variants by default +mldsa44 = ["libcrux-testbench/mldsa44"] +mldsa65 = ["libcrux-testbench/mldsa65"] +mldsa87 = ["libcrux-testbench/mldsa87"] +mlkem512 = ["libcrux-testbench/mlkem512"] +mlkem768 = ["libcrux-testbench/mlkem768"] +mlkem1024 = ["libcrux-testbench/mlkem1024"] + +# cargo build/run +[profile.dev] +codegen-units = 1 +debug = 2 +debug-assertions = true # <- +incremental = false +opt-level = 'z' # <- +overflow-checks = true # <- + +# cargo test +[profile.test] +codegen-units = 1 +debug = 2 +debug-assertions = true # <- +incremental = false +opt-level = 3 # <- +overflow-checks = true # <- + +# cargo build/run --release +[profile.release] +codegen-units = 1 +debug = 2 +debug-assertions = false # <- +incremental = false +lto = 'fat' +opt-level = 3 # <- +overflow-checks = false # <- + +# cargo test --release +[profile.bench] +codegen-units = 1 +debug = 2 +debug-assertions = false # <- +incremental = false +lto = 'fat' +opt-level = 3 # <- +overflow-checks = false # <- + diff --git a/libcrux-nrf52832/README.md b/libcrux-nrf52832/README.md new file mode 100644 index 0000000..d86fd3e --- /dev/null +++ b/libcrux-nrf52832/README.md @@ -0,0 +1,32 @@ +# Libcrux ML-KEM / DSA on nRF52480 + +This setup is based on the app template found at [https://github.com/knurling-rs/app-template]. + +## Dependencies + +#### 1. `flip-link`: + +```console +$ cargo install flip-link +``` + +#### 2. `probe-rs`: + +``` console +$ # make sure to install v0.2.0 or later +$ cargo install probe-rs --features cli +``` + + +## Running Benchmarks + +With the device attached, run +```console +$ cargo rb mlkem --release +``` +for a crude benchmark of ML-KEM 1024 and + +```console +$ cargo rb mldsa_65 --release +``` +for a crude benchmark of ML-DSA 65. diff --git a/libcrux-nrf52832/memory.x b/libcrux-nrf52832/memory.x new file mode 100644 index 0000000..917f43d --- /dev/null +++ b/libcrux-nrf52832/memory.x @@ -0,0 +1,23 @@ +/* Linker script for the nRF52 - WITHOUT SOFT DEVICE */ +MEMORY +{ + /* NOTE K = KiBi = 1024 bytes */ + FLASH : ORIGIN = 0x00000000, LENGTH = 512K + RAM : ORIGIN = 0x20000000, LENGTH = 64K +} + +/* This is where the call stack will be allocated. */ +/* The stack is of the full descending type. */ +/* You may want to use this variable to locate the call stack and static + variables in different memory regions. Below is shown the default value */ +/* _stack_start = ORIGIN(RAM) + LENGTH(RAM); */ + +/* You can use this symbol to customize the location of the .text section */ +/* If omitted the .text section will be placed right after the .vector_table + section */ +/* This is required only on microcontrollers that store some configuration right + after the vector table */ +/* _stext = ORIGIN(FLASH) + 0x400; */ + +/* Size of the heap (in bytes) */ +/* _heap_size = 1024; */ diff --git a/libcrux-nrf52832/src/bin/mldsa.rs b/libcrux-nrf52832/src/bin/mldsa.rs new file mode 100644 index 0000000..f72d4c7 --- /dev/null +++ b/libcrux-nrf52832/src/bin/mldsa.rs @@ -0,0 +1,30 @@ +#![no_main] +#![no_std] + +use cortex_m::peripheral::Peripherals; +use libcrux_nrf52832 as _; // global logger + panicking-behavior + memory layout + +use core::ptr::addr_of_mut; +use embedded_alloc::LlffHeap as Heap; + +#[global_allocator] +static HEAP: Heap = Heap::empty(); + +#[cortex_m_rt::entry] +fn main() -> ! { + // Initialize the allocator BEFORE you use it + { + use core::mem::MaybeUninit; + const HEAP_SIZE: usize = 1024; + static mut HEAP_MEM: [MaybeUninit; HEAP_SIZE] = [MaybeUninit::uninit(); HEAP_SIZE]; + unsafe { HEAP.init(addr_of_mut!(HEAP_MEM) as usize, HEAP_SIZE) } + + let mut peripherals = Peripherals::take().unwrap(); + peripherals.DCB.enable_trace(); + peripherals.DWT.enable_cycle_counter(); + } + + libcrux_testbench::mldsa::run_benchmarks(); + + libcrux_nrf52832::exit() +} diff --git a/libcrux-nrf52832/src/bin/mlkem.rs b/libcrux-nrf52832/src/bin/mlkem.rs new file mode 100644 index 0000000..1e918e7 --- /dev/null +++ b/libcrux-nrf52832/src/bin/mlkem.rs @@ -0,0 +1,30 @@ +#![no_main] +#![no_std] + +use cortex_m::peripheral::Peripherals; +use libcrux_nrf52832 as _; // global logger + panicking-behavior + memory layout + +use core::ptr::addr_of_mut; +use embedded_alloc::LlffHeap as Heap; + +#[global_allocator] +static HEAP: Heap = Heap::empty(); + +#[cortex_m_rt::entry] +fn main() -> ! { + // Initialize the allocator BEFORE you use it + { + use core::mem::MaybeUninit; + const HEAP_SIZE: usize = 1024; + static mut HEAP_MEM: [MaybeUninit; HEAP_SIZE] = [MaybeUninit::uninit(); HEAP_SIZE]; + unsafe { HEAP.init(addr_of_mut!(HEAP_MEM) as usize, HEAP_SIZE) } + + let mut peripherals = Peripherals::take().unwrap(); + peripherals.DCB.enable_trace(); + peripherals.DWT.enable_cycle_counter(); + } + + libcrux_testbench::mlkem::run_benchmarks(); + + libcrux_nrf52832::exit() +} diff --git a/libcrux-nrf52832/src/lib.rs b/libcrux-nrf52832/src/lib.rs new file mode 100644 index 0000000..d551cd1 --- /dev/null +++ b/libcrux-nrf52832/src/lib.rs @@ -0,0 +1,51 @@ +#![no_main] +#![no_std] + +use cortex_m_semihosting::debug; + +use defmt_rtt as _; // global logger + +use embassy_nrf as _; // memory layout + +use panic_probe as _; + +// same panicking *behavior* as `panic-probe` but doesn't print a panic message +// this prevents the panic message being printed *twice* when `defmt::panic` is invoked +#[defmt::panic_handler] +fn panic() -> ! { + cortex_m::asm::udf() +} + +/// Terminates the application and makes a semihosting-capable debug tool exit +/// with status code 0. +pub fn exit() -> ! { + loop { + debug::exit(debug::EXIT_SUCCESS); + } +} + +/// Hardfault handler. +/// +/// Terminates the application and makes a semihosting-capable debug tool exit +/// with an error. This seems better than the default, which is to spin in a +/// loop. +#[cortex_m_rt::exception] +unsafe fn HardFault(_frame: &cortex_m_rt::ExceptionFrame) -> ! { + loop { + debug::exit(debug::EXIT_FAILURE); + } +} + +// defmt-test 0.3.0 has the limitation that this `#[tests]` attribute can only be used +// once within a crate. the module can be in any file but there can only be at most +// one `#[tests]` module in this library crate +#[cfg(test)] +#[defmt_test::tests] +mod unit_tests { + use defmt::assert; + + #[test] + fn it_works() { + assert!(true) + } +} diff --git a/libcrux-nrf52832/tests/integration.rs b/libcrux-nrf52832/tests/integration.rs new file mode 100644 index 0000000..013a8a4 --- /dev/null +++ b/libcrux-nrf52832/tests/integration.rs @@ -0,0 +1,16 @@ +#![no_std] +#![no_main] + +use libcrux_nrf52840 as _; // memory layout + panic handler + +// See https://crates.io/crates/defmt-test/0.3.0 for more documentation (e.g. about the 'state' +// feature) +#[defmt_test::tests] +mod tests { + use defmt::assert; + + #[test] + fn it_works() { + assert!(true) + } +} diff --git a/libcrux-nrf5340/.gitignore b/libcrux-nrf5340/.gitignore new file mode 100644 index 0000000..96ef6c0 --- /dev/null +++ b/libcrux-nrf5340/.gitignore @@ -0,0 +1,2 @@ +/target +Cargo.lock diff --git a/libcrux-nrf5340/Cargo.toml b/libcrux-nrf5340/Cargo.toml new file mode 100644 index 0000000..0bf1ba6 --- /dev/null +++ b/libcrux-nrf5340/Cargo.toml @@ -0,0 +1,76 @@ +[package] +authors = ["Jonas Schneider-Bensch "] +name = "libcrux-nrf5340" +edition = "2021" +version = "0.1.0" + +[lib] +harness = false + +# needed for each integration test +[[test]] +name = "integration" +harness = false + +[dependencies] +cortex-m = { version = "0.7", features = ["critical-section-single-core"] } +cortex-m-rt = "0.7" +defmt = "0.3" +defmt-rtt = "0.4" +panic-probe = { version = "0.3", features = ["print-defmt"] } +cortex-m-semihosting = "0.5.0" +libcrux-iot-testutil = { path = "../libcrux-iot-testutil" } +libcrux-testbench = { path = "../libcrux-testbench" } +embassy-nrf = { version = "0.1.0", features = [ "nrf5340-app-ns", "defmt", ] } +embedded-alloc = "0.6.0" + +[dev-dependencies] +defmt-test = "0.3" + +[features] +default = ["mldsa87", "mlkem1024"] # We want to benchmark the most expensive variants by default +mldsa44 = ["libcrux-testbench/mldsa44"] +mldsa65 = ["libcrux-testbench/mldsa65"] +mldsa87 = ["libcrux-testbench/mldsa87"] +mlkem512 = ["libcrux-testbench/mlkem512"] +mlkem768 = ["libcrux-testbench/mlkem768"] +mlkem1024 = ["libcrux-testbench/mlkem1024"] + +# cargo build/run +[profile.dev] +codegen-units = 1 +debug = 2 +debug-assertions = true # <- +incremental = false +opt-level = 'z' # <- +overflow-checks = true # <- + +# cargo test +[profile.test] +codegen-units = 1 +debug = 2 +debug-assertions = true # <- +incremental = false +opt-level = 3 # <- +overflow-checks = true # <- + +# cargo build/run --release +[profile.release] +codegen-units = 1 +debug = 2 +debug-assertions = false # <- +incremental = false +lto = 'fat' +opt-level = 3 # <- +overflow-checks = false # <- + +# cargo test --release +[profile.bench] +codegen-units = 1 +debug = 2 +debug-assertions = false # <- +incremental = false +lto = 'fat' +opt-level = 3 # <- +overflow-checks = false # <- + diff --git a/libcrux-nrf5340/README.md b/libcrux-nrf5340/README.md new file mode 100644 index 0000000..d86fd3e --- /dev/null +++ b/libcrux-nrf5340/README.md @@ -0,0 +1,32 @@ +# Libcrux ML-KEM / DSA on nRF52480 + +This setup is based on the app template found at [https://github.com/knurling-rs/app-template]. + +## Dependencies + +#### 1. `flip-link`: + +```console +$ cargo install flip-link +``` + +#### 2. `probe-rs`: + +``` console +$ # make sure to install v0.2.0 or later +$ cargo install probe-rs --features cli +``` + + +## Running Benchmarks + +With the device attached, run +```console +$ cargo rb mlkem --release +``` +for a crude benchmark of ML-KEM 1024 and + +```console +$ cargo rb mldsa_65 --release +``` +for a crude benchmark of ML-DSA 65. diff --git a/libcrux-nrf5340/memory.x b/libcrux-nrf5340/memory.x new file mode 100644 index 0000000..3dffc20 --- /dev/null +++ b/libcrux-nrf5340/memory.x @@ -0,0 +1,34 @@ +/* Linker script for the nRF5340 in Non-secure mode. It assumes you have the +Nordic Secure Partition Manager installed at the bottom of flash and that +the SPM is set to boot a non-secure application from the FLASH origin below. */ + +/* +MEMORY +{ + /* + * This is where the Bootloader, Secure Partition Manager or + * Trusted-Firmware-M lives. + */ + /* + SECURE_FLASH : ORIGIN = 0x00000000, LENGTH = 256K + * This is where your non-secure Rust application lives. Note that SPM must agree this + * is where your application lives, or it will jump to garbage and crash the CPU. + SIG : ORIGIN = 0x00050000, LENGTH = 1K + */ + FLASH : ORIGIN = 0x00050000, LENGTH = 767K + /* FLASH : ORIGIN = 0x00000000, LENGTH = 767K */ + /* + * This RAM is reserved for the Secure-Mode code located in the `SECURE_FLASH` region. + */ + SECURE_RAM : ORIGIN = 0x20000000, LENGTH = 64K + /* + * This RAM is available to your non-secure Rust application. + */ + RAM : ORIGIN = 0x20020000, LENGTH = 128K +} */ + +MEMORY +{ + FLASH : ORIGIN = 0x00000000, LENGTH = 1024K + RAM : ORIGIN = 0x20000000, LENGTH = 256K +} diff --git a/libcrux-nrf5340/src/bin/mldsa.rs b/libcrux-nrf5340/src/bin/mldsa.rs new file mode 100644 index 0000000..086fdd7 --- /dev/null +++ b/libcrux-nrf5340/src/bin/mldsa.rs @@ -0,0 +1,30 @@ +#![no_main] +#![no_std] + +use cortex_m::peripheral::Peripherals; +use libcrux_nrf5340 as _; // global logger + panicking-behavior + memory layout + +use core::ptr::addr_of_mut; +use embedded_alloc::LlffHeap as Heap; + +#[global_allocator] +static HEAP: Heap = Heap::empty(); + +#[cortex_m_rt::entry] +fn main() -> ! { + // Initialize the allocator BEFORE you use it + { + use core::mem::MaybeUninit; + const HEAP_SIZE: usize = 1024; + static mut HEAP_MEM: [MaybeUninit; HEAP_SIZE] = [MaybeUninit::uninit(); HEAP_SIZE]; + unsafe { HEAP.init(addr_of_mut!(HEAP_MEM) as usize, HEAP_SIZE) } + + let mut peripherals = Peripherals::take().unwrap(); + peripherals.DCB.enable_trace(); + peripherals.DWT.enable_cycle_counter(); + } + + libcrux_testbench::mldsa::run_benchmarks(); + + libcrux_nrf5340::exit() +} diff --git a/libcrux-nrf5340/src/bin/mlkem.rs b/libcrux-nrf5340/src/bin/mlkem.rs new file mode 100644 index 0000000..4019da8 --- /dev/null +++ b/libcrux-nrf5340/src/bin/mlkem.rs @@ -0,0 +1,30 @@ +#![no_main] +#![no_std] + +use cortex_m::peripheral::Peripherals; +use libcrux_nrf5340 as _; // global logger + panicking-behavior + memory layout + +use core::ptr::addr_of_mut; +use embedded_alloc::LlffHeap as Heap; + +#[global_allocator] +static HEAP: Heap = Heap::empty(); + +#[cortex_m_rt::entry] +fn main() -> ! { + // Initialize the allocator BEFORE you use it + { + use core::mem::MaybeUninit; + const HEAP_SIZE: usize = 1024; + static mut HEAP_MEM: [MaybeUninit; HEAP_SIZE] = [MaybeUninit::uninit(); HEAP_SIZE]; + unsafe { HEAP.init(addr_of_mut!(HEAP_MEM) as usize, HEAP_SIZE) } + + let mut peripherals = Peripherals::take().unwrap(); + peripherals.DCB.enable_trace(); + peripherals.DWT.enable_cycle_counter(); + } + + libcrux_testbench::mlkem::run_benchmarks(); + + libcrux_nrf5340::exit() +} diff --git a/libcrux-nrf5340/src/lib.rs b/libcrux-nrf5340/src/lib.rs new file mode 100644 index 0000000..d551cd1 --- /dev/null +++ b/libcrux-nrf5340/src/lib.rs @@ -0,0 +1,51 @@ +#![no_main] +#![no_std] + +use cortex_m_semihosting::debug; + +use defmt_rtt as _; // global logger + +use embassy_nrf as _; // memory layout + +use panic_probe as _; + +// same panicking *behavior* as `panic-probe` but doesn't print a panic message +// this prevents the panic message being printed *twice* when `defmt::panic` is invoked +#[defmt::panic_handler] +fn panic() -> ! { + cortex_m::asm::udf() +} + +/// Terminates the application and makes a semihosting-capable debug tool exit +/// with status code 0. +pub fn exit() -> ! { + loop { + debug::exit(debug::EXIT_SUCCESS); + } +} + +/// Hardfault handler. +/// +/// Terminates the application and makes a semihosting-capable debug tool exit +/// with an error. This seems better than the default, which is to spin in a +/// loop. +#[cortex_m_rt::exception] +unsafe fn HardFault(_frame: &cortex_m_rt::ExceptionFrame) -> ! { + loop { + debug::exit(debug::EXIT_FAILURE); + } +} + +// defmt-test 0.3.0 has the limitation that this `#[tests]` attribute can only be used +// once within a crate. the module can be in any file but there can only be at most +// one `#[tests]` module in this library crate +#[cfg(test)] +#[defmt_test::tests] +mod unit_tests { + use defmt::assert; + + #[test] + fn it_works() { + assert!(true) + } +} diff --git a/libcrux-nrf5340/tests/integration.rs b/libcrux-nrf5340/tests/integration.rs new file mode 100644 index 0000000..013a8a4 --- /dev/null +++ b/libcrux-nrf5340/tests/integration.rs @@ -0,0 +1,16 @@ +#![no_std] +#![no_main] + +use libcrux_nrf52840 as _; // memory layout + panic handler + +// See https://crates.io/crates/defmt-test/0.3.0 for more documentation (e.g. about the 'state' +// feature) +#[defmt_test::tests] +mod tests { + use defmt::assert; + + #[test] + fn it_works() { + assert!(true) + } +} From d8424de0f22bca48caf95cdac63e22336bdb7f24 Mon Sep 17 00:00:00 2001 From: Jonas Schneider-Bensch Date: Thu, 31 Oct 2024 14:32:55 +0100 Subject: [PATCH 08/12] Let caller provide testbench config --- libcrux-nrf52810/src/bin/mldsa.rs | 18 +++++++++++++++--- libcrux-nrf52810/src/bin/mlkem.rs | 17 ++++++++++++++--- libcrux-nrf52810/src/lib.rs | 2 ++ libcrux-nrf52832/src/bin/mldsa.rs | 18 +++++++++++++++--- libcrux-nrf52832/src/bin/mlkem.rs | 17 ++++++++++++++--- libcrux-nrf52832/src/lib.rs | 2 ++ libcrux-nrf52840/src/bin/mldsa.rs | 18 +++++++++++++++--- libcrux-nrf52840/src/bin/mlkem.rs | 17 ++++++++++++++--- libcrux-nrf52840/src/lib.rs | 2 ++ libcrux-nrf5340/src/bin/mldsa.rs | 18 +++++++++++++++--- libcrux-nrf5340/src/bin/mlkem.rs | 17 ++++++++++++++--- libcrux-nrf5340/src/lib.rs | 2 ++ libcrux-testbench/src/mldsa.rs | 10 +--------- libcrux-testbench/src/mlkem.rs | 11 +---------- 14 files changed, 126 insertions(+), 43 deletions(-) diff --git a/libcrux-nrf52810/src/bin/mldsa.rs b/libcrux-nrf52810/src/bin/mldsa.rs index fed07cd..4838699 100644 --- a/libcrux-nrf52810/src/bin/mldsa.rs +++ b/libcrux-nrf52810/src/bin/mldsa.rs @@ -2,7 +2,9 @@ #![no_std] use cortex_m::peripheral::Peripherals; -use libcrux_nrf52810 as _; // global logger + panicking-behavior + memory layout +use libcrux_nrf52810 as board; // global logger + panicking-behavior + memory layout + +extern crate alloc; use core::ptr::addr_of_mut; use embedded_alloc::LlffHeap as Heap; @@ -12,6 +14,8 @@ static HEAP: Heap = Heap::empty(); #[cortex_m_rt::entry] fn main() -> ! { + use libcrux_iot_testutil::*; + // Initialize the allocator BEFORE you use it { use core::mem::MaybeUninit; @@ -23,8 +27,16 @@ fn main() -> ! { peripherals.DCB.enable_trace(); peripherals.DWT.enable_cycle_counter(); } + + // set up the test config + let test_config = TestConfig { + core_freq: board::COREFREQ, + only_names: alloc::vec![], + early_abort: false, + benchmark_runs: 5, + }; - libcrux_testbench::mldsa::run_benchmarks(); + libcrux_testbench::mldsa::run_benchmarks(test_config); - libcrux_nrf52810::exit() + board::exit() } diff --git a/libcrux-nrf52810/src/bin/mlkem.rs b/libcrux-nrf52810/src/bin/mlkem.rs index 39ec1dd..57a36e4 100644 --- a/libcrux-nrf52810/src/bin/mlkem.rs +++ b/libcrux-nrf52810/src/bin/mlkem.rs @@ -2,7 +2,9 @@ #![no_std] use cortex_m::peripheral::Peripherals; -use libcrux_nrf52810 as _; // global logger + panicking-behavior + memory layout +use libcrux_nrf52810 as board; // global logger + panicking-behavior + memory layout + +extern crate alloc; use core::ptr::addr_of_mut; use embedded_alloc::LlffHeap as Heap; @@ -12,6 +14,7 @@ static HEAP: Heap = Heap::empty(); #[cortex_m_rt::entry] fn main() -> ! { + use libcrux_iot_testutil::*; // Initialize the allocator BEFORE you use it { use core::mem::MaybeUninit; @@ -24,7 +27,15 @@ fn main() -> ! { peripherals.DWT.enable_cycle_counter(); } - libcrux_testbench::mlkem::run_benchmarks(); + // set up the test config + let test_config = TestConfig { + core_freq: board::COREFREQ, + only_names: alloc::vec![], + early_abort: false, + benchmark_runs: 5, + }; + + libcrux_testbench::mlkem::run_benchmarks(test_config); - libcrux_nrf52810::exit() + board::exit() } diff --git a/libcrux-nrf52810/src/lib.rs b/libcrux-nrf52810/src/lib.rs index d551cd1..10cc03c 100644 --- a/libcrux-nrf52810/src/lib.rs +++ b/libcrux-nrf52810/src/lib.rs @@ -9,6 +9,8 @@ use embassy_nrf as _; // memory layout use panic_probe as _; +pub const COREFREQ: u32 = 4_000_000; + // same panicking *behavior* as `panic-probe` but doesn't print a panic message // this prevents the panic message being printed *twice* when `defmt::panic` is invoked #[defmt::panic_handler] diff --git a/libcrux-nrf52832/src/bin/mldsa.rs b/libcrux-nrf52832/src/bin/mldsa.rs index f72d4c7..a2534f6 100644 --- a/libcrux-nrf52832/src/bin/mldsa.rs +++ b/libcrux-nrf52832/src/bin/mldsa.rs @@ -2,7 +2,9 @@ #![no_std] use cortex_m::peripheral::Peripherals; -use libcrux_nrf52832 as _; // global logger + panicking-behavior + memory layout +use libcrux_nrf52832 as board; // global logger + panicking-behavior + memory layout + +extern crate alloc; use core::ptr::addr_of_mut; use embedded_alloc::LlffHeap as Heap; @@ -12,6 +14,8 @@ static HEAP: Heap = Heap::empty(); #[cortex_m_rt::entry] fn main() -> ! { + use libcrux_iot_testutil::*; + // Initialize the allocator BEFORE you use it { use core::mem::MaybeUninit; @@ -23,8 +27,16 @@ fn main() -> ! { peripherals.DCB.enable_trace(); peripherals.DWT.enable_cycle_counter(); } + + // set up the test config + let test_config = TestConfig { + core_freq: board::COREFREQ, + only_names: alloc::vec![], + early_abort: false, + benchmark_runs: 5, + }; - libcrux_testbench::mldsa::run_benchmarks(); + libcrux_testbench::mldsa::run_benchmarks(test_config); - libcrux_nrf52832::exit() + board::exit() } diff --git a/libcrux-nrf52832/src/bin/mlkem.rs b/libcrux-nrf52832/src/bin/mlkem.rs index 1e918e7..159bca6 100644 --- a/libcrux-nrf52832/src/bin/mlkem.rs +++ b/libcrux-nrf52832/src/bin/mlkem.rs @@ -2,7 +2,9 @@ #![no_std] use cortex_m::peripheral::Peripherals; -use libcrux_nrf52832 as _; // global logger + panicking-behavior + memory layout +use libcrux_nrf52832 as board; // global logger + panicking-behavior + memory layout + +extern crate alloc; use core::ptr::addr_of_mut; use embedded_alloc::LlffHeap as Heap; @@ -12,6 +14,7 @@ static HEAP: Heap = Heap::empty(); #[cortex_m_rt::entry] fn main() -> ! { + use libcrux_iot_testutil::*; // Initialize the allocator BEFORE you use it { use core::mem::MaybeUninit; @@ -24,7 +27,15 @@ fn main() -> ! { peripherals.DWT.enable_cycle_counter(); } - libcrux_testbench::mlkem::run_benchmarks(); + // set up the test config + let test_config = TestConfig { + core_freq: board::COREFREQ, + only_names: alloc::vec![], + early_abort: false, + benchmark_runs: 5, + }; + + libcrux_testbench::mlkem::run_benchmarks(test_config); - libcrux_nrf52832::exit() + board::exit() } diff --git a/libcrux-nrf52832/src/lib.rs b/libcrux-nrf52832/src/lib.rs index d551cd1..10cc03c 100644 --- a/libcrux-nrf52832/src/lib.rs +++ b/libcrux-nrf52832/src/lib.rs @@ -9,6 +9,8 @@ use embassy_nrf as _; // memory layout use panic_probe as _; +pub const COREFREQ: u32 = 4_000_000; + // same panicking *behavior* as `panic-probe` but doesn't print a panic message // this prevents the panic message being printed *twice* when `defmt::panic` is invoked #[defmt::panic_handler] diff --git a/libcrux-nrf52840/src/bin/mldsa.rs b/libcrux-nrf52840/src/bin/mldsa.rs index 0939be0..b9d9633 100644 --- a/libcrux-nrf52840/src/bin/mldsa.rs +++ b/libcrux-nrf52840/src/bin/mldsa.rs @@ -2,7 +2,9 @@ #![no_std] use cortex_m::peripheral::Peripherals; -use libcrux_nrf52840 as _; // global logger + panicking-behavior + memory layout +use libcrux_nrf52840 as board; // global logger + panicking-behavior + memory layout + +extern crate alloc; use core::ptr::addr_of_mut; use embedded_alloc::LlffHeap as Heap; @@ -12,6 +14,8 @@ static HEAP: Heap = Heap::empty(); #[cortex_m_rt::entry] fn main() -> ! { + use libcrux_iot_testutil::*; + // Initialize the allocator BEFORE you use it { use core::mem::MaybeUninit; @@ -23,8 +27,16 @@ fn main() -> ! { peripherals.DCB.enable_trace(); peripherals.DWT.enable_cycle_counter(); } + + // set up the test config + let test_config = TestConfig { + core_freq: board::COREFREQ, + only_names: alloc::vec![], + early_abort: false, + benchmark_runs: 5, + }; - libcrux_testbench::mldsa::run_benchmarks(); + libcrux_testbench::mldsa::run_benchmarks(test_config); - libcrux_nrf52840::exit() + board::exit() } diff --git a/libcrux-nrf52840/src/bin/mlkem.rs b/libcrux-nrf52840/src/bin/mlkem.rs index 3d4349b..cefd045 100644 --- a/libcrux-nrf52840/src/bin/mlkem.rs +++ b/libcrux-nrf52840/src/bin/mlkem.rs @@ -2,7 +2,9 @@ #![no_std] use cortex_m::peripheral::Peripherals; -use libcrux_nrf52840 as _; // global logger + panicking-behavior + memory layout +use libcrux_nrf52840 as board; // global logger + panicking-behavior + memory layout + +extern crate alloc; use core::ptr::addr_of_mut; use embedded_alloc::LlffHeap as Heap; @@ -12,6 +14,7 @@ static HEAP: Heap = Heap::empty(); #[cortex_m_rt::entry] fn main() -> ! { + use libcrux_iot_testutil::*; // Initialize the allocator BEFORE you use it { use core::mem::MaybeUninit; @@ -24,7 +27,15 @@ fn main() -> ! { peripherals.DWT.enable_cycle_counter(); } - libcrux_testbench::mlkem::run_benchmarks(); + // set up the test config + let test_config = TestConfig { + core_freq: board::COREFREQ, + only_names: alloc::vec![], + early_abort: false, + benchmark_runs: 5, + }; + + libcrux_testbench::mlkem::run_benchmarks(test_config); - libcrux_nrf52840::exit() + board::exit() } diff --git a/libcrux-nrf52840/src/lib.rs b/libcrux-nrf52840/src/lib.rs index d551cd1..10cc03c 100644 --- a/libcrux-nrf52840/src/lib.rs +++ b/libcrux-nrf52840/src/lib.rs @@ -9,6 +9,8 @@ use embassy_nrf as _; // memory layout use panic_probe as _; +pub const COREFREQ: u32 = 4_000_000; + // same panicking *behavior* as `panic-probe` but doesn't print a panic message // this prevents the panic message being printed *twice* when `defmt::panic` is invoked #[defmt::panic_handler] diff --git a/libcrux-nrf5340/src/bin/mldsa.rs b/libcrux-nrf5340/src/bin/mldsa.rs index 086fdd7..a348af4 100644 --- a/libcrux-nrf5340/src/bin/mldsa.rs +++ b/libcrux-nrf5340/src/bin/mldsa.rs @@ -2,7 +2,9 @@ #![no_std] use cortex_m::peripheral::Peripherals; -use libcrux_nrf5340 as _; // global logger + panicking-behavior + memory layout +use libcrux_nrf5340 as board; // global logger + panicking-behavior + memory layout + +extern crate alloc; use core::ptr::addr_of_mut; use embedded_alloc::LlffHeap as Heap; @@ -12,6 +14,8 @@ static HEAP: Heap = Heap::empty(); #[cortex_m_rt::entry] fn main() -> ! { + use libcrux_iot_testutil::*; + // Initialize the allocator BEFORE you use it { use core::mem::MaybeUninit; @@ -23,8 +27,16 @@ fn main() -> ! { peripherals.DCB.enable_trace(); peripherals.DWT.enable_cycle_counter(); } + + // set up the test config + let test_config = TestConfig { + core_freq: board::COREFREQ, + only_names: alloc::vec![], + early_abort: false, + benchmark_runs: 5, + }; - libcrux_testbench::mldsa::run_benchmarks(); + libcrux_testbench::mldsa::run_benchmarks(test_config); - libcrux_nrf5340::exit() + board::exit() } diff --git a/libcrux-nrf5340/src/bin/mlkem.rs b/libcrux-nrf5340/src/bin/mlkem.rs index 4019da8..2b739c6 100644 --- a/libcrux-nrf5340/src/bin/mlkem.rs +++ b/libcrux-nrf5340/src/bin/mlkem.rs @@ -2,7 +2,9 @@ #![no_std] use cortex_m::peripheral::Peripherals; -use libcrux_nrf5340 as _; // global logger + panicking-behavior + memory layout +use libcrux_nrf5340 as board; // global logger + panicking-behavior + memory layout + +extern crate alloc; use core::ptr::addr_of_mut; use embedded_alloc::LlffHeap as Heap; @@ -12,6 +14,7 @@ static HEAP: Heap = Heap::empty(); #[cortex_m_rt::entry] fn main() -> ! { + use libcrux_iot_testutil::*; // Initialize the allocator BEFORE you use it { use core::mem::MaybeUninit; @@ -24,7 +27,15 @@ fn main() -> ! { peripherals.DWT.enable_cycle_counter(); } - libcrux_testbench::mlkem::run_benchmarks(); + // set up the test config + let test_config = TestConfig { + core_freq: board::COREFREQ, + only_names: alloc::vec![], + early_abort: false, + benchmark_runs: 5, + }; + + libcrux_testbench::mlkem::run_benchmarks(test_config); - libcrux_nrf5340::exit() + board::exit() } diff --git a/libcrux-nrf5340/src/lib.rs b/libcrux-nrf5340/src/lib.rs index d551cd1..10cc03c 100644 --- a/libcrux-nrf5340/src/lib.rs +++ b/libcrux-nrf5340/src/lib.rs @@ -9,6 +9,8 @@ use embassy_nrf as _; // memory layout use panic_probe as _; +pub const COREFREQ: u32 = 4_000_000; + // same panicking *behavior* as `panic-probe` but doesn't print a panic message // this prevents the panic message being printed *twice* when `defmt::panic` is invoked #[defmt::panic_handler] diff --git a/libcrux-testbench/src/mldsa.rs b/libcrux-testbench/src/mldsa.rs index 8ba74da..9c09d5f 100644 --- a/libcrux-testbench/src/mldsa.rs +++ b/libcrux-testbench/src/mldsa.rs @@ -2,7 +2,6 @@ use libcrux_iot_testutil::DefmtInfoLogger; use libcrux_iot_testutil::*; extern crate alloc; use alloc::string::String; -use alloc::vec; #[cfg(feature = "mldsa44")] use libcrux_ml_dsa::ml_dsa_44 as mldsa; @@ -59,7 +58,7 @@ fn bench_verify(_l: &mut L, state: &MLDSABenchState) -> Result<( Ok(()) } -pub fn run_benchmarks() { +pub fn run_benchmarks(test_config: TestConfig) { // set up the test suite let test_cases = [ TestCase::new("bench_keygen", bench_keygen), @@ -69,13 +68,6 @@ pub fn run_benchmarks() { let test_suite = TestSuite::new("ML-DSA Benchmark", &test_cases); - // set up the test config - let test_config = TestConfig { - core_freq: 4_000_000, - only_names: vec!["bench_keygen", "bench_sign", "bench_verify"], - early_abort: false, - benchmark_runs: 1, - }; // prepare the state for the benchmarked functions let randomness_gen = [1u8; 32]; diff --git a/libcrux-testbench/src/mlkem.rs b/libcrux-testbench/src/mlkem.rs index d797193..e86fb85 100644 --- a/libcrux-testbench/src/mlkem.rs +++ b/libcrux-testbench/src/mlkem.rs @@ -2,7 +2,6 @@ use libcrux_iot_testutil::DefmtInfoLogger; use libcrux_iot_testutil::*; extern crate alloc; use alloc::string::String; -use alloc::vec; #[cfg(feature = "mlkem512")] use libcrux_ml_kem::mlkem512 as mlkem; @@ -55,7 +54,7 @@ fn bench_decaps(_l: &mut L, state: &MlKemBenchState) -> Result<( Ok(()) } -pub fn run_benchmarks() { +pub fn run_benchmarks(test_config: TestConfig) { // set up the test suite let test_cases = [ TestCase::new("bench_keygen", bench_keygen), @@ -65,14 +64,6 @@ pub fn run_benchmarks() { let test_suite = TestSuite::new("ML-KEM Benchmark", &test_cases); - // set up the test config - let test_config = TestConfig { - core_freq: 4_000_000, - only_names: vec!["bench_keygen", "bench_encaps", "bench_decaps"], - early_abort: false, - benchmark_runs: 1, - }; - // prepare the state for the benchmarked functions let randomness_gen = [1u8; libcrux_ml_kem::KEY_GENERATION_SEED_SIZE]; let pair = mlkem::generate_key_pair(randomness_gen); From f024b8f4036e1a92c7399ed5d82a2ac8ac0bbc6e Mon Sep 17 00:00:00 2001 From: Jonas Schneider-Bensch Date: Thu, 31 Oct 2024 14:34:05 +0100 Subject: [PATCH 09/12] Fixed nRF5340 linker script --- libcrux-nrf5340/memory.x | 29 ----------------------------- 1 file changed, 29 deletions(-) diff --git a/libcrux-nrf5340/memory.x b/libcrux-nrf5340/memory.x index 3dffc20..58900a7 100644 --- a/libcrux-nrf5340/memory.x +++ b/libcrux-nrf5340/memory.x @@ -1,32 +1,3 @@ -/* Linker script for the nRF5340 in Non-secure mode. It assumes you have the -Nordic Secure Partition Manager installed at the bottom of flash and that -the SPM is set to boot a non-secure application from the FLASH origin below. */ - -/* -MEMORY -{ - /* - * This is where the Bootloader, Secure Partition Manager or - * Trusted-Firmware-M lives. - */ - /* - SECURE_FLASH : ORIGIN = 0x00000000, LENGTH = 256K - * This is where your non-secure Rust application lives. Note that SPM must agree this - * is where your application lives, or it will jump to garbage and crash the CPU. - SIG : ORIGIN = 0x00050000, LENGTH = 1K - */ - FLASH : ORIGIN = 0x00050000, LENGTH = 767K - /* FLASH : ORIGIN = 0x00000000, LENGTH = 767K */ - /* - * This RAM is reserved for the Secure-Mode code located in the `SECURE_FLASH` region. - */ - SECURE_RAM : ORIGIN = 0x20000000, LENGTH = 64K - /* - * This RAM is available to your non-secure Rust application. - */ - RAM : ORIGIN = 0x20020000, LENGTH = 128K -} */ - MEMORY { FLASH : ORIGIN = 0x00000000, LENGTH = 1024K From 324cc51eb5ede7126cdc5911748f5b6bc8dde2bb Mon Sep 17 00:00:00 2001 From: Jonas Schneider-Bensch Date: Thu, 31 Oct 2024 14:38:40 +0100 Subject: [PATCH 10/12] Fix l4r5zi build --- libcrux-nucleo-l4r5zi/src/bin/mldsa.rs | 17 +++++++++++++---- libcrux-nucleo-l4r5zi/src/bin/mlkem.rs | 16 +++++++++++++--- libcrux-nucleo-l4r5zi/src/lib.rs | 2 ++ 3 files changed, 28 insertions(+), 7 deletions(-) diff --git a/libcrux-nucleo-l4r5zi/src/bin/mldsa.rs b/libcrux-nucleo-l4r5zi/src/bin/mldsa.rs index d8ac40e..675a929 100644 --- a/libcrux-nucleo-l4r5zi/src/bin/mldsa.rs +++ b/libcrux-nucleo-l4r5zi/src/bin/mldsa.rs @@ -2,7 +2,9 @@ #![no_std] use cortex_m::peripheral::Peripherals; -use libcrux_nucleo_l4r5zi as _; // global logger + panicking-behavior + memory layout +use libcrux_nucleo_l4r5zi as board; // global logger + panicking-behavior + memory layout + +extern crate alloc; use core::ptr::addr_of_mut; use embedded_alloc::LlffHeap as Heap; @@ -23,8 +25,15 @@ fn main() -> ! { peripherals.DCB.enable_trace(); peripherals.DWT.enable_cycle_counter(); } + // set up the test config + let test_config = TestConfig { + core_freq: board::COREFREQ, + only_names: alloc::vec![], + early_abort: false, + benchmark_runs: 5, + }; + + libcrux_testbench::mldsa::run_benchmarks(test_config); - libcrux_testbench::mldsa::run_benchmarks(); - - libcrux_nucleo_l4r5zi::exit() + board::exit() } diff --git a/libcrux-nucleo-l4r5zi/src/bin/mlkem.rs b/libcrux-nucleo-l4r5zi/src/bin/mlkem.rs index a560623..530b72a 100644 --- a/libcrux-nucleo-l4r5zi/src/bin/mlkem.rs +++ b/libcrux-nucleo-l4r5zi/src/bin/mlkem.rs @@ -2,7 +2,9 @@ #![no_std] use cortex_m::peripheral::Peripherals; -use libcrux_nucleo_l4r5zi as _; // global logger + panicking-behavior + memory layout +use libcrux_nucleo_l4r5zi as board; // global logger + panicking-behavior + memory layout + +extern crate alloc; use core::ptr::addr_of_mut; use embedded_alloc::LlffHeap as Heap; @@ -24,7 +26,15 @@ fn main() -> ! { peripherals.DWT.enable_cycle_counter(); } - libcrux_testbench::mlkem::run_benchmarks(); + // set up the test config + let test_config = TestConfig { + core_freq: board::COREFREQ, + only_names: alloc::vec![], + early_abort: false, + benchmark_runs: 5, + }; + + libcrux_testbench::mlkem::run_benchmarks(test_config); - libcrux_nucleo_l4r5zi::exit() + board::exit() } diff --git a/libcrux-nucleo-l4r5zi/src/lib.rs b/libcrux-nucleo-l4r5zi/src/lib.rs index 4d97cee..17629b4 100644 --- a/libcrux-nucleo-l4r5zi/src/lib.rs +++ b/libcrux-nucleo-l4r5zi/src/lib.rs @@ -9,6 +9,8 @@ use embassy_stm32 as _; // memory layout use panic_probe as _; +pub const COREFREQ: u32 = 4_000_000; + // same panicking *behavior* as `panic-probe` but doesn't print a panic message // this prevents the panic message being printed *twice* when `defmt::panic` is invoked #[defmt::panic_handler] From 5f468f6e65cd82f52fe8446ceb2b1bab9b0f3632 Mon Sep 17 00:00:00 2001 From: Jonas Schneider-Bensch Date: Thu, 31 Oct 2024 14:40:40 +0100 Subject: [PATCH 11/12] Forgot imports --- libcrux-nucleo-l4r5zi/src/bin/mldsa.rs | 1 + libcrux-nucleo-l4r5zi/src/bin/mlkem.rs | 1 + 2 files changed, 2 insertions(+) diff --git a/libcrux-nucleo-l4r5zi/src/bin/mldsa.rs b/libcrux-nucleo-l4r5zi/src/bin/mldsa.rs index 675a929..bb32d73 100644 --- a/libcrux-nucleo-l4r5zi/src/bin/mldsa.rs +++ b/libcrux-nucleo-l4r5zi/src/bin/mldsa.rs @@ -14,6 +14,7 @@ static HEAP: Heap = Heap::empty(); #[cortex_m_rt::entry] fn main() -> ! { + use libcrux_iot_testutil::*; // Initialize the allocator BEFORE you use it { use core::mem::MaybeUninit; diff --git a/libcrux-nucleo-l4r5zi/src/bin/mlkem.rs b/libcrux-nucleo-l4r5zi/src/bin/mlkem.rs index 530b72a..c9ffe93 100644 --- a/libcrux-nucleo-l4r5zi/src/bin/mlkem.rs +++ b/libcrux-nucleo-l4r5zi/src/bin/mlkem.rs @@ -14,6 +14,7 @@ static HEAP: Heap = Heap::empty(); #[cortex_m_rt::entry] fn main() -> ! { + use libcrux_iot_testutil::*; // Initialize the allocator BEFORE you use it { use core::mem::MaybeUninit; From 362ec647743acbca28f3563d16234794feb05c71 Mon Sep 17 00:00:00 2001 From: Jonas Schneider-Bensch Date: Thu, 31 Oct 2024 14:45:57 +0100 Subject: [PATCH 12/12] Remove warning for bare-metal targets --- libcrux/libcrux-platform/src/lib.rs | 1 + 1 file changed, 1 insertion(+) diff --git a/libcrux/libcrux-platform/src/lib.rs b/libcrux/libcrux-platform/src/lib.rs index 380389c..40adfc6 100644 --- a/libcrux/libcrux-platform/src/lib.rs +++ b/libcrux/libcrux-platform/src/lib.rs @@ -54,6 +54,7 @@ mod platform { #[cfg(not(hax))] mod platform { + #[cfg(not(target_os = "none"))] use super::*; // TODO: Check for z14 or z15