generated from PaulRBerg/foundry-template
-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathERC20Rewards.sol
219 lines (175 loc) · 10.1 KB
/
ERC20Rewards.sol
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
// SPDX-License-Identifier: AGPL-3.0-only
pragma solidity ^0.8.0;
import { ERC20 } from "@solmate/tokens/ERC20.sol";
import { SafeTransferLib } from "@solmate/utils/SafeTransferLib.sol";
import { Owned } from "@solmate/auth/Owned.sol";
/// @dev A token inheriting from ERC20Rewards will reward token holders with a rewards token.
/// The rewarded amount will be a fixed wei per second, distributed proportionally to token holders
/// by the size of their holdings.
contract ERC20Rewards is Owned, ERC20 {
using SafeTransferLib for ERC20;
using Cast for uint256;
event RewardsSet(uint32 start, uint32 end, uint256 rate);
event RewardsPerTokenUpdated(uint256 accumulated);
event UserRewardsUpdated(address user, uint256 userRewards, uint256 paidRewardPerToken);
event Claimed(address user, address receiver, uint256 claimed);
struct RewardsInterval {
uint32 start; // Start time for the current rewardsToken schedule
uint32 end; // End time for the current rewardsToken schedule
uint96 rate; // Wei rewarded per second among all token holders
}
struct RewardsPerToken {
uint128 accumulated; // Accumulated rewards per token for the interval, scaled up by 1e18
uint32 lastUpdated; // Last time the rewards per token accumulator was updated
}
struct UserRewards {
uint128 accumulated; // Accumulated rewards for the user until the checkpoint
uint128 checkpoint; // RewardsPerToken the last time the user rewards were updated
}
ERC20 public immutable rewardsToken; // Token used as rewards
RewardsInterval public rewardsInterval; // Interval in which rewards are accumulated by users
RewardsPerToken public rewardsPerToken; // Accumulator to track rewards per token
mapping (address => UserRewards) public accumulatedRewards; // Rewards accumulated per user
constructor(address _owner, ERC20 rewardsToken_, string memory name, string memory symbol, uint8 decimals)
ERC20(name, symbol, decimals)
Owned(_owner)
{
rewardsToken = rewardsToken_;
}
/// @dev Set a rewards schedule
function setRewardsInterval(uint256 start, uint256 end, uint256 totalRewards)
external
onlyOwner
{
require(
start < end,
"Incorrect interval"
);
// A new rewards program can be set if one is not running
require(
block.timestamp.u32() < rewardsInterval.start || block.timestamp.u32() > rewardsInterval.end,
"Rewards still ongoing"
);
// Update the rewards per token so that we don't lose any rewards
_updateRewardsPerToken();
uint256 rate = totalRewards / (end - start);
rewardsInterval.start = start.u32();
rewardsInterval.end = end.u32();
rewardsInterval.rate = rate.u96();
// If setting up a new rewards program, the rewardsPerToken.accumulated is used and built upon
// New rewards start accumulating from the new rewards program start
// Any unaccounted rewards from last program can still be added to the user rewards
// Any unclaimed rewards can still be claimed
rewardsPerToken.lastUpdated = start.u32();
emit RewardsSet(start.u32(), end.u32(), rate);
}
/// @notice Update the rewards per token accumulator according to the rate, the time elapsed since the last update, and the current total staked amount.
function _calculateRewardsPerToken(RewardsPerToken memory rewardsPerTokenIn, RewardsInterval memory rewardsInterval_) internal view returns(RewardsPerToken memory) {
RewardsPerToken memory rewardsPerTokenOut = RewardsPerToken(rewardsPerTokenIn.accumulated, rewardsPerTokenIn.lastUpdated);
uint256 totalSupply_ = totalSupply;
// No changes if the program hasn't started
if (block.timestamp < rewardsInterval_.start) return rewardsPerTokenOut;
// Stop accumulating at the end of the rewards interval
uint256 updateTime = block.timestamp < rewardsInterval_.end ? block.timestamp : rewardsInterval_.end;
uint256 elapsed = updateTime - rewardsPerTokenIn.lastUpdated;
// No changes if no time has passed
if (elapsed == 0) return rewardsPerTokenOut;
rewardsPerTokenOut.lastUpdated = updateTime.u32();
// If there are no stakers we just change the last update time, the rewards for intervals without stakers are not accumulated
if (totalSupply_ == 0) return rewardsPerTokenOut;
// Calculate and update the new value of the accumulator.
rewardsPerTokenOut.accumulated = (rewardsPerTokenIn.accumulated + 1e18 * elapsed * rewardsInterval_.rate / totalSupply_).u128(); // The rewards per token are scaled up for precision
return rewardsPerTokenOut;
}
/// @notice Calculate the rewards accumulated by a stake between two checkpoints.
function _calculateUserRewards(uint256 stake_, uint256 earlierCheckpoint, uint256 latterCheckpoint) internal pure returns (uint256) {
return stake_ * (latterCheckpoint - earlierCheckpoint) / 1e18; // We must scale down the rewards by the precision factor
}
/// @notice Update and return the rewards per token accumulator according to the rate, the time elapsed since the last update, and the current total staked amount.
function _updateRewardsPerToken() internal returns (RewardsPerToken memory){
RewardsPerToken memory rewardsPerTokenIn = rewardsPerToken;
RewardsPerToken memory rewardsPerTokenOut = _calculateRewardsPerToken(rewardsPerTokenIn, rewardsInterval);
// We skip the storage changes if already updated in the same block, or if the program has ended and was updated at the end
if (rewardsPerTokenIn.lastUpdated == rewardsPerTokenOut.lastUpdated) return rewardsPerTokenOut;
rewardsPerToken = rewardsPerTokenOut;
emit RewardsPerTokenUpdated(rewardsPerTokenOut.accumulated);
return rewardsPerTokenOut;
}
/// @notice Calculate and store current rewards for an user. Checkpoint the rewardsPerToken value with the user.
function _updateUserRewards(address user) internal returns (UserRewards memory) {
RewardsPerToken memory rewardsPerToken_ = _updateRewardsPerToken();
UserRewards memory userRewards_ = accumulatedRewards[user];
// We skip the storage changes if there are no changes to the rewards per token accumulator
if (userRewards_.checkpoint == rewardsPerToken_.accumulated) return userRewards_;
// Calculate and update the new value user reserves.
userRewards_.accumulated += _calculateUserRewards(balanceOf[user], userRewards_.checkpoint, rewardsPerToken_.accumulated).u128();
userRewards_.checkpoint = rewardsPerToken_.accumulated;
accumulatedRewards[user] = userRewards_;
emit UserRewardsUpdated(user, userRewards_.accumulated, userRewards_.checkpoint);
return userRewards_;
}
/// @dev Mint tokens, after accumulating rewards for an user and update the rewards per token accumulator.
function _mint(address to, uint256 amount)
internal virtual override
{
_updateUserRewards(to);
super._mint(to, amount);
}
/// @dev Burn tokens, after accumulating rewards for an user and update the rewards per token accumulator.
function _burn(address from, uint256 amount)
internal virtual override
{
_updateUserRewards(from);
super._burn(from, amount);
}
/// @notice Claim rewards for an user
function _claim(address from, address to, uint256 amount) internal virtual {
_updateUserRewards(from);
accumulatedRewards[from].accumulated -= amount.u128();
rewardsToken.safeTransfer(to, amount);
emit Claimed(from, to, amount);
}
/// @dev Transfer tokens, after updating rewards for source and destination.
function transfer(address to, uint amount) public virtual override returns (bool) {
_updateUserRewards(msg.sender);
_updateUserRewards(to);
return super.transfer(to, amount);
}
/// @dev Transfer tokens, after updating rewards for source and destination.
function transferFrom(address from, address to, uint amount) public virtual override returns (bool) {
_updateUserRewards(from);
_updateUserRewards(to);
return super.transferFrom(from, to, amount);
}
/// @notice Claim all rewards for the caller
function claim(address to) public virtual returns (uint256) {
uint256 claimed = currentUserRewards(msg.sender);
_claim(msg.sender, to, claimed);
return claimed;
}
/// @notice Calculate and return current rewards per token.
function currentRewardsPerToken() public view returns (uint256) {
return _calculateRewardsPerToken(rewardsPerToken, rewardsInterval).accumulated;
}
/// @notice Calculate and return current rewards for a user.
/// @dev This repeats the logic used on transactions, but doesn't update the storage.
function currentUserRewards(address user) public view returns (uint256) {
UserRewards memory accumulatedRewards_ = accumulatedRewards[user];
RewardsPerToken memory rewardsPerToken_ = _calculateRewardsPerToken(rewardsPerToken, rewardsInterval);
return accumulatedRewards_.accumulated + _calculateUserRewards(balanceOf[user], accumulatedRewards_.checkpoint, rewardsPerToken_.accumulated);
}
}
library Cast {
function u128(uint256 x) internal pure returns (uint128 y) {
require(x <= type(uint128).max, "Cast overflow");
y = uint128(x);
}
function u96(uint256 x) internal pure returns (uint96 y) {
require(x <= type(uint96).max, "Cast overflow");
y = uint96(x);
}
function u32(uint256 x) internal pure returns (uint32 y) {
require(x <= type(uint32).max, "Cast overflow");
y = uint32(x);
}
}