1use elf::abi::{PF_NONE, PF_R, PF_W, PF_X};
2
3pub const MAXIMUM_MEMORY_SIZE: u64 = (1u64 << 48) - 1;
5
6pub const BYTE_SIZE: usize = 8;
8
9pub const LIMB_SIZE: usize = 16;
11
12pub const WORD_SIZE: usize = 4;
14
15pub const WORD_BYTE_SIZE: usize = 2 * WORD_SIZE;
17
18pub const INSTRUCTION_WORD_SIZE: usize = 4;
20
21pub const LONG_WORD_BYTE_SIZE: usize = 2 * WORD_BYTE_SIZE;
23
24pub const LOG_PAGE_SIZE: usize = 12;
26
27pub const PAGE_SIZE: usize = 1 << LOG_PAGE_SIZE;
29
30pub const PROT_NONE: u8 = PF_NONE as u8;
32pub const PROT_READ: u8 = PF_R as u8;
33pub const PROT_WRITE: u8 = PF_W as u8;
34pub const PROT_EXEC: u8 = PF_X as u8;
35pub const DEFAULT_PAGE_PROT: u8 = PROT_READ | PROT_WRITE;
36
37pub const NOTE_UNTRUSTED_PROGRAM_ENABLED: u32 = 1;
39
40pub const STACK_TOP: u64 = 0x78000000;
42
43pub mod fd {
44 pub const LOWEST_ALLOWED_FD: u32 = 10;
50
51 macro_rules! create_fd {
53 ($(
54 #[$attr:meta]
55 pub const $name:ident: u32 = $value:expr;
56 )*) => {
57 $(
58 #[$attr]
59 pub const $name: u32 = $value + $crate::consts::fd::LOWEST_ALLOWED_FD;
60 )*
61 }
62 }
63
64 create_fd! {
65 pub const FD_PUBLIC_VALUES: u32 = 3;
67
68 pub const FD_HINT: u32 = 4;
70
71 pub const FD_ECRECOVER_HOOK: u32 = 5;
73
74 pub const FD_EDDECOMPRESS: u32 = 6;
76
77 pub const FD_RSA_MUL_MOD: u32 = 7;
79
80 pub const FD_BLS12_381_SQRT: u32 = 8;
82
83 pub const FD_BLS12_381_INVERSE: u32 = 9;
85
86 pub const FD_FP_SQRT: u32 = 10;
88
89 pub const FD_FP_INV: u32 = 11;
91 }
92}
93
94pub fn words_to_bytes_le_vec<'a>(words: impl IntoIterator<Item = &'a u64>) -> Vec<u8> {
96 words.into_iter().flat_map(|word| word.to_le_bytes().into_iter()).collect::<Vec<_>>()
97}
98
99pub fn words_to_bytes_le<'a, const B: usize>(words: impl IntoIterator<Item = &'a u64>) -> [u8; B] {
101 let mut iter = words.into_iter().flat_map(|word| word.to_le_bytes().into_iter());
102 core::array::from_fn(|_| iter.next().unwrap())
103}
104
105pub fn bytes_to_words_le<const W: usize>(bytes: &[u8]) -> [u64; W] {
107 debug_assert_eq!(bytes.len(), W * 8);
108 let mut iter = bytes.chunks_exact(8).map(|chunk| u64::from_le_bytes(chunk.try_into().unwrap()));
109 core::array::from_fn(|_| iter.next().unwrap())
110}
111
112pub fn bytes_to_words_le_vec(bytes: &[u8]) -> Vec<u64> {
114 bytes
115 .chunks_exact(8)
116 .map(|chunk| u64::from_le_bytes(chunk.try_into().unwrap()))
117 .collect::<Vec<_>>()
118}
119
120pub fn num_to_comma_separated<T: ToString>(value: T) -> String {
122 value
123 .to_string()
124 .chars()
125 .rev()
126 .collect::<Vec<_>>()
127 .chunks(3)
128 .map(|chunk| chunk.iter().collect::<String>())
129 .collect::<Vec<_>>()
130 .join(",")
131 .chars()
132 .rev()
133 .collect()
134}
135
136pub fn u32_to_u64(limbs: &[u32]) -> Vec<u64> {
138 debug_assert!(limbs.len().is_multiple_of(2), "need an even number of u32s");
139 limbs.chunks_exact(2).map(|pair| (pair[0] as u64) | ((pair[1] as u64) << 32)).collect()
140}
141
142pub fn u64_to_u32<'a>(limbs: impl IntoIterator<Item = &'a u64>) -> Vec<u32> {
144 limbs
145 .into_iter()
146 .flat_map(|x| {
147 let lo = *x as u32;
148 let hi = (*x >> 32) as u32;
149 [lo, hi]
150 })
151 .collect()
152}
153
154pub fn u32_to_u16_limbs(value: u32) -> [u16; 2] {
156 [(value & 0xFFFF) as u16, (value >> 16) as u16]
157}
158
159pub fn u64_to_u16_limbs(value: u64) -> [u16; 4] {
161 [(value & 0xFFFF) as u16, (value >> 16) as u16, (value >> 32) as u16, (value >> 48) as u16]
162}
163
164pub fn split_page_idx(page_idx: u64) -> [u16; 3] {
167 [(page_idx & 0xF) as u16, ((page_idx >> 4) & 0xFFFF) as u16, ((page_idx >> 20) & 0xFFFF) as u16]
168}