Skip to content

Commit ecbeb88

Browse files
authored
Modify to_bits implementations (#61)
1 parent ea3ec65 commit ecbeb88

File tree

4 files changed

+57
-95
lines changed

4 files changed

+57
-95
lines changed

crates/starknet-types-core/Cargo.toml

-2
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,6 @@ description = "Core types representation for Starknet"
1111
readme = "README.md"
1212

1313
[dependencies]
14-
bitvec = { version = "1.0.1", default-features = false }
1514
lambdaworks-math = { version = "0.6.0", default-features = false }
1615

1716
num-traits = { version = "0.2.18", default-features = false }
@@ -30,7 +29,6 @@ lazy_static = { version = "1.4.0", default-features = false, optional = true }
3029
[features]
3130
default = ["std", "serde", "curve", "num-traits"]
3231
std = [
33-
"bitvec/std",
3432
"lambdaworks-math/std",
3533
"num-traits/std",
3634
"num-bigint/std",

crates/starknet-types-core/src/felt/mod.rs

+55-91
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,6 @@ mod felt_arbitrary;
33

44
use core::ops::{Add, Mul, Neg};
55

6-
use bitvec::array::BitArray;
76
use num_bigint::{BigInt, BigUint, Sign};
87
use num_integer::Integer;
98
use num_traits::{One, Zero};
@@ -24,12 +23,6 @@ lazy_static! {
2423
.unwrap();
2524
}
2625

27-
#[cfg(target_pointer_width = "64")]
28-
pub type BitArrayStore = [u64; 4];
29-
30-
#[cfg(not(target_pointer_width = "64"))]
31-
pub type BitArrayStore = [u32; 8];
32-
3326
#[cfg(any(test, feature = "alloc"))]
3427
pub extern crate alloc;
3528

@@ -255,34 +248,6 @@ impl Felt {
255248
self.0.to_bytes_le()
256249
}
257250

258-
/// Converts to big-endian bit representation.
259-
/// This is as performant as [to_bits_le](Felt::to_bits_le)
260-
#[cfg(target_pointer_width = "64")]
261-
pub fn to_bits_be(&self) -> BitArray<BitArrayStore> {
262-
let mut limbs = self.0.representative().limbs;
263-
limbs.reverse();
264-
265-
BitArray::new(limbs)
266-
}
267-
268-
/// Converts to big-endian bit representation.
269-
/// This is as performant as [to_bits_le](Felt::to_bits_le)
270-
#[cfg(all(feature = "alloc", not(target_pointer_width = "64")))]
271-
pub fn to_bits_be(&self) -> BitArray<BitArrayStore> {
272-
let mut limbs = self.0.representative().limbs;
273-
limbs.reverse();
274-
275-
// Split limbs to adjust to BitArrayStore = [u32; 8]
276-
let limbs: [u32; 8] = limbs
277-
.into_iter()
278-
.flat_map(|n| [(n >> 32) as u32, n as u32])
279-
.collect::<alloc::vec::Vec<u32>>()
280-
.try_into()
281-
.unwrap();
282-
283-
BitArray::new(limbs)
284-
}
285-
286251
/// Helper to produce a hexadecimal formatted string.
287252
/// Equivalent to calling `format!("{self:#x}")`.
288253
#[cfg(feature = "alloc")]
@@ -302,29 +267,15 @@ impl Felt {
302267
}
303268

304269
/// Converts to little-endian bit representation.
305-
/// This is as performant as [to_bits_be](Felt::to_bits_be)
306-
#[cfg(target_pointer_width = "64")]
307-
pub fn to_bits_le(&self) -> BitArray<BitArrayStore> {
308-
let limbs = self.0.representative().limbs;
309-
310-
BitArray::new(limbs)
270+
pub fn to_bits_le(&self) -> [bool; 256] {
271+
self.0.to_bits_le()
311272
}
312273

313-
/// Converts to little-endian bit representation.
314-
/// This is as performant as [to_bits_be](Felt::to_bits_be)
315-
#[cfg(all(feature = "alloc", not(target_pointer_width = "64")))]
316-
pub fn to_bits_le(&self) -> BitArray<BitArrayStore> {
317-
let limbs = self.0.representative().limbs;
318-
319-
// Split limbs to adjust to BitArrayStore = [u32; 8]
320-
let limbs: [u32; 8] = limbs
321-
.into_iter()
322-
.flat_map(|n| [n as u32, (n >> 32) as u32])
323-
.collect::<alloc::vec::Vec<u32>>()
324-
.try_into()
325-
.unwrap();
326-
327-
BitArray::new(limbs)
274+
/// Converts to big-endian bit representation.
275+
pub fn to_bits_be(&self) -> [bool; 256] {
276+
let mut bits = self.0.to_bits_le();
277+
bits.reverse();
278+
bits
328279
}
329280

330281
/// Finite field division.
@@ -1105,6 +1056,32 @@ mod test {
11051056
#[cfg(feature = "serde")]
11061057
use serde_test::{assert_de_tokens, assert_ser_tokens, Configure, Token};
11071058

1059+
// Helper function to generate a vector of bits for testing purposes
1060+
fn generate_be_bits(x: &Felt) -> Vec<bool> {
1061+
// Initialize an empty vector to store the expected bits
1062+
let mut bits = Vec::new();
1063+
1064+
// Iterate over each limb in the representative of x
1065+
for limb in x.0.representative().limbs {
1066+
// Convert the limb to a sequence of 8 bytes (u8) in big-endian
1067+
let bytes = limb.to_be_bytes();
1068+
1069+
// Iterate over each byte
1070+
for byte in &bytes {
1071+
// Iterate over each bit of the byte
1072+
for i in 0..8 {
1073+
// Extract the i-th bit of the byte
1074+
let bit = (*byte >> (7 - i)) & 1;
1075+
1076+
// Push the bit into the expected_bits vector
1077+
bits.push(bit == 1);
1078+
}
1079+
}
1080+
}
1081+
1082+
bits
1083+
}
1084+
11081085
proptest! {
11091086
#[test]
11101087
fn new_in_range(ref x in any::<[u8; 32]>()) {
@@ -1129,46 +1106,33 @@ mod test {
11291106
}
11301107

11311108
#[test]
1132-
fn to_bits_be(ref x in any::<Felt>()) {
1133-
let bits: Vec<bool> = x.to_bits_be().into_iter().rev().collect();
1134-
let mut res = [0;32];
1135-
let mut acc: u8 = 0;
1136-
for (i, bits64) in bits.chunks(8).enumerate() {
1137-
for bit in bits64.iter() {
1138-
acc <<= 1;
1139-
acc += *bit as u8;
1140-
}
1141-
res[i] = acc;
1142-
acc = 0;
1143-
}
1144-
let y = &Felt::from_bytes_be(&res);
1145-
prop_assert_eq!(x, y);
1109+
fn to_bits_le(ref x in any::<Felt>()) {
1110+
// Generate the little-endian representation of Felt type x
1111+
let bits: Vec<bool> = x.to_bits_le().into_iter().collect();
1112+
1113+
// Get the expected bits in big-endian order
1114+
let mut expected_bits = generate_be_bits(x);
1115+
1116+
// Reverse the expected_bits to convert from big-endian to little-endian
1117+
expected_bits.reverse();
1118+
1119+
// Assert that the generated bits match the expected bits
1120+
prop_assert_eq!(bits, expected_bits);
11461121
}
11471122

11481123
#[test]
1149-
fn to_bits_le(ref x in any::<Felt>()) {
1150-
let bits: Vec<bool> = x.to_bits_le().into_iter().collect();
1151-
let mut res = [0;4];
1152-
let mut acc: u64 = 0;
1153-
for (i, bits64) in bits.chunks(64).enumerate() {
1154-
for bit in bits64.iter().rev() {
1155-
acc <<= 1;
1156-
acc += *bit as u64;
1157-
}
1158-
res[i] = acc;
1159-
acc = 0;
1160-
}
1161-
let mut bytes = [0u8; 32];
1162-
for i in (0..4).rev() {
1163-
let limb_bytes = res[i].to_le_bytes();
1164-
for j in 0..8 {
1165-
bytes[(3 - i) * 8 + j] = limb_bytes[j]
1166-
}
1167-
}
1168-
let y = &Felt::from_bytes_le(&bytes);
1169-
prop_assert_eq!(x, y);
1124+
fn to_bits_be(ref x in any::<Felt>()) {
1125+
// Generate the big-endian representation of Felt type x
1126+
let bits: Vec<bool> = x.to_bits_be().into_iter().collect();
1127+
1128+
// Get the expected bits in big-endian order
1129+
let expected_bits = generate_be_bits(x);
1130+
1131+
// Assert that the generated bits match the expected bits
1132+
prop_assert_eq!(bits, expected_bits);
11701133
}
11711134

1135+
11721136
#[test]
11731137
#[cfg(feature = "alloc")]
11741138
fn to_hex_string_is_same_as_format(ref x in any::<Felt>()) {

ensure_no_std/rust-toolchain.toml

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
[toolchain]
2-
channel = "1.74"
2+
channel = "1.77.2"
33
components = ["rustfmt", "clippy"]
44
targets = ["wasm32-unknown-unknown"]
55
profile = "minimal"

rust-toolchain.toml

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
11
[toolchain]
2-
channel = "1.77"
2+
channel = "1.77.2"
33
components = ["rustfmt", "clippy"]
44
profile = "minimal"

0 commit comments

Comments
 (0)