Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Extend Checkpoints with new sizes and lookup mechanisms #3589

Merged
merged 53 commits into from
Aug 30, 2022
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
53 commits
Select commit Hold shift + click to select a range
4176852
move mock
Amxx Jul 28, 2022
63edb74
Expand Checkpoint library
Amxx Jul 28, 2022
da0ea31
rename upperLookupExpEnd → upperLookupRecent
Amxx Jul 28, 2022
ed81388
sanity check when pushing checkpoints
Amxx Jul 28, 2022
b91832d
Skip redundant optimisation in VotesQuorumFraction
Amxx Jul 29, 2022
d2a829c
Merge branch 'master' into feature/checkpoint
Amxx Jul 29, 2022
2a6adb3
reset previous history lookup algorithm and reenable quorum checks
Amxx Jul 29, 2022
28f6e6f
Use unsafeAccess to save gas
Amxx Jul 29, 2022
b7e2f64
Merge branch 'master' into feature/checkpoint
Amxx Jul 29, 2022
616c51b
Regenerate contracts
Amxx Jul 29, 2022
3fb185b
Add changelog entry
Amxx Jul 29, 2022
ce0c7a6
use StorageSlot for unsafeAccess returns
Amxx Jul 29, 2022
33741e7
minor rewrite for readability
Amxx Jul 29, 2022
8425e4a
fix typo
Amxx Jul 29, 2022
74fef95
testing unsafeAccess
Amxx Jul 29, 2022
df3a9b1
fix lint
Amxx Jul 29, 2022
e182730
Update CHANGELOG.md
Amxx Jul 29, 2022
9b1f559
Merge branch 'master' into feature/checkpoint
Amxx Aug 14, 2022
768a681
Apply suggestions from code review
Amxx Aug 17, 2022
8d9b177
Update Arrays.sol
Amxx Aug 17, 2022
8dc9d4f
Run prettier as part of the procedural generation
Amxx Aug 17, 2022
4e6aa05
Reactor checkpoints to keep legacy types
Amxx Aug 17, 2022
fffde15
add a comment about files being procedurally generated
Amxx Aug 17, 2022
ac5466a
add extensions for the comments
Amxx Aug 17, 2022
a9a69cf
lint
Amxx Aug 17, 2022
4654237
shorter message
Amxx Aug 17, 2022
b486b6b
Merge branch 'master' into feature/checkpoint
Amxx Aug 25, 2022
2848c3a
regenerate & codespell
Amxx Aug 25, 2022
4c6c65e
update generation script path resolution
Amxx Aug 25, 2022
068513c
address comment from PR
Amxx Aug 26, 2022
bf12a2b
add more neatspec
Amxx Aug 26, 2022
b1d5a09
add more neatspec
Amxx Aug 26, 2022
543d516
Apply suggestions from code review
Amxx Aug 26, 2022
634e279
regenerate
Amxx Aug 26, 2022
17e1ec7
don't insert newline when no previous version is recorded
Amxx Aug 26, 2022
d1af38d
improve checkpoint opts generation
Amxx Aug 26, 2022
253e12a
codespell
Amxx Aug 26, 2022
1ccff75
test coverage
Amxx Aug 26, 2022
4d1d3da
grammar
frangio Aug 29, 2022
2d03e54
improve tests
Amxx Aug 30, 2022
69ac69e
Merge remote-tracking branch 'amxx/feature/checkpoint' into feature/c…
Amxx Aug 30, 2022
5661e71
Update test/utils/Checkpoints.test.js
Amxx Aug 30, 2022
6cd42ce
Apply suggestions from code review
Amxx Aug 30, 2022
9c84034
fix generation
Amxx Aug 30, 2022
b878b23
Update scripts/generate/templates/Checkpoints.js
Amxx Aug 30, 2022
c832e53
Update scripts/generate/templates/Checkpoints.js
Amxx Aug 30, 2022
a450545
Update scripts/generate/templates/Checkpoints.js
Amxx Aug 30, 2022
96122a3
Update scripts/generate/templates/Checkpoints.js
Amxx Aug 30, 2022
c891f58
fix generation
Amxx Aug 30, 2022
47ed78e
add History.getAtRecentBlock & tests
Amxx Aug 30, 2022
533ccef
wording
Amxx Aug 30, 2022
6caea6a
Merge branch 'master' into feature/checkpoint
Amxx Aug 30, 2022
22491c0
wrap docs
frangio Aug 30, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
Use unsafeAccess to save gas
  • Loading branch information
