@@ -202,13 +202,20 @@ pub struct Cheatcodes {
202
202
pub breakpoints : Breakpoints ,
203
203
204
204
/// Track if cool cheatcode was called on each address
205
- pub addresses : HashMap < Address , AddressState > ,
206
- /// Track if an addresses storage slot is cool or not
207
- pub address_storage : HashMap < Address , HashMap < U256 , StorageSlotState > > ,
205
+ pub addresses : HashMap < Address , CoolState > ,
208
206
/// How much gas to charge in the next step (op code) based on cool cheatcode calculations
209
207
pub additional_gas_next_op : u64 ,
210
208
}
211
209
210
+ /// Tracking if an address and it's slots are cool
211
+ #[ derive( Debug , Clone ) ]
212
+ pub struct CoolState {
213
+ /// Track if address is cool
214
+ pub address : AddressState ,
215
+ /// Track if each storage slot is cool or not
216
+ pub slots : HashMap < U256 , StorageSlotState > ,
217
+ }
218
+
212
219
/// Whether an Address is accessed or not
213
220
#[ derive( Clone , PartialEq , Debug ) ]
214
221
pub enum AddressState {
@@ -235,7 +242,7 @@ fn add_gas_from_cool_cheatcode<DB: DatabaseExt>(
235
242
interpreter : & mut Interpreter ,
236
243
data : & mut EVMData < ' _ , DB > ,
237
244
) -> InstructionResult {
238
- if state. addresses . len ( ) == 0 {
245
+ if state. addresses . is_empty ( ) {
239
246
return InstructionResult :: Continue
240
247
}
241
248
@@ -268,9 +275,11 @@ fn add_gas_from_cool_cheatcode<DB: DatabaseExt>(
268
275
// COLD_ACCOUNT_ACCESS_COST is 2600
269
276
// check this is done once per address, unless cheatcode is called again
270
277
// ignore if same as contract address
271
- if state. addresses . get ( & addr) == Some ( & AddressState :: Cool ) {
272
- state. additional_gas_next_op = 2500 ;
273
- state. addresses . insert ( addr, AddressState :: Warm ) ;
278
+ if let Some ( cool_state) = state. addresses . get_mut ( & addr) {
279
+ if cool_state. address == AddressState :: Cool {
280
+ state. additional_gas_next_op = 2500 ;
281
+ cool_state. address = AddressState :: Warm ;
282
+ }
274
283
}
275
284
}
276
285
}
@@ -283,32 +292,34 @@ fn add_gas_from_cool_cheatcode<DB: DatabaseExt>(
283
292
// COLD_ACCOUNT_ACCESS_COST is 2600
284
293
// check this is done once per address, unless cheatcode is called again
285
294
// ignore if same as contract address
286
- if state. addresses . get ( & addr) == Some ( & AddressState :: Cool ) {
287
- state. additional_gas_next_op = 2500 ;
288
- state. addresses . insert ( addr, AddressState :: Warm ) ;
295
+ if let Some ( cool_state) = state. addresses . get_mut ( & addr) {
296
+ if cool_state. address == AddressState :: Cool {
297
+ state. additional_gas_next_op = 2500 ;
298
+ cool_state. address = AddressState :: Warm ;
299
+ }
289
300
}
290
301
}
291
302
}
292
303
_ => { }
293
304
}
294
305
// check target's slots
295
- if let Some ( contract_storage ) = state. address_storage . get_mut ( & contract_address) {
306
+ if let Some ( cool_state ) = state. addresses . get_mut ( & contract_address) {
296
307
match interpreter. current_opcode ( ) {
297
308
opcode:: SLOAD => {
298
309
let key = try_or_continue ! ( interpreter. stack( ) . peek( 0 ) ) ;
299
310
300
311
let account = data. journaled_state . state ( ) . get ( & contract_address) . unwrap ( ) ;
301
312
if account. storage . get ( & key) . is_some ( ) {
302
- match contract_storage . get ( & key) {
313
+ match cool_state . slots . get ( & key) {
303
314
None | Some ( StorageSlotState :: Cool ) => {
304
315
// COLD_SLOAD_COST is 2100
305
316
state. additional_gas_next_op = 2000 ;
306
- contract_storage . insert ( key, StorageSlotState :: WarmWithSLOAD ) ;
317
+ cool_state . slots . insert ( key, StorageSlotState :: WarmWithSLOAD ) ;
307
318
}
308
319
Some ( _) => { }
309
320
}
310
321
} else {
311
- contract_storage . insert ( key, StorageSlotState :: WarmWithSLOAD ) ;
322
+ cool_state . slots . insert ( key, StorageSlotState :: WarmWithSLOAD ) ;
312
323
}
313
324
}
314
325
opcode:: SSTORE => {
@@ -318,7 +329,7 @@ fn add_gas_from_cool_cheatcode<DB: DatabaseExt>(
318
329
let account = data. journaled_state . state ( ) . get ( & contract_address) . unwrap ( ) ;
319
330
if account. storage . get ( & key) . is_some ( ) {
320
331
// only add gas the first time the storage is touched again
321
- match contract_storage . get ( & key) {
332
+ match cool_state . slots . get ( & key) {
322
333
Some ( StorageSlotState :: WarmWithSLOAD ) => {
323
334
// cool keeps the slot value changes
324
335
// as if the previous_or_original_value = present_value`
@@ -337,7 +348,7 @@ fn add_gas_from_cool_cheatcode<DB: DatabaseExt>(
337
348
}
338
349
339
350
// set slot is_warm to true
340
- contract_storage . insert ( key, StorageSlotState :: WarmWithSSTORE ) ;
351
+ cool_state . slots . insert ( key, StorageSlotState :: WarmWithSSTORE ) ;
341
352
}
342
353
None | Some ( StorageSlotState :: Cool ) => {
343
354
// Means SSTORE was called without SLOAD before
@@ -359,12 +370,12 @@ fn add_gas_from_cool_cheatcode<DB: DatabaseExt>(
359
370
state. additional_gas_next_op += 2900 - 100
360
371
}
361
372
}
362
- contract_storage . insert ( key, StorageSlotState :: WarmWithSSTORE ) ;
373
+ cool_state . slots . insert ( key, StorageSlotState :: WarmWithSSTORE ) ;
363
374
}
364
375
Some ( StorageSlotState :: WarmWithSSTORE ) => { }
365
376
}
366
377
} else {
367
- contract_storage . insert ( key, StorageSlotState :: WarmWithSSTORE ) ;
378
+ cool_state . slots . insert ( key, StorageSlotState :: WarmWithSSTORE ) ;
368
379
}
369
380
}
370
381
_ => { }
0 commit comments