-
Notifications
You must be signed in to change notification settings - Fork 127
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: implement byte tracking in EVM memory (#242)
* Add byte-to-opcode mapping * Use bit-tracked memory stores during VM execution * Add getter for bit tracking * Add ByteTracker type * Add documentation comments * Add unit tests for ByteTracker type * Refactor range collision handling * Move core ByteTracker implementation to its own RangeMap module * Nightly format run * Address review comments * Update common/src/resources/range_map.rs Co-authored-by: Jonathan Becker <[email protected]> * Move range map implementation to appropriate module and lint * Patch range map unit tests --------- Co-authored-by: Jonathan Becker <[email protected]>
- Loading branch information
1 parent
71aaaeb
commit 950c763
Showing
4 changed files
with
274 additions
and
9 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,236 @@ | ||
use std::{collections::HashMap, ops::Range}; | ||
|
||
use crate::ether::evm::core::opcodes::WrappedOpcode; | ||
|
||
#[derive(Copy, Clone, Debug)] | ||
enum CollisionKind { | ||
Deletion, | ||
Splitting, | ||
Shortening, | ||
} | ||
|
||
#[derive(Clone, Debug, Eq, PartialEq)] | ||
pub struct RangeMap(pub HashMap<Range<usize>, WrappedOpcode>); | ||
|
||
impl RangeMap { | ||
pub fn new() -> Self { | ||
Self(HashMap::new()) | ||
} | ||
|
||
/// Given an offset into memory, returns the associated opcode if it exists | ||
pub fn get_by_offset(&self, offset: usize) -> Option<WrappedOpcode> { | ||
self.0.get(self.find_range(offset).expect("RangeMap::have_range is broken")).cloned() | ||
} | ||
|
||
/// Associates the provided opcode with the range of memory modified by writing a `size`-byte | ||
/// value to `offset`. | ||
/// | ||
/// This range is exactly `[offset, offset + size - 1]`. This function ensures that any existing | ||
/// ranges that our new range would collide with are dealt with accordingly, that is: | ||
/// | ||
/// - deleted, if our range completely overwrites it, | ||
/// - split, if our range overwrites a subset that partitions it, | ||
/// - shortened, if our range overwrites such that only one "end" of it is overwritten | ||
pub fn write(&mut self, offset: usize, size: usize, opcode: WrappedOpcode) { | ||
let range: Range<usize> = Range { start: offset, end: offset + size - 1 }; | ||
let incumbents: Vec<Range<usize>> = self.affected_ranges(range.clone()); | ||
|
||
if incumbents.is_empty() { | ||
self.0.insert(range, opcode); | ||
} else { | ||
incumbents.iter().for_each(|incumbent| { | ||
match Self::classify_collision(&range, incumbent) { | ||
CollisionKind::Deletion => { | ||
self.0.remove(incumbent); | ||
} | ||
CollisionKind::Splitting => { | ||
let left: Range<usize> = | ||
Range { start: incumbent.start, end: range.start - 1 }; | ||
let right: Range<usize> = | ||
Range { start: range.end + 1, end: incumbent.end }; | ||
let old_opcode: WrappedOpcode = self.0.get(incumbent).expect("").clone(); | ||
|
||
self.0.remove(incumbent); | ||
self.0.insert(left, old_opcode.clone()); | ||
self.0.insert(right, old_opcode.clone()); | ||
} | ||
CollisionKind::Shortening => { | ||
let needs_right_shortening = | ||
|incoming: &Range<usize>, incumbent: &Range<usize>| { | ||
incoming.start >= incumbent.start | ||
}; | ||
|
||
if needs_right_shortening(&range, incumbent) { | ||
let remainder: Range<usize> = | ||
Range { start: incumbent.start, end: range.start - 1 }; | ||
let old_opcode: WrappedOpcode = self.0.get(incumbent).cloned().unwrap(); | ||
self.0.remove(incumbent); | ||
self.0.insert(remainder, old_opcode); | ||
} else { | ||
let remainder: Range<usize> = | ||
Range { start: range.end + 1, end: incumbent.end }; | ||
let old_opcode: WrappedOpcode = self.0.get(incumbent).cloned().unwrap(); | ||
self.0.remove(incumbent); | ||
self.0.insert(remainder, old_opcode); | ||
} | ||
} | ||
} | ||
|
||
self.0.insert(range.clone(), opcode.clone()); | ||
}); | ||
} | ||
} | ||
|
||
fn classify_collision(incoming: &Range<usize>, incumbent: &Range<usize>) -> CollisionKind { | ||
let range_needs_deletion = |incoming: &Range<usize>, incumbent: &Range<usize>| { | ||
incoming.start <= incumbent.start && incoming.end >= incumbent.end | ||
}; | ||
let range_needs_splitting = |incoming: &Range<usize>, incumbent: &Range<usize>| { | ||
incoming.start > incumbent.start && incoming.end < incumbent.end | ||
}; | ||
|
||
if range_needs_deletion(incoming, incumbent) { | ||
CollisionKind::Deletion | ||
} else if range_needs_splitting(incoming, incumbent) { | ||
CollisionKind::Splitting | ||
} else { | ||
CollisionKind::Shortening | ||
} | ||
} | ||
|
||
fn find_range(&self, offset: usize) -> Option<&Range<usize>> { | ||
self.0.keys().find(|range| range.contains(&offset)) | ||
} | ||
|
||
fn affected_ranges(&self, range: Range<usize>) -> Vec<Range<usize>> { | ||
self.0 | ||
.keys() | ||
.filter(|incumbent| Self::range_collides(&range, *incumbent)) | ||
.cloned() | ||
.collect() | ||
} | ||
|
||
fn range_collides(incoming: &Range<usize>, incumbent: &Range<usize>) -> bool { | ||
(incoming.start <= incumbent.start && incoming.end >= incumbent.end) || | ||
(incoming.start <= incumbent.start && incoming.end >= incumbent.start) || | ||
(incoming.start <= incumbent.end && incoming.end >= incumbent.end) || | ||
(incoming.start > incumbent.start && incoming.end < incumbent.end) | ||
} | ||
} | ||
|
||
#[cfg(test)] | ||
mod tests { | ||
use std::{collections::HashMap, ops::Range}; | ||
|
||
use crate::{ether::evm::core::opcodes::WrappedOpcode, utils::range_map::RangeMap}; | ||
|
||
#[test] | ||
fn test_one_incumbent_and_needs_deletion() { | ||
/* the values of the mapping are irrelevant for the purposes of this test, so we | ||
* construct an arbitrary one and reuse it everywhere for simplicity */ | ||
let some_op: WrappedOpcode = WrappedOpcode::default(); | ||
let initial_pairs: Vec<((usize, usize), WrappedOpcode)> = | ||
vec![((8, 16), some_op.clone()), ((32, 64), some_op.clone())]; | ||
|
||
let mut actual_byte_tracker: RangeMap = RangeMap(HashMap::from_iter( | ||
initial_pairs.iter().cloned().map(|((a, b), v)| (Range { start: a, end: b }, v)), | ||
)); | ||
|
||
let offset: usize = 7; | ||
let size: usize = 11; | ||
actual_byte_tracker.write(offset, size, some_op.clone()); | ||
|
||
let expected_pairs: Vec<((usize, usize), WrappedOpcode)> = | ||
vec![((7, 17), some_op.clone()), ((32, 64), some_op.clone())]; | ||
let expected_byte_tracker: RangeMap = RangeMap(HashMap::from_iter( | ||
expected_pairs.iter().cloned().map(|((a, b), v)| (Range { start: a, end: b }, v)), | ||
)); | ||
|
||
assert_eq!(actual_byte_tracker, expected_byte_tracker); | ||
} | ||
|
||
#[test] | ||
fn test_one_incumbent_and_needs_splitting() { | ||
/* the values of the mapping are irrelevant for the purposes of this test, so we | ||
* construct an arbitrary one and reuse it everywhere for simplicity */ | ||
let some_op: WrappedOpcode = WrappedOpcode::default(); | ||
let initial_pairs: Vec<((usize, usize), WrappedOpcode)> = | ||
vec![((7, 18), some_op.clone()), ((32, 64), some_op.clone())]; | ||
|
||
let mut actual_byte_tracker: RangeMap = RangeMap(HashMap::from_iter( | ||
initial_pairs.iter().cloned().map(|((a, b), v)| (Range { start: a, end: b }, v)), | ||
)); | ||
|
||
let offset: usize = 8; | ||
let size: usize = 8; | ||
actual_byte_tracker.write(offset, size, some_op.clone()); | ||
|
||
let expected_pairs: Vec<((usize, usize), WrappedOpcode)> = vec![ | ||
((7, 7), some_op.clone()), | ||
((8, 15), some_op.clone()), | ||
((16, 18), some_op.clone()), | ||
((32, 64), some_op.clone()), | ||
]; | ||
let expected_byte_tracker: RangeMap = RangeMap(HashMap::from_iter( | ||
expected_pairs.iter().cloned().map(|((a, b), v)| (Range { start: a, end: b }, v)), | ||
)); | ||
|
||
assert_eq!(actual_byte_tracker, expected_byte_tracker); | ||
} | ||
|
||
#[test] | ||
fn test_one_incumbent_and_needs_right_shortening() { | ||
/* the values of the mapping are irrelevant for the purposes of this test, so we | ||
* construct an arbitrary one and reuse it everywhere for simplicity */ | ||
let some_op: WrappedOpcode = WrappedOpcode::default(); | ||
let initial_pairs: Vec<((usize, usize), WrappedOpcode)> = | ||
vec![((7, 18), some_op.clone()), ((32, 64), some_op.clone())]; | ||
|
||
let mut actual_byte_tracker: RangeMap = RangeMap(HashMap::from_iter( | ||
initial_pairs.iter().cloned().map(|((a, b), v)| (Range { start: a, end: b }, v)), | ||
)); | ||
|
||
let offset: usize = 10; | ||
let size: usize = 14; | ||
actual_byte_tracker.write(offset, size, some_op.clone()); | ||
|
||
let expected_pairs: Vec<((usize, usize), WrappedOpcode)> = vec![ | ||
((7, 9), some_op.clone()), | ||
((10, 23), some_op.clone()), | ||
((32, 64), some_op.clone()), | ||
]; | ||
let expected_byte_tracker: RangeMap = RangeMap(HashMap::from_iter( | ||
expected_pairs.iter().cloned().map(|((a, b), v)| (Range { start: a, end: b }, v)), | ||
)); | ||
|
||
assert_eq!(actual_byte_tracker, expected_byte_tracker); | ||
} | ||
|
||
#[test] | ||
fn test_one_incumbent_and_needs_left_shortening() { | ||
/* the values of the mapping are irrelevant for the purposes of this test, so we | ||
* construct an arbitrary one and reuse it everywhere for simplicity */ | ||
let some_op: WrappedOpcode = WrappedOpcode::default(); | ||
let initial_pairs: Vec<((usize, usize), WrappedOpcode)> = | ||
vec![((7, 18), some_op.clone()), ((32, 64), some_op.clone())]; | ||
|
||
let mut actual_byte_tracker: RangeMap = RangeMap(HashMap::from_iter( | ||
initial_pairs.iter().cloned().map(|((a, b), v)| (Range { start: a, end: b }, v)), | ||
)); | ||
|
||
let offset: usize = 2; | ||
let size: usize = 8; | ||
actual_byte_tracker.write(offset, size, some_op.clone()); | ||
|
||
let expected_pairs: Vec<((usize, usize), WrappedOpcode)> = vec![ | ||
((2, 9), some_op.clone()), | ||
((10, 18), some_op.clone()), | ||
((32, 64), some_op.clone()), | ||
]; | ||
let expected_byte_tracker: RangeMap = RangeMap(HashMap::from_iter( | ||
expected_pairs.iter().cloned().map(|((a, b), v)| (Range { start: a, end: b }, v)), | ||
)); | ||
|
||
assert_eq!(actual_byte_tracker, expected_byte_tracker); | ||
} | ||
} |