1use crate::constants::{GAS_PER_BLOB, MAX_RLP_BLOCK_SIZE, POST_OSAKA_GAS_LIMIT_CAP};
7use crate::errors::InvalidBlockError;
8use crate::types::requests::{EncodedRequests, Requests, compute_requests_hash};
9use crate::types::{
10 Block, BlockHeader, ChainConfig, EIP4844Transaction, Receipt, Transaction,
11 compute_receipts_root_and_logs_bloom, validate_block_header, validate_cancun_header_fields,
12 validate_prague_header_fields, validate_pre_cancun_header_fields,
13};
14use ethrex_crypto::Crypto;
15use ethrex_rlp::encode::RLPEncode;
16
17pub fn validate_block_pre_execution(
28 block: &Block,
29 parent_header: &BlockHeader,
30 chain_config: &ChainConfig,
31 elasticity_multiplier: u64,
32) -> Result<(), InvalidBlockError> {
33 validate_block_header(&block.header, parent_header, elasticity_multiplier)?;
35
36 if chain_config.is_osaka_activated(block.header.timestamp) {
37 let block_rlp_size = block.length();
38 if block_rlp_size > MAX_RLP_BLOCK_SIZE as usize {
39 return Err(InvalidBlockError::MaximumRlpSizeExceeded(
40 MAX_RLP_BLOCK_SIZE,
41 block_rlp_size as u64,
42 ));
43 }
44 }
45 if chain_config.is_prague_activated(block.header.timestamp) {
46 validate_prague_header_fields(&block.header, parent_header, chain_config)?;
47 verify_blob_gas_usage(block, chain_config)?;
48 if chain_config.is_osaka_activated(block.header.timestamp)
49 && !chain_config.is_amsterdam_activated(block.header.timestamp)
50 {
51 verify_transaction_max_gas_limit(block)?;
52 }
53 } else if chain_config.is_cancun_activated(block.header.timestamp) {
54 validate_cancun_header_fields(&block.header, parent_header, chain_config)?;
55 verify_blob_gas_usage(block, chain_config)?;
56 } else {
57 validate_pre_cancun_header_fields(&block.header)?;
58 }
59
60 for tx in &block.body.transactions {
73 if matches!(tx, Transaction::PrivilegedL2Transaction(_)) {
74 continue;
75 }
76 if let Some(tx_chain_id) = tx.chain_id()
77 && tx_chain_id != chain_config.chain_id
78 {
79 return Err(InvalidBlockError::InvalidTransactionChainId {
80 have: tx_chain_id,
81 want: chain_config.chain_id,
82 });
83 }
84 }
85
86 Ok(())
87}
88
89pub fn validate_gas_used(
93 block_gas_used: u64,
94 block_header: &BlockHeader,
95) -> Result<(), InvalidBlockError> {
96 if block_gas_used != block_header.gas_used {
97 return Err(InvalidBlockError::GasUsedMismatch(
98 block_gas_used,
99 block_header.gas_used,
100 ));
101 }
102 Ok(())
103}
104
105pub fn validate_receipts_root_and_logs_bloom(
114 block_header: &BlockHeader,
115 receipts: &[Receipt],
116 crypto: &dyn Crypto,
117) -> Result<(), InvalidBlockError> {
118 let (receipts_root, logs_bloom) = compute_receipts_root_and_logs_bloom(receipts, crypto);
119
120 if receipts_root != block_header.receipts_root {
121 return Err(InvalidBlockError::ReceiptsRootMismatch);
122 }
123 if logs_bloom != block_header.logs_bloom {
124 return Err(InvalidBlockError::LogsBloomMismatch);
125 }
126 Ok(())
127}
128
129pub fn validate_requests_hash(
131 header: &BlockHeader,
132 chain_config: &ChainConfig,
133 requests: &[Requests],
134) -> Result<(), InvalidBlockError> {
135 if !chain_config.is_prague_activated(header.timestamp) {
136 return Ok(());
137 }
138
139 let encoded_requests: Vec<EncodedRequests> = requests.iter().map(|r| r.encode()).collect();
140 let computed_requests_hash = compute_requests_hash(&encoded_requests);
141 let valid = header
142 .requests_hash
143 .map(|requests_hash| requests_hash == computed_requests_hash)
144 .unwrap_or(false);
145
146 if !valid {
147 return Err(InvalidBlockError::RequestsHashMismatch);
148 }
149
150 Ok(())
151}
152
153fn validate_bal_indices(
155 indices: impl Iterator<Item = u32>,
156 max_valid_index: u32,
157) -> Result<(), InvalidBlockError> {
158 for index in indices {
159 if index > max_valid_index {
160 return Err(InvalidBlockError::BlockAccessListIndexOutOfBounds {
161 index,
162 max: max_valid_index,
163 });
164 }
165 }
166 Ok(())
167}
168
169pub fn validate_header_bal_indices(
174 bal: &crate::types::block_access_list::BlockAccessList,
175 transaction_count: usize,
176) -> Result<(), InvalidBlockError> {
177 let max_valid_index = u32::try_from(transaction_count + 1).unwrap_or(u32::MAX);
178
179 for account in bal.accounts() {
180 validate_bal_indices(
181 account
182 .storage_changes
183 .iter()
184 .flat_map(|slot| slot.slot_changes.iter().map(|c| c.block_access_index)),
185 max_valid_index,
186 )?;
187 validate_bal_indices(
188 account.balance_changes.iter().map(|c| c.block_access_index),
189 max_valid_index,
190 )?;
191 validate_bal_indices(
192 account.nonce_changes.iter().map(|c| c.block_access_index),
193 max_valid_index,
194 )?;
195 validate_bal_indices(
196 account.code_changes.iter().map(|c| c.block_access_index),
197 max_valid_index,
198 )?;
199 }
200 Ok(())
201}
202
203pub fn validate_block_access_list_hash(
207 header: &BlockHeader,
208 chain_config: &ChainConfig,
209 computed_bal: &crate::types::block_access_list::BlockAccessList,
210 transaction_count: usize,
211) -> Result<(), InvalidBlockError> {
212 use crate::constants::BAL_ITEM_COST;
213
214 if !chain_config.is_amsterdam_activated(header.timestamp) {
216 return Ok(());
217 }
218
219 let max_valid_index = u32::try_from(transaction_count + 1).unwrap_or(u32::MAX);
222
223 let mut bal_items: u64 = 0;
225 for account in computed_bal.accounts() {
226 bal_items += 1; bal_items += account.storage_reads.len() as u64;
228 bal_items += account.storage_changes.len() as u64;
229
230 validate_bal_indices(
232 account
233 .storage_changes
234 .iter()
235 .flat_map(|slot| slot.slot_changes.iter().map(|c| c.block_access_index)),
236 max_valid_index,
237 )?;
238
239 validate_bal_indices(
241 account.balance_changes.iter().map(|c| c.block_access_index),
242 max_valid_index,
243 )?;
244
245 validate_bal_indices(
247 account.nonce_changes.iter().map(|c| c.block_access_index),
248 max_valid_index,
249 )?;
250
251 validate_bal_indices(
253 account.code_changes.iter().map(|c| c.block_access_index),
254 max_valid_index,
255 )?;
256 }
257
258 let max_items = header.gas_limit / BAL_ITEM_COST;
260 if bal_items > max_items {
261 return Err(InvalidBlockError::BlockAccessListSizeExceeded {
262 items: bal_items,
263 max_items,
264 });
265 }
266
267 let computed_hash = computed_bal.compute_hash();
268 let valid = header
269 .block_access_list_hash
270 .map(|expected_hash| expected_hash == computed_hash)
271 .unwrap_or(false);
272
273 if !valid {
274 return Err(InvalidBlockError::BlockAccessListHashMismatch);
275 }
276
277 Ok(())
278}
279
280pub fn validate_block_access_list_size(
286 header: &BlockHeader,
287 chain_config: &ChainConfig,
288 computed_bal: &crate::types::block_access_list::BlockAccessList,
289) -> Result<(), InvalidBlockError> {
290 use crate::constants::BAL_ITEM_COST;
291
292 if !chain_config.is_amsterdam_activated(header.timestamp) {
293 return Ok(());
294 }
295
296 let bal_items = computed_bal.item_count();
297 let max_items = header.gas_limit / BAL_ITEM_COST;
298
299 if bal_items > max_items {
300 return Err(InvalidBlockError::BlockAccessListSizeExceeded {
301 items: bal_items,
302 max_items,
303 });
304 }
305
306 Ok(())
307}
308
309fn verify_blob_gas_usage(block: &Block, config: &ChainConfig) -> Result<(), InvalidBlockError> {
312 let mut blob_gas_used = 0_u32;
313 let mut blobs_in_block = 0_u32;
314 let max_blob_number_per_block = config
315 .get_fork_blob_schedule(block.header.timestamp)
316 .map(|schedule| schedule.max)
317 .ok_or(InvalidBlockError::InvalidBlockFork)?;
318 let max_blob_gas_per_block = max_blob_number_per_block * GAS_PER_BLOB;
319
320 for transaction in block.body.transactions.iter() {
321 if let crate::types::Transaction::EIP4844Transaction(tx) = transaction {
322 blob_gas_used += get_total_blob_gas(tx);
323 blobs_in_block += tx.blob_versioned_hashes.len() as u32;
324 }
325 }
326 if blob_gas_used > max_blob_gas_per_block {
327 return Err(InvalidBlockError::ExceededMaxBlobGasPerBlock);
328 }
329 if blobs_in_block > max_blob_number_per_block {
330 return Err(InvalidBlockError::ExceededMaxBlobNumberPerBlock);
331 }
332 if block
333 .header
334 .blob_gas_used
335 .is_some_and(|header_blob_gas_used| header_blob_gas_used != blob_gas_used as u64)
336 {
337 return Err(InvalidBlockError::BlobGasUsedMismatch);
338 }
339 Ok(())
340}
341
342fn verify_transaction_max_gas_limit(block: &Block) -> Result<(), InvalidBlockError> {
346 for transaction in block.body.transactions.iter() {
347 if transaction.gas_limit() > POST_OSAKA_GAS_LIMIT_CAP {
348 return Err(InvalidBlockError::InvalidTransaction(format!(
349 "Transaction gas limit exceeds maximum. Transaction hash: {}, transaction gas limit: {}",
350 transaction.hash(),
351 transaction.gas_limit()
352 )));
353 }
354 }
355 Ok(())
356}
357
358pub fn get_total_blob_gas(tx: &EIP4844Transaction) -> u32 {
360 GAS_PER_BLOB * tx.blob_versioned_hashes.len() as u32
361}