From ec623f00087a5d9f5f1cccfb46433059fc3161c8 Mon Sep 17 00:00:00 2001 From: "remy.baranx@gmail.com" Date: Wed, 7 Aug 2024 10:44:58 +0200 Subject: [PATCH 1/2] bump to Cairo 2.7.1 (edition 2024_07) + snfoundry 0.27.0 --- onchain/Scarb.lock | 4 +- onchain/Scarb.toml | 6 +- onchain/src/bip340.cairo | 5 +- onchain/src/erc20.cairo | 8 +- onchain/src/lib.cairo | 1 - onchain/src/sha256.cairo | 262 ------------------------------- onchain/src/social/account.cairo | 50 +++--- onchain/src/social/bech32.cairo | 3 +- onchain/src/social/deposit.cairo | 32 ++-- onchain/src/social/profile.cairo | 3 +- onchain/src/social/request.cairo | 2 +- onchain/src/utils.cairo | 63 +------- 12 files changed, 65 insertions(+), 374 deletions(-) delete mode 100644 onchain/src/sha256.cairo diff --git a/onchain/Scarb.lock b/onchain/Scarb.lock index f88bfeea..472d1eaa 100644 --- a/onchain/Scarb.lock +++ b/onchain/Scarb.lock @@ -10,5 +10,5 @@ dependencies = [ [[package]] name = "snforge_std" -version = "0.25.0" -source = "git+https://github.com/foundry-rs/starknet-foundry?tag=v0.25.0#5b366e24821e530fea97f11b211d220e8493fbea" +version = "0.27.0" +source = "git+https://github.com/foundry-rs/starknet-foundry?tag=v0.27.0#2d99b7c00678ef0363881ee0273550c44a9263de" diff --git a/onchain/Scarb.toml b/onchain/Scarb.toml index 1e3b4ae7..5658de74 100644 --- a/onchain/Scarb.toml +++ b/onchain/Scarb.toml @@ -1,15 +1,15 @@ [package] name = "joyboy" version = "0.1.0" -edition = "2023_11" +edition = "2024_07" # See more keys and their definitions at https://docs.swmansion.com/scarb/docs/reference/manifest.html [dependencies] -starknet = "2.6.3" +starknet = "2.7.0" [dev-dependencies] -snforge_std = { git = "https://github.com/foundry-rs/starknet-foundry", tag = "v0.25.0" } +snforge_std = { git = "https://github.com/foundry-rs/starknet-foundry", tag = "v0.27.0" } #[lib] diff --git a/onchain/src/bip340.cairo b/onchain/src/bip340.cairo index a73b1603..c1eb504f 100644 --- a/onchain/src/bip340.cairo +++ b/onchain/src/bip340.cairo @@ -3,13 +3,12 @@ use core::byte_array::ByteArrayTrait; use core::option::OptionTrait; use core::result::ResultTrait; -// TODO: uncomment once Cairo 2.7 is available -// use core::sha256::compute_sha256_byte_array; +use core::sha256::compute_sha256_byte_array; use core::starknet::SyscallResultTrait; use core::to_byte_array::{AppendFormattedToByteArray, FormatAsByteArray}; use core::traits::Into; -use joyboy::utils::{shl, shr, compute_sha256_byte_array}; +use joyboy::utils::{shl, shr}; use starknet::{secp256k1::{Secp256k1Point}, secp256_trait::{Secp256Trait, Secp256PointTrait}}; const TWO_POW_32: u128 = 0x100000000; diff --git a/onchain/src/erc20.cairo b/onchain/src/erc20.cairo index afe8f2f5..b5b2f3d3 100644 --- a/onchain/src/erc20.cairo +++ b/onchain/src/erc20.cairo @@ -25,6 +25,10 @@ pub mod ERC20 { use starknet::ContractAddress; use starknet::contract_address_const; use starknet::get_caller_address; + use starknet::storage::{ + StorageMapReadAccess, StorageMapWriteAccess, StoragePointerReadAccess, + StoragePointerWriteAccess + }; #[storage] struct Storage { @@ -32,8 +36,8 @@ pub mod ERC20 { symbol: felt252, decimals: u8, total_supply: u256, - balances: LegacyMap::, - allowances: LegacyMap::<(ContractAddress, ContractAddress), u256>, + balances: starknet::storage::Map::, + allowances: starknet::storage::Map::<(ContractAddress, ContractAddress), u256>, } #[event] diff --git a/onchain/src/lib.cairo b/onchain/src/lib.cairo index 8b8dacae..1089e80a 100644 --- a/onchain/src/lib.cairo +++ b/onchain/src/lib.cairo @@ -1,5 +1,4 @@ pub mod bip340; pub mod erc20; -pub mod sha256; pub mod social; pub mod utils; diff --git a/onchain/src/sha256.cairo b/onchain/src/sha256.cairo deleted file mode 100644 index f0156369..00000000 --- a/onchain/src/sha256.cairo +++ /dev/null @@ -1,262 +0,0 @@ -// from alexandria - -use core::integer::{u32_wrapping_add, BoundedInt}; - -fn ch(x: u32, y: u32, z: u32) -> u32 { - (x & y) ^ ((x ^ BoundedInt::::max().into()) & z) -} - -fn maj(x: u32, y: u32, z: u32) -> u32 { - (x & y) ^ (x & z) ^ (y & z) -} - -fn bsig0(x: u32) -> u32 { - let x: u128 = x.into(); - let x1 = (x / 0x4) | (x * 0x40000000); - let x2 = (x / 0x2000) | (x * 0x80000); - let x3 = (x / 0x400000) | (x * 0x400); - let result = (x1 ^ x2 ^ x3) & BoundedInt::::max().into(); - result.try_into().unwrap() -} - -fn bsig1(x: u32) -> u32 { - let x: u128 = x.into(); - let x1 = (x / 0x40) | (x * 0x4000000); - let x2 = (x / 0x800) | (x * 0x200000); - let x3 = (x / 0x2000000) | (x * 0x80); - let result = (x1 ^ x2 ^ x3) & BoundedInt::::max().into(); - result.try_into().unwrap() -} - -fn ssig0(x: u32) -> u32 { - let x: u128 = x.into(); - let x1 = (x / 0x80) | (x * 0x2000000); - let x2 = (x / 0x40000) | (x * 0x4000); - let x3 = (x / 0x8); - let result = (x1 ^ x2 ^ x3) & BoundedInt::::max().into(); - result.try_into().unwrap() -} - -fn ssig1(x: u32) -> u32 { - let x: u128 = x.into(); - let x1 = (x / 0x20000) | (x * 0x8000); - let x2 = (x / 0x80000) | (x * 0x2000); - let x3 = (x / 0x400); - let result = (x1 ^ x2 ^ x3) & BoundedInt::::max().into(); - result.try_into().unwrap() -} - -pub fn sha256(mut data: Array) -> Array { - let data_len: u64 = (data.len() * 8).into(); - - // add one - data.append(0x80); - // add padding - while ((64 * ((data.len() - 1) / 64 + 1)) - 8 != data.len()) { - data.append(0); - }; - - // add length to the end - let mut res = (data_len & 0xff00000000000000) / 0x100000000000000; - data.append(res.try_into().unwrap()); - res = (data_len.into() & 0xff000000000000) / 0x1000000000000; - data.append(res.try_into().unwrap()); - res = (data_len.into() & 0xff0000000000) / 0x10000000000; - data.append(res.try_into().unwrap()); - res = (data_len.into() & 0xff00000000) / 0x100000000; - data.append(res.try_into().unwrap()); - res = (data_len.into() & 0xff000000) / 0x1000000; - data.append(res.try_into().unwrap()); - res = (data_len.into() & 0xff0000) / 0x10000; - data.append(res.try_into().unwrap()); - res = (data_len.into() & 0xff00) / 0x100; - data.append(res.try_into().unwrap()); - res = data_len.into() & 0xff; - data.append(res.try_into().unwrap()); - - let data = from_u8Array_to_u32Array(data.span()); - let h = get_h(); - let k = get_k(); - let res = sha256_inner(data.span(), 0, k.span(), h.span()); - - from_u32Array_to_u8Array(res) -} - -fn from_u32Array_to_u8Array(mut data: Span) -> Array { - let mut result = array![]; - while let Option::Some(val) = data - .pop_front() { - let mut res = (*val & 0xff000000) / 0x1000000; - result.append(res.try_into().unwrap()); - res = (*val & 0xff0000) / 0x10000; - result.append(res.try_into().unwrap()); - res = (*val & 0xff00) / 0x100; - result.append(res.try_into().unwrap()); - res = *val & 0xff; - result.append(res.try_into().unwrap()); - }; - result -} - -fn sha256_inner(mut data: Span, i: usize, k: Span, mut h: Span) -> Span { - if 16 * i >= data.len() { - return h; - } - let w = create_message_schedule(data, i); - let h2 = compression(w, 0, k, h); - - let mut t = array![]; - t.append(u32_wrapping_add(*h[0], *h2[0])); - t.append(u32_wrapping_add(*h[1], *h2[1])); - t.append(u32_wrapping_add(*h[2], *h2[2])); - t.append(u32_wrapping_add(*h[3], *h2[3])); - t.append(u32_wrapping_add(*h[4], *h2[4])); - t.append(u32_wrapping_add(*h[5], *h2[5])); - t.append(u32_wrapping_add(*h[6], *h2[6])); - t.append(u32_wrapping_add(*h[7], *h2[7])); - h = t.span(); - sha256_inner(data, i + 1, k, h) -} - -fn compression(w: Span, i: usize, k: Span, mut h: Span) -> Span { - if i >= 64 { - return h; - } - let s1 = bsig1(*h[4]); - let ch = ch(*h[4], *h[5], *h[6]); - let temp1 = u32_wrapping_add( - u32_wrapping_add(u32_wrapping_add(u32_wrapping_add(*h[7], s1), ch), *k[i]), *w[i] - ); - let s0 = bsig0(*h[0]); - let maj = maj(*h[0], *h[1], *h[2]); - let temp2 = u32_wrapping_add(s0, maj); - let mut t = array![]; - t.append(u32_wrapping_add(temp1, temp2)); - t.append(*h[0]); - t.append(*h[1]); - t.append(*h[2]); - t.append(u32_wrapping_add(*h[3], temp1)); - t.append(*h[4]); - t.append(*h[5]); - t.append(*h[6]); - h = t.span(); - compression(w, i + 1, k, h) -} - -fn create_message_schedule(data: Span, i: usize) -> Span { - let mut j = 0; - let mut result = array![]; - while (j < 16) { - result.append(*data[i * 16 + j]); - j += 1; - }; - let mut i = 16; - while (i < 64) { - let s0 = ssig0(*result[i - 15]); - let s1 = ssig1(*result[i - 2]); - let res = u32_wrapping_add( - u32_wrapping_add(u32_wrapping_add(*result[i - 16], s0), *result[i - 7]), s1 - ); - result.append(res); - i += 1; - }; - result.span() -} - -fn from_u8Array_to_u32Array(mut data: Span) -> Array { - let mut result = array![]; - while let Option::Some(val1) = data - .pop_front() { - let val2 = data.pop_front().unwrap(); - let val3 = data.pop_front().unwrap(); - let val4 = data.pop_front().unwrap(); - let mut value = (*val1).into() * 0x1000000; - value = value + (*val2).into() * 0x10000; - value = value + (*val3).into() * 0x100; - value = value + (*val4).into(); - result.append(value); - }; - result -} - -fn get_h() -> Array { - array![ - 0x6a09e667, - 0xbb67ae85, - 0x3c6ef372, - 0xa54ff53a, - 0x510e527f, - 0x9b05688c, - 0x1f83d9ab, - 0x5be0cd19 - ] -} - -fn get_k() -> Array { - array![ - 0x428a2f98, - 0x71374491, - 0xb5c0fbcf, - 0xe9b5dba5, - 0x3956c25b, - 0x59f111f1, - 0x923f82a4, - 0xab1c5ed5, - 0xd807aa98, - 0x12835b01, - 0x243185be, - 0x550c7dc3, - 0x72be5d74, - 0x80deb1fe, - 0x9bdc06a7, - 0xc19bf174, - 0xe49b69c1, - 0xefbe4786, - 0x0fc19dc6, - 0x240ca1cc, - 0x2de92c6f, - 0x4a7484aa, - 0x5cb0a9dc, - 0x76f988da, - 0x983e5152, - 0xa831c66d, - 0xb00327c8, - 0xbf597fc7, - 0xc6e00bf3, - 0xd5a79147, - 0x06ca6351, - 0x14292967, - 0x27b70a85, - 0x2e1b2138, - 0x4d2c6dfc, - 0x53380d13, - 0x650a7354, - 0x766a0abb, - 0x81c2c92e, - 0x92722c85, - 0xa2bfe8a1, - 0xa81a664b, - 0xc24b8b70, - 0xc76c51a3, - 0xd192e819, - 0xd6990624, - 0xf40e3585, - 0x106aa070, - 0x19a4c116, - 0x1e376c08, - 0x2748774c, - 0x34b0bcb5, - 0x391c0cb3, - 0x4ed8aa4a, - 0x5b9cca4f, - 0x682e6ff3, - 0x748f82ee, - 0x78a5636f, - 0x84c87814, - 0x8cc70208, - 0x90befffa, - 0xa4506ceb, - 0xbef9a3f7, - 0xc67178f2 - ] -} diff --git a/onchain/src/social/account.cairo b/onchain/src/social/account.cairo index c7477b28..71011e85 100644 --- a/onchain/src/social/account.cairo +++ b/onchain/src/social/account.cairo @@ -34,6 +34,10 @@ pub mod SocialAccount { }; use starknet::account::Call; use starknet::{get_caller_address, get_contract_address, get_tx_info, ContractAddress}; + use starknet::storage::{ + StorageMapReadAccess, StorageMapWriteAccess, StoragePointerReadAccess, + StoragePointerWriteAccess + }; use super::ISRC6; use super::super::request::{ @@ -46,19 +50,19 @@ pub mod SocialAccount { struct Storage { #[key] public_key: u256, - transfers: LegacyMap, + transfers: starknet::storage::Map, } #[event] #[derive(Drop, starknet::Event)] - enum Event { + pub enum Event { AccountCreated: AccountCreated, } #[derive(Drop, starknet::Event)] - struct AccountCreated { + pub struct AccountCreated { #[key] - public_key: u256 + pub public_key: u256 } #[constructor] @@ -159,15 +163,15 @@ mod tests { use core::traits::Into; use joyboy::erc20::{ERC20, IERC20Dispatcher, IERC20DispatcherTrait}; use snforge_std::{ - declare, ContractClass, ContractClassTrait, spy_events, SpyOn, EventSpy, EventFetcher, - Event, EventAssertions, cheat_transaction_hash_global, cheat_signature_global, + declare, ContractClass, ContractClassTrait, spy_events, EventSpy, EventSpyTrait, + Event, EventSpyAssertionsTrait, start_cheat_transaction_hash_global, start_cheat_signature_global, stop_cheat_transaction_hash_global, stop_cheat_signature_global }; use starknet::{ ContractAddress, get_caller_address, get_contract_address, contract_address_const }; use super::super::profile::NostrProfile; - + use super::SocialAccount; use super::super::request::{SocialRequest, Signature, Encode}; use super::super::transfer::Transfer; use super::{ @@ -189,26 +193,20 @@ mod tests { let mut calldata = array![]; public_key.serialize(ref calldata); - let address = class.precalculate_address(@calldata); - - let mut spy = spy_events(SpyOn::One(address)); + let mut spy = spy_events(); let (contract_address, _) = class.deploy(@calldata).unwrap(); - spy.fetch_events(); - - assert(spy.events.len() == 1, 'there should be one event'); - - // TODO: deserialize event instead of manual decoding - let (_, event) = spy.events.at(0); - assert(event.keys.at(0) == @selector!("AccountCreated"), 'wrong event name'); - - let event_key = u256 { - low: (*event.keys.at(1)).try_into().unwrap(), - high: (*event.keys.at(2)).try_into().unwrap() - }; + assert(spy.get_events().events.len() == 1, 'there should be one event'); - assert(event_key == public_key, 'wrong public key'); + spy.assert_emitted(@array![ + ( + contract_address, + SocialAccount::Event::AccountCreated( + SocialAccount::AccountCreated { public_key } + ) + ) + ]); ISocialAccountDispatcher { contract_address } } @@ -426,13 +424,13 @@ mod tests { r.serialize(ref signature); s.serialize(ref signature); - cheat_transaction_hash_global(hash); - cheat_signature_global(signature.span()); + start_cheat_transaction_hash_global(hash); + start_cheat_signature_global(signature.span()); assert!(account.__validate__(Default::default()) == starknet::VALIDATED); let invalid_hash = 0x5a8885a308d313198a2e03707344a4093822299f31d0082efa98ec4e6c89; - cheat_transaction_hash_global(invalid_hash); + start_cheat_transaction_hash_global(invalid_hash); assert!(account.__validate__(Default::default()) != starknet::VALIDATED); diff --git a/onchain/src/social/bech32.cairo b/onchain/src/social/bech32.cairo index 453f0b0f..26a1d837 100644 --- a/onchain/src/social/bech32.cairo +++ b/onchain/src/social/bech32.cairo @@ -1,8 +1,7 @@ use core::array::ArrayTrait; use core::byte_array::ByteArrayTrait; use core::cmp::min; -// TODO: uncomment once Cairo 2.7 is available -// use core::array::ToSpanTrait; +use core::array::ToSpanTrait; use core::option::OptionTrait; use core::to_byte_array::FormatAsByteArray; //! bech32 encoding implementation diff --git a/onchain/src/social/deposit.cairo b/onchain/src/social/deposit.cairo index e5807960..55aa455f 100644 --- a/onchain/src/social/deposit.cairo +++ b/onchain/src/social/deposit.cairo @@ -68,6 +68,10 @@ pub mod DepositEscrow { use starknet::{ get_block_timestamp, get_caller_address, get_contract_address, get_tx_info, ContractAddress }; + use starknet::storage::{ + StorageMapReadAccess, StorageMapWriteAccess, StoragePointerReadAccess, + StoragePointerWriteAccess + }; use super::super::request::{ SocialRequest, SocialRequestImpl, SocialRequestTrait, Encode, Signature }; @@ -90,8 +94,8 @@ pub mod DepositEscrow { #[storage] struct Storage { next_deposit_id: DepositId, - deposits: LegacyMap, - nostr_to_sn: LegacyMap + deposits: starknet::storage::Map, + nostr_to_sn: starknet::storage::Map } #[derive(Drop, starknet::Event)] @@ -292,8 +296,8 @@ mod tests { use joyboy::erc20::{ERC20, IERC20Dispatcher, IERC20DispatcherTrait}; use snforge_std::{ - declare, ContractClass, ContractClassTrait, spy_events, SpyOn, EventSpy, EventFetcher, - Event, EventAssertions, start_cheat_caller_address, cheat_caller_address_global, + declare, ContractClass, ContractClassTrait, spy_events, EventSpy, + Event, start_cheat_caller_address, start_cheat_caller_address_global, stop_cheat_caller_address_global, start_cheat_block_timestamp, }; use starknet::{ @@ -405,7 +409,7 @@ mod tests { let recipient_address: ContractAddress = 678.try_into().unwrap(); let amount = 100_u256; - cheat_caller_address_global(sender_address); + start_cheat_caller_address_global(sender_address); erc20.approve(escrow.contract_address, amount); stop_cheat_caller_address_global(); @@ -468,7 +472,7 @@ mod tests { ..request }; - cheat_caller_address_global(sender_address); + start_cheat_caller_address_global(sender_address); erc20.approve(escrow.contract_address, amount); stop_cheat_caller_address_global(); @@ -522,7 +526,7 @@ mod tests { let amount = 100_u256; - cheat_caller_address_global(sender_address); + start_cheat_caller_address_global(sender_address); erc20.approve(escrow.contract_address, amount); stop_cheat_caller_address_global(); @@ -540,7 +544,7 @@ mod tests { let amount = 100_u256; - cheat_caller_address_global(sender_address); + start_cheat_caller_address_global(sender_address); erc20.approve(escrow.contract_address, amount); stop_cheat_caller_address_global(); @@ -568,7 +572,7 @@ mod tests { let recipient_address: ContractAddress = 789.try_into().unwrap(); let amount = 100_u256; - cheat_caller_address_global(sender_address); + start_cheat_caller_address_global(sender_address); erc20.approve(escrow.contract_address, amount); stop_cheat_caller_address_global(); @@ -593,7 +597,7 @@ mod tests { let amount = 100_u256; - cheat_caller_address_global(sender_address); + start_cheat_caller_address_global(sender_address); erc20.approve(escrow.contract_address, amount); stop_cheat_caller_address_global(); @@ -615,7 +619,7 @@ mod tests { let amount = 100_u256; - cheat_caller_address_global(sender_address); + start_cheat_caller_address_global(sender_address); erc20.approve(escrow.contract_address, amount); stop_cheat_caller_address_global(); @@ -634,7 +638,7 @@ mod tests { let amount = 100_u256; - cheat_caller_address_global(sender_address); + start_cheat_caller_address_global(sender_address); erc20.approve(escrow.contract_address, amount); stop_cheat_caller_address_global(); @@ -657,7 +661,7 @@ mod tests { let amount = 100_u256; - cheat_caller_address_global(sender_address); + start_cheat_caller_address_global(sender_address); erc20.approve(escrow.contract_address, amount); stop_cheat_caller_address_global(); @@ -676,7 +680,7 @@ mod tests { let recipient_address: ContractAddress = 345.try_into().unwrap(); let amount = 100_u256; - cheat_caller_address_global(sender_address); + start_cheat_caller_address_global(sender_address); erc20.approve(escrow.contract_address, amount); stop_cheat_caller_address_global(); diff --git a/onchain/src/social/profile.cairo b/onchain/src/social/profile.cairo index fdb610be..d80f792c 100644 --- a/onchain/src/social/profile.cairo +++ b/onchain/src/social/profile.cairo @@ -2,8 +2,7 @@ use core::array::SpanTrait; use core::byte_array::ByteArrayTrait; use core::option::OptionTrait; use core::traits::TryInto; -// TODO: uncomment once Cairo 2.7 is available -// use core::array::ToSpanTrait; +use core::array::ToSpanTrait; //! Representation of Nostr profiles diff --git a/onchain/src/social/request.cairo b/onchain/src/social/request.cairo index cb75ac2a..2ff710ac 100644 --- a/onchain/src/social/request.cairo +++ b/onchain/src/social/request.cairo @@ -1,8 +1,8 @@ use core::fmt::Display; +use core::sha256::compute_sha256_byte_array; use core::to_byte_array::FormatAsByteArray; use core::traits::Into; use joyboy::bip340; -use joyboy::utils::{compute_sha256_byte_array}; const TWO_POW_32: u128 = 0x100000000; const TWO_POW_64: u128 = 0x10000000000000000; diff --git a/onchain/src/utils.cairo b/onchain/src/utils.cairo index 1b3788cd..cb22e1e6 100644 --- a/onchain/src/utils.cairo +++ b/onchain/src/utils.cairo @@ -1,6 +1,7 @@ use core::ecdsa::check_ecdsa_signature; -use core::integer::{u32_wide_mul, u8_wide_mul, BoundedInt}; -use joyboy::sha256::sha256; +use core::traits::BitAnd; +use core::num::traits::Bounded; +use core::num::traits::WideMul as CoreWideMul; use starknet::SyscallResultTrait; use starknet::account::Call; @@ -79,13 +80,13 @@ trait WideMul { impl WideMuluU32 of WideMul { fn wide_mul(x: u32, y: u32) -> u64 { - u32_wide_mul(x, y) + CoreWideMul::wide_mul(x, y) } } impl WideMuluU8 of WideMul { fn wide_mul(x: u8, y: u8) -> u16 { - u8_wide_mul(x, y) + CoreWideMul::wide_mul(x, y) } } @@ -96,7 +97,7 @@ pub fn shl< N, +BitAnd, +Pow2, - +BoundedInt, + +Bounded, +WideMul, +Into, +TryInto, @@ -105,59 +106,9 @@ pub fn shl< >( x: V, n: N ) -> V { - (WideMul::wide_mul(x, Pow2::pow2(n)) & BoundedInt::::max().into()).try_into().unwrap() + (WideMul::wide_mul(x, Pow2::pow2(n)) & Bounded::::MAX.into()).try_into().unwrap() } -pub fn compute_sha256_byte_array(m: @ByteArray) -> [u32; 8] { - let mut ba = ArrayTrait::new(); - let len = m.len(); - let mut i = 0; - while i != len { - ba.append(m.at(i).unwrap()); - i += 1; - }; - - let sha = sha256(ba); - - let r = [ - shl((*sha.at(0)).into(), 24_u32) - + shl((*sha.at(1)).into(), 16_u32) - + shl((*sha.at(2)).into(), 8_u32) - + (*sha.at(3)).into(), - shl((*sha.at(4)).into(), 24_u32) - + shl((*sha.at(5)).into(), 16_u32) - + shl((*sha.at(6)).into(), 8_u32) - + (*sha.at(7)).into(), - shl((*sha.at(8)).into(), 24_u32) - + shl((*sha.at(9)).into(), 16_u32) - + shl((*sha.at(10)).into(), 8_u32) - + (*sha.at(11)).into(), - shl((*sha.at(12)).into(), 24_u32) - + shl((*sha.at(13)).into(), 16_u32) - + shl((*sha.at(14)).into(), 8_u32) - + (*sha.at(15)).into(), - shl((*sha.at(16)).into(), 24_u32) - + shl((*sha.at(17)).into(), 16_u32) - + shl((*sha.at(18)).into(), 8_u32) - + (*sha.at(19)).into(), - shl((*sha.at(20)).into(), 24_u32) - + shl((*sha.at(21)).into(), 16_u32) - + shl((*sha.at(22)).into(), 8_u32) - + (*sha.at(23)).into(), - shl((*sha.at(24)).into(), 24_u32) - + shl((*sha.at(25)).into(), 16_u32) - + shl((*sha.at(26)).into(), 8_u32) - + (*sha.at(27)).into(), - shl((*sha.at(28)).into(), 24_u32) - + shl((*sha.at(29)).into(), 16_u32) - + shl((*sha.at(30)).into(), 8_u32) - + (*sha.at(31)).into(), - ]; - - r -} - - pub const MIN_TRANSACTION_VERSION: u256 = 1; pub const QUERY_OFFSET: u256 = 0x100000000000000000000000000000000; // QUERY_OFFSET + TRANSACTION_VERSION From 3e56307e3c4339c6b7e1e595a49a0d8b1b56b13d Mon Sep 17 00:00:00 2001 From: "remy.baranx@gmail.com" Date: Sun, 18 Aug 2024 15:17:40 +0200 Subject: [PATCH 2/2] improve bech32 perf --- onchain/src/social/bech32.cairo | 162 ++++++++++++++------------------ 1 file changed, 69 insertions(+), 93 deletions(-) diff --git a/onchain/src/social/bech32.cairo b/onchain/src/social/bech32.cairo index 26a1d837..0c7796b6 100644 --- a/onchain/src/social/bech32.cairo +++ b/onchain/src/social/bech32.cairo @@ -1,13 +1,4 @@ -use core::array::ArrayTrait; -use core::byte_array::ByteArrayTrait; use core::cmp::min; -use core::array::ToSpanTrait; -use core::option::OptionTrait; -use core::to_byte_array::FormatAsByteArray; -//! bech32 encoding implementation - -use core::traits::{Into, TryInto}; - use joyboy::utils::{shl, shr}; //! bech32 encoding implementation @@ -16,70 +7,58 @@ use joyboy::utils::{shl, shr}; //! https://github.com/sipa/bech32/blob/master/ref/javascript/bech32.js#L86 //! https://github.com/paulmillr/scure-base/blob/main/index.ts#L479 -// const GENERATOR: [ -// felt252 -// ; 5] = [ -// 1, 2, 3, 4 -// ]; -fn polymod(values: Array) -> u32 { - let generator = array![ - 0x3b6a57b2_u32, 0x26508e6d_u32, 0x1ea119fa_u32, 0x3d4233dd_u32, 0x2a1462b3_u32 - ]; - let generator = generator.span(); +const alphabet: [u8; 32] = [ + 'q', 'p', 'z', 'r', 'y', '9', 'x', '8', + 'g', 'f', '2', 't', 'v', 'd', 'w', '0', + 's', '3', 'j', 'n', '5', '4', 'k', 'h', + 'c', 'e', '6', 'm', 'u', 'a', '7', 'l' +]; - let mut chk = 1_u32; +#[inline(always)] +fn polymod(ref chk: u32, value: u8) { + let top = chk; + chk = shl((chk & 0x1ffffff_u32), 5) ^ value.into(); - let len = values.len(); - let mut p: usize = 0; - while p != len { - let top = shr(chk, 25); - chk = shl((chk & 0x1ffffff_u32), 5) ^ (*values.at(p)).into(); - let mut i = 0_usize; - while i != 5 { - if shr(top, i) & 1_u32 != 0 { - chk = chk ^ *generator.at(i.into()); - } - i += 1; - }; - p += 1; - }; - - chk + if top & 33554432_u32 != 0 { // bit 25 + chk = chk ^ 0x3b6a57b2_u32; + } + if top & 67108864_u32 != 0 { // bit 26 + chk = chk ^ 0x26508e6d_u32; + } + if top & 134217728_u32 != 0 { // bit 27 + chk = chk ^ 0x1ea119fa_u32; + } + if top & 268435456_u32 != 0 { // bit 28 + chk = chk ^ 0x3d4233dd_u32; + } + if top & 536870912_u32 != 0 { // bit 29 + chk = chk ^ 0x2a1462b3_u32; + } } -fn hrp_expand(hrp: @Array) -> Array { +fn convert_bytearray_to_bytes(data: @ByteArray) -> Array { let mut r: Array = ArrayTrait::new(); - - let len = hrp.len(); + let len = data.len(); let mut i = 0; while i != len { - r.append(shr(*hrp.at(i), 5)); + r.append(data[i]); i += 1; }; - r.append(0); - - let len = hrp.len(); - let mut i = 0; - while i != len { - r.append(*hrp.at(i) & 31); - i += 1; - }; - r } -fn convert_bytes_to_5bit_chunks(bytes: @Array) -> Array { +fn convert_bytearray_to_5bit_chunks(data: @ByteArray) -> Array { let mut r = ArrayTrait::new(); - let len = bytes.len(); + let len = data.len(); let mut i = 0; let mut acc = 0_u8; let mut missing_bits = 5_u8; while i != len { - let mut byte: u8 = *bytes.at(i); + let mut byte: u8 = data[i]; let mut bits_left = 8_u8; loop { let chunk_size = min(missing_bits, bits_left); @@ -104,61 +83,58 @@ fn convert_bytes_to_5bit_chunks(bytes: @Array) -> Array { r } -impl ByteArrayTraitIntoArray of Into<@ByteArray, Array> { - fn into(self: @ByteArray) -> Array { - let mut r = ArrayTrait::new(); - let len = self.len(); - let mut i = 0; - while i != len { - r.append(self.at(i).unwrap()); - i += 1; - }; - r - } -} +fn checksum(hrp: Array, data: @Array) -> Array { + let mut chk = 1_u32; -fn checksum(hrp: @ByteArray, data: @Array) -> Array { - let mut values = ArrayTrait::new(); + for x in hrp.span() { + polymod(ref chk, shr(*x, 5)); + }; + polymod(ref chk, 0); + for x in hrp { + polymod(ref chk, x & 31); + }; - values.append_span(hrp_expand(@hrp.into()).span()); - values.append_span(data.span()); - let the_data: Array = array![0, 0, 0, 0, 0, 0]; - values.append_span(the_data.span()); + for x in data.span() { + polymod(ref chk, *x); + }; - let m = polymod(values) ^ 1; + polymod(ref chk, 0); + polymod(ref chk, 0); + polymod(ref chk, 0); + polymod(ref chk, 0); + polymod(ref chk, 0); + polymod(ref chk, 0); + + chk = chk ^ 1; let mut r = ArrayTrait::new(); - r.append((shr(m, 25) & 31).try_into().unwrap()); - r.append((shr(m, 20) & 31).try_into().unwrap()); - r.append((shr(m, 15) & 31).try_into().unwrap()); - r.append((shr(m, 10) & 31).try_into().unwrap()); - r.append((shr(m, 5) & 31).try_into().unwrap()); - r.append((m & 31).try_into().unwrap()); + r.append(shr(chk, 25) & 31); + r.append(shr(chk, 20) & 31); + r.append(shr(chk, 15) & 31); + r.append(shr(chk, 10) & 31); + r.append(shr(chk, 5) & 31); + r.append(chk & 31); r } pub fn encode(hrp: @ByteArray, data: @ByteArray, limit: usize) -> ByteArray { - // change into an array and a const - let alphabet: ByteArray = "qpzry9x8gf2tvdw0s3jn54khce6mua7l"; - - let data_5bits = convert_bytes_to_5bit_chunks(@data.into()); - - let cs = checksum(hrp, @data_5bits); + let alphabet = alphabet.span(); + let data_5bits = convert_bytearray_to_5bit_chunks(data); + let hrp_bytes = convert_bytearray_to_bytes(hrp); - let mut combined = ArrayTrait::new(); - combined.append_span(data_5bits.span()); - combined.append_span(cs.span()); + let cs = checksum(hrp_bytes, @data_5bits); - let mut encoded: ByteArray = Default::default(); - let mut i = 0; - let len = combined.len(); - while i != len { - encoded.append_byte(alphabet.at((*combined.at(i)).into()).unwrap()); - i += 1; + let mut encoded: ByteArray = hrp.clone(); + encoded.append_byte('1'); + for x in data_5bits { + encoded.append_byte(*alphabet[x.into()]); + }; + for x in cs { + encoded.append_byte(*alphabet[x]); }; - format!("{hrp}1{encoded}") + encoded } #[cfg(test)]