Skip to content

Commit

Permalink
Add clarifying comments to the RMN 1.5 contract (#1116)
Browse files Browse the repository at this point in the history
  • Loading branch information
gtklocker authored Jun 28, 2024
1 parent 4bd9746 commit ecdd5c2
Show file tree
Hide file tree
Showing 2 changed files with 21 additions and 4 deletions.
4 changes: 2 additions & 2 deletions CODEOWNERS
Original file line number Diff line number Diff line change
Expand Up @@ -148,8 +148,8 @@ flake.lock @smartcontractkit/prodsec-public
/core/services/relay/evm/liquidity_manager.go @smartcontractkit/liquidity-manager
/contracts/**/liquiditymanager/ @smartcontractkit/liquidity-manager

# CCIP ARM
# CCIP RMN
/contracts/src/v0.8/ccip/RMN.sol @smartcontractkit/rmn
/contracts/src/v0.8/ccip/ARMProxy.sol @smartcontractkit/rmn
/contracts/src/v0.8/ccip/interfaces/IARM.sol @smartcontractkit/rmn
/contracts/src/v0.8/ccip/interfaces/IRMN.sol @smartcontractkit/rmn
/contracts/src/v0.8/ccip/test/arm @smartcontractkit/rmn
21 changes: 19 additions & 2 deletions contracts/src/v0.8/ccip/RMN.sol
Original file line number Diff line number Diff line change
Expand Up @@ -102,6 +102,7 @@ contract RMN is IRMN, OwnerIsCreator, ITypeAndVersion {
// Care must be taken that the bitmap has at least as many bits as MAX_NUM_VOTERS.
// uint200 is much larger than we need, but it saves us ~100 gas per voteToBless call to fill the word instead of
// using a smaller type.
// _bitmapGet(voterBitmap, i) = true indicates that the i-th voter has voted to bless
uint200 voterBitmap;
}

Expand Down Expand Up @@ -139,6 +140,8 @@ contract RMN is IRMN, OwnerIsCreator, ITypeAndVersion {
mapping(bytes16 subject => CurseVoteProgress curseVoteProgress) private
s_potentiallyOutdatedCurseVoteProgressBySubject;

// We intentionally use a struct here, even though it contains a single field, to make it obvious to future editors
// that there is space for more fields.
struct CurseHotVars {
uint64 numSubjectsCursed; // incremented by voteToCurse, ownerCurse; decremented by ownerUnvoteToCurse
}
Expand Down Expand Up @@ -752,6 +755,12 @@ contract RMN is IRMN, OwnerIsCreator, ITypeAndVersion {
}
}

/// @return curseVoteAddrs the curseVoteAddr of each voter with an active vote to curse
/// @return cursesHashes the i-th value is the curses hash of curseVoteAddrs[i]
/// @return accumulatedWeight the accumulated weight of all voters with an active vote to curse who are part of the
/// current config
/// @return cursed might be true even if the owner has no active vote and accumulatedWeight < curseWeightThreshold,
/// due to a retained curse from a prior config
/// @dev This is a helper method for offchain code so efficiency is not really a concern.
function getCurseProgress(bytes16 subject)
external
Expand Down Expand Up @@ -868,6 +877,10 @@ contract RMN is IRMN, OwnerIsCreator, ITypeAndVersion {
address[] memory allAddrs = new address[](2 * config.voters.length);
for (uint256 i = 0; i < config.voters.length; ++i) {
Voter memory voter = config.voters[i];
// The owner can always curse using the ownerCurse method, and is not supposed to be included in the voters list.
// Even though the intent is for the actual owner address to NOT be included in the voters list, we don't
// explicitly disallow curseVoteAddr == owner() here. Even if we did, the owner could transfer ownership of the
// contract, and so we couldn't guarantee that the owner is not eventually included in the voters list.
if (
voter.blessVoteAddr == address(0) || voter.curseVoteAddr == address(0)
|| voter.curseVoteAddr == LIFT_CURSE_VOTE_ADDR || voter.curseVoteAddr == OWNER_CURSE_VOTE_ADDR
Expand Down Expand Up @@ -926,9 +939,13 @@ contract RMN is IRMN, OwnerIsCreator, ITypeAndVersion {
}
}
{
// Initialize the owner's CurserRecord
// We could in principle perform this initialization once in the constructor instead, and save a small bit of gas.
// But configuration changes are relatively infrequent, and keeping the initialization here makes the contract's
// correctness easier to reason about.
CurserRecord storage sptr_ownerCurserRecord = s_curserRecords[OWNER_CURSE_VOTE_ADDR];
sptr_ownerCurserRecord.active = true;
sptr_ownerCurserRecord.weight = 0; // pseudo voter
sptr_ownerCurserRecord.active = true; // Assumed by vote/unvote-to-curse logic
sptr_ownerCurserRecord.weight = 0; // Assumed by vote/unvote-to-curse logic
}
s_versionedConfig.blockNumber = uint32(block.number);
emit ConfigSet(configVersion, config);
Expand Down

0 comments on commit ecdd5c2

Please sign in to comment.