1use alloc::{
2 string::{String, ToString},
3 vec::Vec,
4};
5use elf::abi::{PF_NONE, PF_R, PF_W, PF_X};
6
7pub const MAXIMUM_MEMORY_SIZE: u64 = (1u64 << 48) - 1;
9
10pub const MAX_JIT_LOG_ADDR: usize = 40;
12
13pub const BYTE_SIZE: usize = 8;
15
16pub const LIMB_SIZE: usize = 16;
18
19pub const WORD_SIZE: usize = 4;
21
22pub const WORD_BYTE_SIZE: usize = 2 * WORD_SIZE;
24
25pub const INSTRUCTION_WORD_SIZE: usize = 4;
27
28pub const LONG_WORD_BYTE_SIZE: usize = 2 * WORD_BYTE_SIZE;
30
31pub const LOG_PAGE_SIZE: usize = 12;
33
34pub const PAGE_SIZE: usize = 1 << LOG_PAGE_SIZE;
36
37pub const PROT_NONE: u8 = PF_NONE as u8;
39pub const PROT_READ: u8 = PF_R as u8;
40pub const PROT_WRITE: u8 = PF_W as u8;
41pub const PROT_EXEC: u8 = PF_X as u8;
42pub const DEFAULT_PAGE_PROT: u8 = PROT_READ | PROT_WRITE;
43
44pub const PERMITTED_PROTS: [u8; 4] =
50 [PROT_NONE, PROT_READ | PROT_WRITE, PROT_READ | PROT_EXEC, PROT_READ];
51
52pub const PROT_FAILURE_EXEC: u64 = 1;
54pub const PROT_FAILURE_READ: u64 = 5;
55pub const PROT_FAILURE_WRITE: u64 = 7;
56
57pub const PF_UNTRUSTED: u32 = 0x0010_0000;
59
60pub const NOTE_NAME: [u8; 9] = *b"SUCCINCT\0";
62pub const NOTE_UNTRUSTED_PROGRAM_ENABLED: u32 = 1;
64pub const NOTE_DESC_HEADER: [u8; 4] = [b'1', 0, 0, 0];
66pub const NOTE_DESC_SIZE: usize = NOTE_DESC_HEADER.len() + 8 + 8;
69pub const NOTE_NAME_PADDING_SIZE: usize = (4 - NOTE_NAME.len() % 4) % 4;
71pub const NOTE_DESC_PADDING_SIZE: usize = (4 - NOTE_DESC_SIZE % 4) % 4;
73
74pub const PROFILER_STACK_CUSTOM_SECTION_NAME: &str = "__sp1_profiler_stack";
76
77pub const DUMP_ELF_EXTRA_STACK: u64 = 0x100_0000;
80pub const STACK_TOP: u64 = 0x7800_0000;
81
82pub const MAXIMUM_DUMPED_PERMISSIONS: usize = 128;
85
86pub const PERMISSION_ARRAY_LENGTH: usize = MAXIMUM_DUMPED_PERMISSIONS * 3 + 1;
88pub const PERMISSION_BUFFER_SIZE: usize = (PERMISSION_ARRAY_LENGTH * 8).div_ceil(16) * 16;
91
92pub const MAXIMUM_ELF_SEGMENTS: usize = 256;
94
95pub mod fd {
96 pub const LOWEST_ALLOWED_FD: u32 = 10;
102
103 macro_rules! create_fd {
105 ($(
106 #[$attr:meta]
107 pub const $name:ident: u32 = $value:expr;
108 )*) => {
109 $(
110 #[$attr]
111 pub const $name: u32 = $value + $crate::consts::fd::LOWEST_ALLOWED_FD;
112 )*
113 }
114 }
115
116 create_fd! {
117 pub const FD_PUBLIC_VALUES: u32 = 3;
119
120 pub const FD_HINT: u32 = 4;
122
123 pub const FD_ECRECOVER_HOOK: u32 = 5;
125
126 pub const FD_EDDECOMPRESS: u32 = 6;
128
129 pub const FD_RSA_MUL_MOD: u32 = 7;
131
132 pub const FD_BLS12_381_SQRT: u32 = 8;
134
135 pub const FD_BLS12_381_INVERSE: u32 = 9;
137
138 pub const FD_FP_SQRT: u32 = 10;
140
141 pub const FD_FP_INV: u32 = 11;
143 }
144}
145
146pub fn words_to_bytes_le_vec<'a>(words: impl IntoIterator<Item = &'a u64>) -> Vec<u8> {
148 words.into_iter().flat_map(|word| word.to_le_bytes().into_iter()).collect::<Vec<_>>()
149}
150
151pub fn words_to_bytes_le<'a, const B: usize>(words: impl IntoIterator<Item = &'a u64>) -> [u8; B] {
153 let mut iter = words.into_iter().flat_map(|word| word.to_le_bytes().into_iter());
154 core::array::from_fn(|_| iter.next().unwrap())
155}
156
157pub fn bytes_to_words_le<const W: usize>(bytes: &[u8]) -> [u64; W] {
159 debug_assert_eq!(bytes.len(), W * 8);
160 let mut iter = bytes.chunks_exact(8).map(|chunk| u64::from_le_bytes(chunk.try_into().unwrap()));
161 core::array::from_fn(|_| iter.next().unwrap())
162}
163
164pub fn bytes_to_words_le_vec(bytes: &[u8]) -> Vec<u64> {
166 bytes
167 .chunks_exact(8)
168 .map(|chunk| u64::from_le_bytes(chunk.try_into().unwrap()))
169 .collect::<Vec<_>>()
170}
171
172pub fn num_to_comma_separated<T: ToString>(value: T) -> String {
174 value
175 .to_string()
176 .chars()
177 .rev()
178 .collect::<Vec<_>>()
179 .chunks(3)
180 .map(|chunk| chunk.iter().collect::<String>())
181 .collect::<Vec<_>>()
182 .join(",")
183 .chars()
184 .rev()
185 .collect()
186}
187
188pub fn u32_to_u64(limbs: &[u32]) -> Vec<u64> {
190 debug_assert!(limbs.len().is_multiple_of(2), "need an even number of u32s");
191 limbs.chunks_exact(2).map(|pair| (pair[0] as u64) | ((pair[1] as u64) << 32)).collect()
192}
193
194pub fn u64_to_u32<'a>(limbs: impl IntoIterator<Item = &'a u64>) -> Vec<u32> {
196 limbs
197 .into_iter()
198 .flat_map(|x| {
199 let lo = *x as u32;
200 let hi = (*x >> 32) as u32;
201 [lo, hi]
202 })
203 .collect()
204}
205
206pub fn u32_to_u16_limbs(value: u32) -> [u16; 2] {
208 [(value & 0xFFFF) as u16, (value >> 16) as u16]
209}
210
211pub fn u64_to_u16_limbs(value: u64) -> [u16; 4] {
213 [(value & 0xFFFF) as u16, (value >> 16) as u16, (value >> 32) as u16, (value >> 48) as u16]
214}
215
216pub fn split_page_idx(page_idx: u64) -> [u16; 3] {
219 [(page_idx & 0xF) as u16, ((page_idx >> 4) & 0xFFFF) as u16, ((page_idx >> 20) & 0xFFFF) as u16]
220}