@@ -32,28 +32,22 @@ contract ValidatorRewarder is IValidatorRewarder, UUPSUpgradeable, OwnableUpgrad
32
32
/// @notice The token that this rewarder mints
33
33
Recall public token;
34
34
35
- /// @notice The latest checkpoint height that rewards can be claimed for
36
- /// @dev Using uint64 to match Filecoin's epoch height type and save gas when interacting with the network
37
- uint64 public latestClaimedCheckpoint;
38
-
39
35
/// @notice The bottomup checkpoint period for the subnet.
40
36
/// @dev The checkpoint period is set when the subnet is created.
41
37
uint256 public checkpointPeriod;
42
38
43
- /// @notice The supply of RECALL tokens at each checkpoint
44
- mapping (uint64 checkpointHeight = > uint256 totalSupply ) public checkpointToSupply;
45
-
46
- /// @notice The inflation rate for the subnet
47
- /// @dev The rate is expressed as a decimal*1e18.
48
- /// @dev For example 5% APY is 0.0000928276004952% yield per checkpoint period.
49
- /// @dev This is expressed as 928_276_004_952 or 0.000000928276004952*1e18.
50
- uint256 public constant INFLATION_RATE = 928_276_004_952 ;
39
+ /// @notice The number of blocks required to generate 1 new token (with 18 decimals)
40
+ uint256 public constant BLOCKS_PER_TOKEN = 3 ;
51
41
52
42
// ========== EVENTS & ERRORS ==========
53
43
54
44
event ActiveStateChange (bool active , address account );
55
45
event SubnetUpdated (SubnetID subnet , uint256 checkpointPeriod );
56
- event CheckpointClaimed (uint64 indexed checkpointHeight , address indexed validator , uint256 amount );
46
+ /// @notice Emitted when a validator claims their rewards for a checkpoint
47
+ /// @param checkpointHeight The height of the checkpoint for which rewards are claimed
48
+ /// @param validator The address of the validator claiming rewards
49
+ /// @param amount The amount of tokens claimed as reward
50
+ event RewardsClaimed (uint64 indexed checkpointHeight , address indexed validator , uint256 amount );
57
51
58
52
error SubnetMismatch (SubnetID id );
59
53
error InvalidClaimNotifier (address notifier );
@@ -136,83 +130,52 @@ contract ValidatorRewarder is IValidatorRewarder, UUPSUpgradeable, OwnableUpgrad
136
130
revert InvalidClaimNotifier (msg .sender );
137
131
}
138
132
139
- // When the supply for the checkpoint is 0, it means that this is the first claim
140
- // for this checkpoint.
141
- // In this case we will set the supply for the checkpoint and
142
- // calculate the inflation and mint the rewards to the rewarder and the first claimant.
143
- // Otherwise, we know the supply for the checkpoint.
144
- // We will calculate the rewards and transfer them to the other claimants for this checkpoint.
145
- uint256 supplyAtCheckpoint = checkpointToSupply[claimedCheckpointHeight];
146
- if (supplyAtCheckpoint == 0 ) {
147
- // Check that the checkpoint height is valid.
148
- if (! validateCheckpointHeight (claimedCheckpointHeight)) {
149
- revert InvalidCheckpointHeight (claimedCheckpointHeight);
150
- }
151
-
152
- // Get the current supply of RECALL tokens
153
- uint256 currentSupply = token.totalSupply ();
154
-
155
- // Set the supply for the checkpoint and update latest claimed checkpoint
156
- checkpointToSupply[claimedCheckpointHeight] = currentSupply;
157
- latestClaimedCheckpoint = claimedCheckpointHeight;
158
-
159
- // Calculate rewards
160
- uint256 supplyDelta = calculateInflationForCheckpoint (currentSupply);
161
- uint256 validatorShare = calculateValidatorShare (data.blocksCommitted, supplyDelta);
162
-
163
- // Perform external interactions after state updates
164
- token.mint (address (this ), supplyDelta - validatorShare);
165
- token.mint (data.validator, validatorShare);
166
- emit CheckpointClaimed (claimedCheckpointHeight, data.validator, validatorShare);
167
- } else {
168
- // Calculate the supply delta for the checkpoint
169
- uint256 supplyDelta = calculateInflationForCheckpoint (supplyAtCheckpoint);
170
- // Calculate the validator's share of the supply delta
171
- uint256 validatorShare = calculateValidatorShare (data.blocksCommitted, supplyDelta);
172
- // Transfer the validator's share of the supply delta to the validator
173
- token.safeTransfer (data.validator, validatorShare);
174
- emit CheckpointClaimed (claimedCheckpointHeight, data.validator, validatorShare);
133
+ // Check that the checkpoint height is valid
134
+ if (! validateCheckpointHeight (claimedCheckpointHeight)) {
135
+ revert InvalidCheckpointHeight (claimedCheckpointHeight);
175
136
}
137
+
138
+ // Calculate rewards for this checkpoint
139
+ uint256 newTokens = calculateNewTokensForCheckpoint ();
140
+ uint256 validatorShare = calculateValidatorShare (data.blocksCommitted, newTokens);
141
+
142
+ // Mint the validator's share
143
+ token.mint (data.validator, validatorShare);
144
+ emit RewardsClaimed (claimedCheckpointHeight, data.validator, validatorShare);
176
145
}
177
146
178
147
// ========== INTERNAL FUNCTIONS ==========
179
148
180
- /// @notice The internal method to calculate the supply delta for a checkpoint
181
- /// @param supply The token supply at the checkpoint
182
- /// @return The supply delta, i.e. the amount of new tokens minted for the checkpoint
183
- function calculateInflationForCheckpoint (uint256 supply ) internal pure returns (uint256 ) {
184
- UD60x18 supplyFixed = ud (supply);
185
- UD60x18 inflationRateFixed = ud (INFLATION_RATE);
186
- UD60x18 result = supplyFixed.mul (inflationRateFixed);
187
- return result.unwrap ();
149
+ /// @notice Calculates the total number of new tokens to be minted for a checkpoint
150
+ /// @return The number of new tokens to be minted (in base units with 18 decimals)
151
+ function calculateNewTokensForCheckpoint () internal view returns (uint256 ) {
152
+ UD60x18 blocksPerToken = ud (BLOCKS_PER_TOKEN);
153
+ UD60x18 period = ud (checkpointPeriod);
154
+ UD60x18 oneToken = ud (1 ether);
155
+
156
+ // Calculate (checkpointPeriod * 1 ether) / BLOCKS_PER_TOKEN using fixed-point math
157
+ return period.mul (oneToken).div (blocksPerToken).unwrap ();
188
158
}
189
159
190
- /// @notice The internal method to calculate the validator's share of the supply delta
160
+ /// @notice The internal method to calculate the validator's share of the new tokens
191
161
/// @param blocksCommitted The number of blocks committed by the validator
192
- /// @param supplyDelta The supply delta, i.e. the amount of new tokens minted for the checkpoint
193
- /// @return The validator's share of the supply delta
194
- function calculateValidatorShare (uint256 blocksCommitted , uint256 supplyDelta ) internal view returns (uint256 ) {
195
- UD60x18 blocksFixed = ud (blocksCommitted);
196
- UD60x18 deltaFixed = ud (supplyDelta );
197
- UD60x18 periodFixed = ud (checkpointPeriod);
198
- UD60x18 share = blocksFixed .div (periodFixed );
199
- UD60x18 result = share.mul (deltaFixed );
162
+ /// @param totalNewTokens The total number of new tokens for the checkpoint
163
+ /// @return The validator's share of the new tokens
164
+ function calculateValidatorShare (uint256 blocksCommitted , uint256 totalNewTokens ) internal view returns (uint256 ) {
165
+ UD60x18 blocks = ud (blocksCommitted);
166
+ UD60x18 tokens = ud (totalNewTokens );
167
+ UD60x18 period = ud (checkpointPeriod);
168
+ UD60x18 share = blocks .div (period );
169
+ UD60x18 result = share.mul (tokens );
200
170
return result.unwrap ();
201
171
}
202
172
203
173
/// @notice Validates that the claimed checkpoint height is valid
204
174
/// @param claimedCheckpointHeight The height of the checkpoint that the validator is claiming for
205
175
/// @return True if the checkpoint height is valid, false otherwise
206
- /// @dev When the latest claimable checkpoint is not set (0), it means that _this_ is the first ever claim.
207
- /// @dev In this case, we need to ensure the first claim is at the first checkpoint period.
208
- /// @dev Otherwise, we must ensure that the claimed checkpoint is the next expected checkpoint.
176
+ /// @dev Ensures the checkpoint height is a multiple of the checkpoint period
209
177
function validateCheckpointHeight (uint64 claimedCheckpointHeight ) internal view returns (bool ) {
210
- if (latestClaimedCheckpoint == 0 ) {
211
- // First claim must be at the first checkpoint period
212
- return claimedCheckpointHeight == checkpointPeriod;
213
- }
214
- // Subsequent claims must be at the next checkpoint
215
- return claimedCheckpointHeight == latestClaimedCheckpoint + checkpointPeriod;
178
+ return claimedCheckpointHeight > 0 && claimedCheckpointHeight % checkpointPeriod == 0 ;
216
179
}
217
180
218
181
/// @dev Function that should revert when `msg.sender` is not authorized to upgrade the contract
0 commit comments