Amxx committed Jul 29, 2022
commit 28f6e6fb3ff40cb98841090d1d74c4d30e34f7dd
15 changes: 13 additions & 2 deletions contracts/utils/Arrays.sol
Original file line number Diff line number Diff line change
Expand Up @@ -31,18 +31,29 @@ library Arrays {

// Note that mid will always be strictly less than high (i.e. it will be a valid array index)
// because Math.average rounds down (it does integer division with truncation).
if (array[mid] > element) {
if (unsafeAccess(array, mid) > element) {
high = mid;
} else {
low = mid + 1;
}
}

// At this point `low` is the exclusive upper bound. We will return the inclusive upper bound.
if (low > 0 && array[low - 1] == element) {
if (low > 0 && unsafeAccess(array, low - 1) == element) {
return low - 1;
} else {
return low;
}
}

/**
* @dev Access an array in an "unsafe" way. Skips solidity "index-out-of-range" check.
* @notice WARNING: only use if you are certain pos is lower then the array length.
*/
function unsafeAccess(uint256[] storage arr, uint256 pos) internal view returns (uint256 result) {
assembly {
mstore(0, arr.slot)
result := sload(add(keccak256(0, 0x20), pos))
}
}
}
52 changes: 33 additions & 19 deletions contracts/utils/Checkpoints.sol
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ library Checkpoints {

function latest(Checkpoint224[] storage self) internal view returns (uint224) {
uint256 pos = self.length;
return pos == 0 ? 0 : self[pos - 1]._value;
return pos == 0 ? 0 : _unsafeAccess(self, pos - 1)._value;
}

function push(
Amxx marked this conversation as resolved.
Show resolved Hide resolved
Expand All @@ -35,14 +35,14 @@ library Checkpoints {

if (pos > 0) {
// Use of memory is important here.
Amxx marked this conversation as resolved.
Show resolved Hide resolved
Checkpoint224 memory last = self[pos - 1];
Checkpoint224 memory last = _unsafeAccess(self, pos - 1);

// Checkpoints keys must be increassing.
Amxx marked this conversation as resolved.
Show resolved Hide resolved
require(last._key <= key, "Checkpoint: invalid key");

// Update or push new checkpoint
if (last._key == key) {
self[pos - 1]._value = value;
_unsafeAccess(self, pos - 1)._value = value;
} else {
self.push(Checkpoint224({_key: key, _value: value}));
}
Expand All @@ -56,28 +56,28 @@ library Checkpoints {
function lowerLookup(Checkpoint224[] storage self, uint32 key) internal view returns (uint224) {
uint256 length = self.length;
uint256 pos = _lowerDichotomicLookup(self, key, 0, length);
return pos == length ? 0 : self[pos]._value;
return pos == length ? 0 : _unsafeAccess(self, pos)._value;
}

function upperLookup(Checkpoint224[] storage self, uint32 key) internal view returns (uint224) {
uint256 length = self.length;
uint256 pos = _upperDichotomicLookup(self, key, 0, length);
return pos == 0 ? 0 : self[pos - 1]._value;
return pos == 0 ? 0 : _unsafeAccess(self, pos - 1)._value;
}

function upperLookupRecent(Checkpoint224[] storage self, uint32 key) internal view returns (uint224) {
uint256 length = self.length;
uint256 offset = 1;

while (offset <= length && self[length - offset]._key > key) {
while (offset <= length && _unsafeAccess(self, length - offset)._key > key) {
offset <<= 1;
}

uint256 low = offset < length ? length - offset : 0;
uint256 high = length - (offset >> 1);
uint256 pos = _upperDichotomicLookup(self, key, low, high);

return pos == 0 ? 0 : self[pos - 1]._value;
return pos == 0 ? 0 : _unsafeAccess(self, pos - 1)._value;
}

function _upperDichotomicLookup(
Amxx marked this conversation as resolved.
Show resolved Hide resolved
Expand All @@ -88,7 +88,7 @@ library Checkpoints {
) private view returns (uint256) {
while (low < high) {
uint256 mid = Math.average(low, high);
if (self[mid]._key > key) {
if (_unsafeAccess(self, mid)._key > key) {
high = mid;
} else {
low = mid + 1;
Expand All @@ -105,7 +105,7 @@ library Checkpoints {
) private view returns (uint256) {
while (low < high) {
uint256 mid = Math.average(low, high);
if (self[mid]._key < key) {
if (_unsafeAccess(self, mid)._key < key) {
low = mid + 1;
} else {
high = mid;
Expand All @@ -114,14 +114,21 @@ library Checkpoints {
return high;
}

function _unsafeAccess(Checkpoint224[] storage self, uint256 pos) private view returns (Checkpoint224 storage result) {
assembly {
mstore(0, self.slot)
result.slot := add(keccak256(0, 0x20), pos)
}
}

struct Checkpoint160 {
uint96 _key;
uint160 _value;
}

function latest(Checkpoint160[] storage self) internal view returns (uint160) {
uint256 pos = self.length;
return pos == 0 ? 0 : self[pos - 1]._value;
return pos == 0 ? 0 : _unsafeAccess(self, pos - 1)._value;
}

function push(
Expand All @@ -133,14 +140,14 @@ library Checkpoints {

if (pos > 0) {
// Use of memory is important here.
Checkpoint160 memory last = self[pos - 1];
Checkpoint160 memory last = _unsafeAccess(self, pos - 1);

// Checkpoints keys must be increassing.
require(last._key <= key, "Checkpoint: invalid key");

// Update or push new checkpoint
if (last._key == key) {
self[pos - 1]._value = value;
_unsafeAccess(self, pos - 1)._value = value;
} else {
self.push(Checkpoint160({_key: key, _value: value}));
}
Expand All @@ -154,28 +161,28 @@ library Checkpoints {
function lowerLookup(Checkpoint160[] storage self, uint96 key) internal view returns (uint160) {
uint256 length = self.length;
uint256 pos = _lowerDichotomicLookup(self, key, 0, length);
return pos == length ? 0 : self[pos]._value;
return pos == length ? 0 : _unsafeAccess(self, pos)._value;
}

function upperLookup(Checkpoint160[] storage self, uint96 key) internal view returns (uint160) {
uint256 length = self.length;
uint256 pos = _upperDichotomicLookup(self, key, 0, length);
return pos == 0 ? 0 : self[pos - 1]._value;
return pos == 0 ? 0 : _unsafeAccess(self, pos - 1)._value;
}

function upperLookupRecent(Checkpoint160[] storage self, uint96 key) internal view returns (uint224) {
uint256 length = self.length;
uint256 offset = 1;

while (offset <= length && self[length - offset]._key > key) {
while (offset <= length && _unsafeAccess(self, length - offset)._key > key) {
offset <<= 1;
}

uint256 low = offset < length ? length - offset : 0;
uint256 high = length - (offset >> 1);
uint256 pos = _upperDichotomicLookup(self, key, low, high);

return pos == 0 ? 0 : self[pos - 1]._value;
return pos == 0 ? 0 : _unsafeAccess(self, pos - 1)._value;
}

function _upperDichotomicLookup(
Expand All @@ -186,7 +193,7 @@ library Checkpoints {
) private view returns (uint256) {
while (low < high) {
uint256 mid = Math.average(low, high);
if (self[mid]._key > key) {
if (_unsafeAccess(self, mid)._key > key) {
high = mid;
} else {
low = mid + 1;
Expand All @@ -203,7 +210,7 @@ library Checkpoints {
) private view returns (uint256) {
while (low < high) {
uint256 mid = Math.average(low, high);
if (self[mid]._key < key) {
if (_unsafeAccess(self, mid)._key < key) {
low = mid + 1;
} else {
high = mid;
Expand All @@ -212,6 +219,13 @@ library Checkpoints {
return high;
}

function _unsafeAccess(Checkpoint160[] storage self, uint256 pos) private view returns (Checkpoint160 storage result) {
assembly {
mstore(0, self.slot)
result.slot := add(keccak256(0, 0x20), pos)
}
}

struct History {
Checkpoint224[] _checkpoints;
}
Expand All @@ -230,7 +244,7 @@ library Checkpoints {
function getAtBlock(History storage self, uint256 blockNumber) internal view returns (uint256) {
require(blockNumber < block.number, "Checkpoints: block not yet mined");

return upperLookup(self._checkpoints, SafeCast.toUint32(blockNumber));
return upperLookupRecent(self._checkpoints, SafeCast.toUint32(blockNumber));
}

/**
Expand Down
25 changes: 16 additions & 9 deletions scripts/generate/templates/Checkpoints.js
Original file line number Diff line number Diff line change
Expand Up @@ -74,7 +74,7 @@ struct Checkpoint${length} {

function latest(Checkpoint${length}[] storage self) internal view returns (uint${length}) {
uint256 pos = self.length;
return pos == 0 ? 0 : self[pos - 1]._value;
return pos == 0 ? 0 : _unsafeAccess(self, pos - 1)._value;
}

function push(
Expand All @@ -86,14 +86,14 @@ function push(

if (pos > 0) {
// Use of memory is important here.
Checkpoint${length} memory last = self[pos - 1];
Checkpoint${length} memory last = _unsafeAccess(self, pos - 1);

// Checkpoints keys must be increassing.
require(last._key <= key, "Checkpoint: invalid key");

// Update or push new checkpoint
if (last._key == key) {
self[pos - 1]._value = value;
_unsafeAccess(self, pos - 1)._value = value;
} else {
self.push(Checkpoint${length}({_key: key, _value: value}));
}
Expand All @@ -107,28 +107,28 @@ function push(
function lowerLookup(Checkpoint${length}[] storage self, uint${256 - length} key) internal view returns (uint${length}) {
uint256 length = self.length;
uint256 pos = _lowerDichotomicLookup(self, key, 0, length);
return pos == length ? 0 : self[pos]._value;
return pos == length ? 0 : _unsafeAccess(self, pos)._value;
}

function upperLookup(Checkpoint${length}[] storage self, uint${256 - length} key) internal view returns (uint${length}) {
uint256 length = self.length;
uint256 pos = _upperDichotomicLookup(self, key, 0, length);
return pos == 0 ? 0 : self[pos - 1]._value;
return pos == 0 ? 0 : _unsafeAccess(self, pos - 1)._value;
}

function upperLookupRecent(Checkpoint${length}[] storage self, uint${256 - length} key) internal view returns (uint224) {
uint256 length = self.length;
uint256 offset = 1;

while (offset <= length && self[length - offset]._key > key) {
while (offset <= length && _unsafeAccess(self, length - offset)._key > key) {
offset <<= 1;
}

Amxx marked this conversation as resolved.
Show resolved Hide resolved
uint256 low = offset < length ? length - offset : 0;
uint256 high = length - (offset >> 1);
uint256 pos = _upperDichotomicLookup(self, key, low, high);

return pos == 0 ? 0 : self[pos - 1]._value;
return pos == 0 ? 0 : _unsafeAccess(self, pos - 1)._value;
}

function _upperDichotomicLookup(
Expand All @@ -139,7 +139,7 @@ function _upperDichotomicLookup(
) private view returns (uint256) {
while (low < high) {
uint256 mid = Math.average(low, high);
if (self[mid]._key > key) {
if (_unsafeAccess(self, mid)._key > key) {
high = mid;
} else {
low = mid + 1;
Expand All @@ -156,14 +156,21 @@ function _lowerDichotomicLookup(
) private view returns (uint256) {
while (low < high) {
uint256 mid = Math.average(low, high);
if (self[mid]._key < key) {
if (_unsafeAccess(self, mid)._key < key) {
low = mid + 1;
} else {
high = mid;
}
}
return high;
}

function _unsafeAccess(Checkpoint${length}[] storage self, uint256 pos) private view returns (Checkpoint${length} storage result) {
assembly {
mstore(0, self.slot)
result.slot := add(keccak256(0, 0x20), pos)
}
}
`;
/* eslint-enable max-len */

Expand Down