use alloc::{vec, vec::Vec};
use codec::Encode;
use frame_benchmarking::v2::*;
use frame_support::{dispatch::DispatchClass, storage, traits::Get};
use frame_system::{Call, Pallet as System, RawOrigin};
use sp_core::storage::well_known_keys;
use sp_runtime::traits::Hash;
pub struct Pallet<T: Config>(System<T>);
pub trait Config: frame_system::Config {
fn prepare_set_code_data() -> Vec<u8> {
include_bytes!("../res/kitchensink_runtime.compact.compressed.wasm").to_vec()
}
fn setup_set_code_requirements(_code: &Vec<u8>) -> Result<(), BenchmarkError> {
Ok(())
}
fn verify_set_code() {
System::<Self>::assert_last_event(frame_system::Event::<Self>::CodeUpdated.into());
}
}
#[benchmarks]
mod benchmarks {
use super::*;
#[benchmark]
fn remark(
b: Linear<0, { *T::BlockLength::get().max.get(DispatchClass::Normal) as u32 }>,
) -> Result<(), BenchmarkError> {
let remark_message = vec![1; b as usize];
let caller = whitelisted_caller();
#[extrinsic_call]
remark(RawOrigin::Signed(caller), remark_message);
Ok(())
}
#[benchmark]
fn remark_with_event(
b: Linear<0, { *T::BlockLength::get().max.get(DispatchClass::Normal) as u32 }>,
) -> Result<(), BenchmarkError> {
let remark_message = vec![1; b as usize];
let caller: T::AccountId = whitelisted_caller();
let hash = T::Hashing::hash(&remark_message[..]);
#[extrinsic_call]
remark_with_event(RawOrigin::Signed(caller.clone()), remark_message);
System::<T>::assert_last_event(
frame_system::Event::<T>::Remarked { sender: caller, hash }.into(),
);
Ok(())
}
#[benchmark]
fn set_heap_pages() -> Result<(), BenchmarkError> {
#[extrinsic_call]
set_heap_pages(RawOrigin::Root, Default::default());
Ok(())
}
#[benchmark]
fn set_code() -> Result<(), BenchmarkError> {
let runtime_blob = T::prepare_set_code_data();
T::setup_set_code_requirements(&runtime_blob)?;
#[extrinsic_call]
set_code(RawOrigin::Root, runtime_blob);
T::verify_set_code();
Ok(())
}
#[benchmark(extra)]
fn set_code_without_checks() -> Result<(), BenchmarkError> {
let code = vec![1; 4_000_000 as usize];
T::setup_set_code_requirements(&code)?;
#[block]
{
System::<T>::set_code_without_checks(RawOrigin::Root.into(), code)?;
}
let current_code =
storage::unhashed::get_raw(well_known_keys::CODE).ok_or("Code not stored.")?;
assert_eq!(current_code.len(), 4_000_000 as usize);
Ok(())
}
#[benchmark(skip_meta)]
fn set_storage(i: Linear<0, { 1_000 }>) -> Result<(), BenchmarkError> {
let mut items = Vec::new();
for j in 0..i {
let hash = (i, j).using_encoded(T::Hashing::hash).as_ref().to_vec();
items.push((hash.clone(), hash.clone()));
}
let items_to_verify = items.clone();
#[extrinsic_call]
set_storage(RawOrigin::Root, items);
for (item, _) in items_to_verify {
let value = storage::unhashed::get_raw(&item).ok_or("No value stored")?;
assert_eq!(value, *item);
}
Ok(())
}
#[benchmark(skip_meta)]
fn kill_storage(i: Linear<0, { 1_000 }>) -> Result<(), BenchmarkError> {
let mut items = Vec::with_capacity(i as usize);
for j in 0..i {
let hash = (i, j).using_encoded(T::Hashing::hash).as_ref().to_vec();
storage::unhashed::put_raw(&hash, &hash);
items.push(hash);
}
for item in &items {
let value = storage::unhashed::get_raw(item).ok_or("No value stored")?;
assert_eq!(value, *item);
}
let items_to_verify = items.clone();
#[extrinsic_call]
kill_storage(RawOrigin::Root, items);
for item in items_to_verify {
assert!(storage::unhashed::get_raw(&item).is_none());
}
Ok(())
}
#[benchmark(skip_meta)]
fn kill_prefix(p: Linear<0, { 1_000 }>) -> Result<(), BenchmarkError> {
let prefix = p.using_encoded(T::Hashing::hash).as_ref().to_vec();
let mut items = Vec::with_capacity(p as usize);
for i in 0..p {
let hash = (p, i).using_encoded(T::Hashing::hash).as_ref().to_vec();
let key = [&prefix[..], &hash[..]].concat();
storage::unhashed::put_raw(&key, &key);
items.push(key);
}
for item in &items {
let value = storage::unhashed::get_raw(item).ok_or("No value stored")?;
assert_eq!(value, *item);
}
#[extrinsic_call]
kill_prefix(RawOrigin::Root, prefix, p);
for item in items {
assert!(storage::unhashed::get_raw(&item).is_none());
}
Ok(())
}
#[benchmark]
fn authorize_upgrade() -> Result<(), BenchmarkError> {
let runtime_blob = T::prepare_set_code_data();
T::setup_set_code_requirements(&runtime_blob)?;
let hash = T::Hashing::hash(&runtime_blob);
#[extrinsic_call]
authorize_upgrade(RawOrigin::Root, hash);
assert!(System::<T>::authorized_upgrade().is_some());
Ok(())
}
#[benchmark]
fn apply_authorized_upgrade() -> Result<(), BenchmarkError> {
let runtime_blob = T::prepare_set_code_data();
T::setup_set_code_requirements(&runtime_blob)?;
let hash = T::Hashing::hash(&runtime_blob);
System::<T>::authorize_upgrade(RawOrigin::Root.into(), hash)?;
#[extrinsic_call]
apply_authorized_upgrade(RawOrigin::Root, runtime_blob);
assert!(System::<T>::authorized_upgrade().is_none());
Ok(())
}
impl_benchmark_test_suite!(Pallet, crate::mock::new_test_ext(), crate::mock::Test);
}