2023-06-19 00:08:10 +00:00
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
< html lang = "en" >
< head >
< meta http-equiv = "Content-Type" content = "text/html; charset=UTF-8" >
< title > LCOV - lcov.info - v1/staking/TornadoStakingRewards.sol< / title >
< link rel = "stylesheet" type = "text/css" href = "../../gcov.css" >
< / head >
< body >
< table width = "100%" border = 0 cellspacing = 0 cellpadding = 0 >
< tr > < td class = "title" > LCOV - code coverage report< / td > < / tr >
< tr > < td class = "ruler" > < img src = "../../glass.png" width = 3 height = 3 alt = "" > < / td > < / tr >
< tr >
< td width = "100%" >
< table cellpadding = 1 border = 0 width = "100%" >
< tr >
< td width = "10%" class = "headerItem" > Current view:< / td >
< td width = "35%" class = "headerValue" > < a href = "../../index.html" > top level< / a > - < a href = "index.html" > v1/staking< / a > - TornadoStakingRewards.sol< span style = "font-size: 80%;" > (source / < a href = "TornadoStakingRewards.sol.func-sort-c.html" > functions< / a > )< / span > < / td >
< td width = "5%" > < / td >
< td width = "15%" > < / td >
< td width = "10%" class = "headerCovTableHead" > Hit< / td >
< td width = "10%" class = "headerCovTableHead" > Total< / td >
< td width = "15%" class = "headerCovTableHead" > Coverage< / td >
< / tr >
< tr >
< td class = "headerItem" > Test:< / td >
< td class = "headerValue" > lcov.info< / td >
< td > < / td >
< td class = "headerItem" > Lines:< / td >
< td class = "headerCovTableEntry" > 0< / td >
< td class = "headerCovTableEntry" > 19< / td >
< td class = "headerCovTableEntryLo" > 0.0 %< / td >
< / tr >
< tr >
< td class = "headerItem" > Date:< / td >
2023-06-20 21:04:29 +00:00
< td class = "headerValue" > 2023-06-20 21:04:08< / td >
2023-06-19 00:08:10 +00:00
< td > < / td >
< td class = "headerItem" > Functions:< / td >
< td class = "headerCovTableEntry" > 0< / td >
< td class = "headerCovTableEntry" > 6< / td >
< td class = "headerCovTableEntryLo" > 0.0 %< / td >
< / tr >
< tr >
< td class = "headerItem" > Legend:< / td >
< td class = "headerValueLeg" > Lines:
< span class = "coverLegendCov" > hit< / span >
< span class = "coverLegendNoCov" > not hit< / span >
< / td >
< td > < / td >
< / tr >
< tr > < td > < img src = "../../glass.png" width = 3 height = 3 alt = "" > < / td > < / tr >
< / table >
< / td >
< / tr >
< tr > < td class = "ruler" > < img src = "../../glass.png" width = 3 height = 3 alt = "" > < / td > < / tr >
< / table >
< table cellpadding = 0 cellspacing = 0 border = 0 >
< tr >
< td > < br > < / td >
< / tr >
< tr >
< td >
< pre class = "sourceHeading" > Line data Source code< / pre >
< pre class = "source" >
< a name = "1" > < span class = "lineNum" > 1 < / span > : // SPDX-License-Identifier: MIT< / a >
< a name = "2" > < span class = "lineNum" > 2 < / span > : < / a >
< a name = "3" > < span class = "lineNum" > 3 < / span > : pragma solidity ^0.6.12;< / a >
< a name = "4" > < span class = "lineNum" > 4 < / span > : pragma experimental ABIEncoderV2;< / a >
< a name = "5" > < span class = "lineNum" > 5 < / span > : < / a >
< a name = "6" > < span class = "lineNum" > 6 < / span > : import { IERC20 } from " @openzeppelin/contracts/token/ERC20/IERC20.sol" ;< / a >
< a name = "7" > < span class = "lineNum" > 7 < / span > : import { SafeMath } from " @openzeppelin/contracts/math/SafeMath.sol" ;< / a >
< a name = "8" > < span class = "lineNum" > 8 < / span > : import { SafeERC20 } from " @openzeppelin/contracts/token/ERC20/SafeERC20.sol" ;< / a >
< a name = "9" > < span class = "lineNum" > 9 < / span > : import { Initializable } from " @openzeppelin/contracts/proxy/Initializable.sol" ;< / a >
< a name = "10" > < span class = "lineNum" > 10 < / span > : import { EnsResolve } from " torn-token/contracts/ENS.sol" ;< / a >
< a name = "11" > < span class = "lineNum" > 11 < / span > : import { ITornadoGovernance } from " ../interfaces/ITornadoGovernance.sol" ;< / a >
< a name = "12" > < span class = "lineNum" > 12 < / span > : < / a >
< a name = "13" > < span class = "lineNum" > 13 < / span > : /**< / a >
< a name = "14" > < span class = "lineNum" > 14 < / span > : * @notice This is the staking contract of the governance staking upgrade.< / a >
< a name = "15" > < span class = "lineNum" > 15 < / span > : * This contract should hold the staked funds which are received upon relayer registration,< / a >
< a name = "16" > < span class = "lineNum" > 16 < / span > : * and properly attribute rewards to addresses without security issues.< / a >
< a name = "17" > < span class = "lineNum" > 17 < / span > : * @dev CONTRACT RISKS:< / a >
< a name = "18" > < span class = "lineNum" > 18 < / span > : * - Relayer staked TORN at risk if contract is compromised.< / a >
< a name = "19" > < span class = "lineNum" > 19 < / span > : * */< / a >
< a name = "20" > < span class = "lineNum" > 20 < / span > : contract TornadoStakingRewards is Initializable, EnsResolve {< / a >
< a name = "21" > < span class = "lineNum" > 21 < / span > : using SafeMath for uint256;< / a >
< a name = "22" > < span class = "lineNum" > 22 < / span > : using SafeERC20 for IERC20;< / a >
< a name = "23" > < span class = "lineNum" > 23 < / span > : < / a >
< a name = "24" > < span class = "lineNum" > 24 < / span > : /// @notice 1e25< / a >
< a name = "25" > < span class = "lineNum" > 25 < / span > : uint256 public immutable ratioConstant;< / a >
< a name = "26" > < span class = "lineNum" > 26 < / span > : ITornadoGovernance public immutable Governance;< / a >
< a name = "27" > < span class = "lineNum" > 27 < / span > : IERC20 public immutable torn;< / a >
< a name = "28" > < span class = "lineNum" > 28 < / span > : address public immutable relayerRegistry;< / a >
< a name = "29" > < span class = "lineNum" > 29 < / span > : < / a >
< a name = "30" > < span class = "lineNum" > 30 < / span > : /// @notice the sum torn_burned_i/locked_amount_i*coefficient where i is incremented at each burn< / a >
< a name = "31" > < span class = "lineNum" > 31 < / span > : uint256 public accumulatedRewardPerTorn;< / a >
< a name = "32" > < span class = "lineNum" > 32 < / span > : /// @notice notes down accumulatedRewardPerTorn for an address on a lock/unlock/claim< / a >
< a name = "33" > < span class = "lineNum" > 33 < / span > : mapping(address => uint256) public accumulatedRewardRateOnLastUpdate;< / a >
< a name = "34" > < span class = "lineNum" > 34 < / span > : /// @notice notes down how much an account may claim< / a >
< a name = "35" > < span class = "lineNum" > 35 < / span > : mapping(address => uint256) public accumulatedRewards;< / a >
< a name = "36" > < span class = "lineNum" > 36 < / span > : < / a >
< a name = "37" > < span class = "lineNum" > 37 < / span > : event RewardsUpdated(address indexed account, uint256 rewards);< / a >
< a name = "38" > < span class = "lineNum" > 38 < / span > : event RewardsClaimed(address indexed account, uint256 rewardsClaimed);< / a >
< a name = "39" > < span class = "lineNum" > 39 < / span > : < / a >
< a name = "40" > < span class = "lineNum" > 40 < / span > : modifier onlyGovernance() {< / a >
< a name = "41" > < span class = "lineNum" > 41 < / span > : require(msg.sender == address(Governance), " only governance" );< / a >
< a name = "42" > < span class = "lineNum" > 42 < / span > : _;< / a >
< a name = "43" > < span class = "lineNum" > 43 < / span > : }< / a >
< a name = "44" > < span class = "lineNum" > 44 < / span > : < / a >
< a name = "45" > < span class = "lineNum" > 45 < / span > : constructor(< / a >
< a name = "46" > < span class = "lineNum" > 46 < / span > : address governanceAddress,< / a >
< a name = "47" > < span class = "lineNum" > 47 < / span > : address tornAddress,< / a >
< a name = "48" > < span class = "lineNum" > 48 < / span > : bytes32 _relayerRegistry< / a >
< a name = "49" > < span class = "lineNum" > 49 < / span > : ) public {< / a >
< a name = "50" > < span class = "lineNum" > 50 < / span > : Governance = ITornadoGovernance(governanceAddress);< / a >
< a name = "51" > < span class = "lineNum" > 51 < / span > : torn = IERC20(tornAddress);< / a >
< a name = "52" > < span class = "lineNum" > 52 < / span > : relayerRegistry = resolve(_relayerRegistry);< / a >
< a name = "53" > < span class = "lineNum" > 53 < / span > : ratioConstant = IERC20(tornAddress).totalSupply();< / a >
< a name = "54" > < span class = "lineNum" > 54 < / span > : }< / a >
< a name = "55" > < span class = "lineNum" > 55 < / span > : < / a >
< a name = "56" > < span class = "lineNum" > 56 < / span > : /**< / a >
< a name = "57" > < span class = "lineNum" > 57 < / span > : * @notice This function should safely send a user his rewards.< / a >
< a name = "58" > < span class = "lineNum" > 58 < / span > : * @dev IMPORTANT FUNCTION:< / a >
< a name = "59" > < span class = "lineNum" > 59 < / span > : * We know that rewards are going to be updated every time someone locks or unlocks< / a >
< a name = "60" > < span class = "lineNum" > 60 < / span > : * so we know that this function can't be used to falsely increase the amount of< / a >
< a name = "61" > < span class = "lineNum" > 61 < / span > : * lockedTorn by locking in governance and subsequently calling it.< / a >
< a name = "62" > < span class = "lineNum" > 62 < / span > : * - set rewards to 0 greedily< / a >
< a name = "63" > < span class = "lineNum" > 63 < / span > : */< / a >
< a name = "64" > < span class = "lineNum" > 64 < / span > : function getReward() external {< / a >
< a name = "65" > < span class = "lineNum" > 65 < / span > < span class = "lineNoCov" > 0 : uint256 rewards = _updateReward(msg.sender, Governance.lockedBalance(msg.sender));< / span > < / a >
< a name = "66" > < span class = "lineNum" > 66 < / span > < span class = "lineNoCov" > 0 : rewards = rewards.add(accumulatedRewards[msg.sender]);< / span > < / a >
< a name = "67" > < span class = "lineNum" > 67 < / span > < span class = "lineNoCov" > 0 : accumulatedRewards[msg.sender] = 0;< / span > < / a >
< a name = "68" > < span class = "lineNum" > 68 < / span > < span class = "lineNoCov" > 0 : torn.safeTransfer(msg.sender, rewards);< / span > < / a >
< a name = "69" > < span class = "lineNum" > 69 < / span > < span class = "lineNoCov" > 0 : emit RewardsClaimed(msg.sender, rewards);< / span > < / a >
< a name = "70" > < span class = "lineNum" > 70 < / span > : }< / a >
< a name = "71" > < span class = "lineNum" > 71 < / span > : < / a >
< a name = "72" > < span class = "lineNum" > 72 < / span > : /**< / a >
< a name = "73" > < span class = "lineNum" > 73 < / span > : * @notice This function should increment the proper amount of rewards per torn for the contract< / a >
< a name = "74" > < span class = "lineNum" > 74 < / span > : * @dev IMPORTANT FUNCTION:< / a >
< a name = "75" > < span class = "lineNum" > 75 < / span > : * - calculation must not overflow with extreme values< / a >
< a name = "76" > < span class = "lineNum" > 76 < / span > : * (amount < = 1e25) * 1e25 / (balance of vault < = 1e25) -> (extreme values)< / a >
< a name = "77" > < span class = "lineNum" > 77 < / span > : * @param amount amount to add to the rewards< / a >
< a name = "78" > < span class = "lineNum" > 78 < / span > : */< / a >
< a name = "79" > < span class = "lineNum" > 79 < / span > : function addBurnRewards(uint256 amount) external {< / a >
< a name = "80" > < span class = "lineNum" > 80 < / span > < span class = "lineNoCov" > 0 : require(msg.sender == address(Governance) || msg.sender == relayerRegistry, " unauthorized" );< / span > < / a >
< a name = "81" > < span class = "lineNum" > 81 < / span > < span class = "lineNoCov" > 0 : accumulatedRewardPerTorn = accumulatedRewardPerTorn.add(< / span > < / a >
< a name = "82" > < span class = "lineNum" > 82 < / span > : amount.mul(ratioConstant).div(torn.balanceOf(address(Governance.userVault())))< / a >
< a name = "83" > < span class = "lineNum" > 83 < / span > : );< / a >
< a name = "84" > < span class = "lineNum" > 84 < / span > : }< / a >
< a name = "85" > < span class = "lineNum" > 85 < / span > : < / a >
< a name = "86" > < span class = "lineNum" > 86 < / span > : /**< / a >
< a name = "87" > < span class = "lineNum" > 87 < / span > : * @notice This function should allow governance to properly update the accumulated rewards rate for an account< / a >
< a name = "88" > < span class = "lineNum" > 88 < / span > : * @param account address of account to update data for< / a >
< a name = "89" > < span class = "lineNum" > 89 < / span > : * @param amountLockedBeforehand the balance locked beforehand in the governance contract< / a >
< a name = "90" > < span class = "lineNum" > 90 < / span > : * */< / a >
< a name = "91" > < span class = "lineNum" > 91 < / span > : function updateRewardsOnLockedBalanceChange(address account, uint256 amountLockedBeforehand) external onlyGovernance {< / a >
< a name = "92" > < span class = "lineNum" > 92 < / span > < span class = "lineNoCov" > 0 : uint256 claimed = _updateReward(account, amountLockedBeforehand);< / span > < / a >
< a name = "93" > < span class = "lineNum" > 93 < / span > < span class = "lineNoCov" > 0 : accumulatedRewards[account] = accumulatedRewards[account].add(claimed);< / span > < / a >
< a name = "94" > < span class = "lineNum" > 94 < / span > : }< / a >
< a name = "95" > < span class = "lineNum" > 95 < / span > : < / a >
< a name = "96" > < span class = "lineNum" > 96 < / span > : /**< / a >
< a name = "97" > < span class = "lineNum" > 97 < / span > : * @notice This function should allow governance rescue tokens from the staking rewards contract< / a >
< a name = "98" > < span class = "lineNum" > 98 < / span > : * */< / a >
< a name = "99" > < span class = "lineNum" > 99 < / span > : function withdrawTorn(uint256 amount) external onlyGovernance {< / a >
< a name = "100" > < span class = "lineNum" > 100 < / span > < span class = "lineNoCov" > 0 : if (amount == type(uint256).max) amount = torn.balanceOf(address(this));< / span > < / a >
< a name = "101" > < span class = "lineNum" > 101 < / span > < span class = "lineNoCov" > 0 : torn.safeTransfer(address(Governance), amount);< / span > < / a >
< a name = "102" > < span class = "lineNum" > 102 < / span > : }< / a >
< a name = "103" > < span class = "lineNum" > 103 < / span > : < / a >
< a name = "104" > < span class = "lineNum" > 104 < / span > : /**< / a >
< a name = "105" > < span class = "lineNum" > 105 < / span > : * @notice This function should calculated the proper amount of rewards attributed to user since the last update< / a >
< a name = "106" > < span class = "lineNum" > 106 < / span > : * @dev IMPORTANT FUNCTION:< / a >
< a name = "107" > < span class = "lineNum" > 107 < / span > : * - calculation must not overflow with extreme values< / a >
< a name = "108" > < span class = "lineNum" > 108 < / span > : * (accumulatedReward < = 1e25) * (lockedBeforehand < = 1e25) / 1e25< / a >
< a name = "109" > < span class = "lineNum" > 109 < / span > : * - result may go to 0, since this implies on 1 TORN locked => accumulatedReward < = 1e7, meaning a very small reward< / a >
< a name = "110" > < span class = "lineNum" > 110 < / span > : * @param account address of account to calculate rewards for< / a >
< a name = "111" > < span class = "lineNum" > 111 < / span > : * @param amountLockedBeforehand the balance locked beforehand in the governance contract< / a >
< a name = "112" > < span class = "lineNum" > 112 < / span > : * @return claimed the rewards attributed to user since the last update< / a >
< a name = "113" > < span class = "lineNum" > 113 < / span > : */< / a >
< a name = "114" > < span class = "lineNum" > 114 < / span > : function _updateReward(address account, uint256 amountLockedBeforehand) private returns (uint256 claimed) {< / a >
< a name = "115" > < span class = "lineNum" > 115 < / span > < span class = "lineNoCov" > 0 : if (amountLockedBeforehand != 0)< / span > < / a >
< a name = "116" > < span class = "lineNum" > 116 < / span > < span class = "lineNoCov" > 0 : claimed = (accumulatedRewardPerTorn.sub(accumulatedRewardRateOnLastUpdate[account])).mul(amountLockedBeforehand).div(< / span > < / a >
< a name = "117" > < span class = "lineNum" > 117 < / span > : ratioConstant< / a >
< a name = "118" > < span class = "lineNum" > 118 < / span > : );< / a >
< a name = "119" > < span class = "lineNum" > 119 < / span > < span class = "lineNoCov" > 0 : accumulatedRewardRateOnLastUpdate[account] = accumulatedRewardPerTorn;< / span > < / a >
< a name = "120" > < span class = "lineNum" > 120 < / span > < span class = "lineNoCov" > 0 : emit RewardsUpdated(account, claimed);< / span > < / a >
< a name = "121" > < span class = "lineNum" > 121 < / span > : }< / a >
< a name = "122" > < span class = "lineNum" > 122 < / span > : < / a >
< a name = "123" > < span class = "lineNum" > 123 < / span > : /**< / a >
< a name = "124" > < span class = "lineNum" > 124 < / span > : * @notice This function should show a user his rewards.< / a >
< a name = "125" > < span class = "lineNum" > 125 < / span > : * @param account address of account to calculate rewards for< / a >
< a name = "126" > < span class = "lineNum" > 126 < / span > : */< / a >
< a name = "127" > < span class = "lineNum" > 127 < / span > : function checkReward(address account) external view returns (uint256 rewards) {< / a >
< a name = "128" > < span class = "lineNum" > 128 < / span > < span class = "lineNoCov" > 0 : uint256 amountLocked = Governance.lockedBalance(account);< / span > < / a >
< a name = "129" > < span class = "lineNum" > 129 < / span > < span class = "lineNoCov" > 0 : if (amountLocked != 0)< / span > < / a >
< a name = "130" > < span class = "lineNum" > 130 < / span > < span class = "lineNoCov" > 0 : rewards = (accumulatedRewardPerTorn.sub(accumulatedRewardRateOnLastUpdate[account])).mul(amountLocked).div(ratioConstant);< / span > < / a >
< a name = "131" > < span class = "lineNum" > 131 < / span > < span class = "lineNoCov" > 0 : rewards = rewards.add(accumulatedRewards[account]);< / span > < / a >
< a name = "132" > < span class = "lineNum" > 132 < / span > : }< / a >
< a name = "133" > < span class = "lineNum" > 133 < / span > : }< / a >
< / pre >
< / td >
< / tr >
< / table >
< br >
< table width = "100%" border = 0 cellspacing = 0 cellpadding = 0 >
< tr > < td class = "ruler" > < img src = "../../glass.png" width = 3 height = 3 alt = "" > < / td > < / tr >
< tr > < td class = "versionInfo" > Generated by: < a href = "https://github.com/linux-test-project/lcov" target = "_parent" > LCOV version 1.16< / a > < / td > < / tr >
< / table >
< br >
< / body >
< / html >