// File: contracts/module/safeMath.sol
// SPDX-License-Identifier: BSD-3-Clause
pragma solidity 0.6.12;
// from: https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/math/SafeMath.sol
// Subject to the MIT license.
/**
* @title BiFi's safe-math Contract
* @author BiFi(seinmyung25, Miller-kk, tlatkdgus1, dongchangYoo)
*/
contract safeMathModule {
uint256 constant one = 1 ether;
function expDiv(uint256 a, uint256 b) internal pure returns (uint256) {
return safeDiv( safeMul(a, one), b);
}
function expMul(uint256 a, uint256 b) internal pure returns (uint256) {
return safeDiv( safeMul(a, b), one);
}
function safeAdd(uint256 a, uint256 b) internal pure returns (uint256) {
uint256 c = a + b;
require(c >= a, "SafeMath: addtion overflow");
return c;
}
function safeSub(uint256 a, uint256 b) internal pure returns (uint256) {
require(a >= b, "SafeMath: subtraction overflow");
return a - b;
}
function safeMul(uint256 a, uint256 b) internal pure returns (uint256) {
if(a == 0) { return 0;}
uint256 c = a * b;
require( (c/a) == b, "SafeMath: multiplication overflow");
return c;
}
function safeDiv(uint256 a, uint256 b) internal pure returns (uint256) {
require(b > 0, "SafeMath: division by zero");
return (a/b);
}
}
// File: contracts/ERC20.sol
/**
* @title BiFi's ERC20 Mockup Contract
* @author BiFi(seinmyung25, Miller-kk, tlatkdgus1, dongchangYoo)
*/
contract ERC20 {
string symbol;
string name;
uint8 decimals = 18;
uint256 totalSupply = 1000 * 1e9 * 1e18; // token amount: 1000 Bilions
// Owner of this contract
address public owner;
// Balances for each account
mapping(address => uint256) balances;
// Owner of account approves the transfer of an amount to another account
mapping(address => mapping (address => uint256)) allowed;
// Functions with this modifier can only be executed by the owner
modifier onlyOwner() {
require(msg.sender == owner, "only owner");
_;
}
event Transfer(address, address, uint256);
event Approval(address, address, uint256);
// Constructor
constructor (string memory _name, string memory _symbol) public {
owner = msg.sender;
name = _name;
symbol = _symbol;
balances[msg.sender] = totalSupply;
}
// What is the balance of a particular account?
function balanceOf(address _owner) public view returns (uint256 balance) {
return balances[_owner];
}
// Transfer the balance from owner's account to another account
function transfer(address _to, uint256 _amount) public returns (bool success) {
require(balances[msg.sender] >= _amount, "insuficient sender's balance");
require(_amount > 0, "requested amount must be positive");
require(balances[_to] + _amount > balances[_to], "receiver's balance overflows");
balances[msg.sender] -= _amount;
balances[_to] += _amount;
emit Transfer(msg.sender, _to, _amount);
return true;
}
// Send _value amount of tokens from address _from to address _to
// The transferFrom method is used for a withdraw workflow, allowing contracts to send
// tokens on your behalf, for example to "deposit" to a contract address and/or to charge
// fees in sub-currencies; the command should fail unless the _from account has
// deliberately authorized the sender of the message via some mechanism; we propose
// these standardized APIs for approval:
function transferFrom(address _from, address _to,uint256 _amount) public returns (bool success) {
require(balances[_from] >= _amount, "insuficient sender's balance");
require(allowed[_from][msg.sender] >= _amount, "not allowed transfer");
require(_amount > 0, "requested amount must be positive");
require(balances[_to] + _amount > balances[_to], "receiver's balance overflows");
balances[_from] -= _amount;
allowed[_from][msg.sender] -= _amount;
balances[_to] += _amount;
emit Transfer(_from, _to, _amount);
return true;
}
// Allow _spender to withdraw from your account, multiple times, up to the _value amount.
// If this function is called again it overwrites the current allowance with _value.
function approve(address _spender, uint256 _amount) public returns (bool success) {
allowed[msg.sender][_spender] = _amount;
emit Approval(msg.sender, _spender, _amount);
return true;
}
function allowance(address _owner, address _spender) public view returns (uint256 remaining) {
return allowed[_owner][_spender];
}
}
contract BFCtoken is ERC20 {
constructor() public ERC20 ("Bifrost", "BFC") {}
}
contract LPtoken is ERC20 {
constructor() public ERC20 ("BFC-ETH", "LP") {}
}
contract BiFitoken is ERC20 {
constructor() public ERC20 ("BiFi", "BiFi") {}
}
// File: contracts/module/storageModule.sol
/**
* @title BiFi's Reward Distribution Storage Contract
* @notice Define the basic Contract State
* @author BiFi(seinmyung25, Miller-kk, tlatkdgus1, dongchangYoo)
*/
contract storageModule {
address public owner;
bool public claimLock;
bool public withdrawLock;
uint256 public rewardPerBlock;
uint256 public decrementUnitPerBlock;
uint256 public rewardLane;
uint256 public lastBlockNum;
uint256 public totalDeposited;
ERC20 public lpErc; ERC20 public rewardErc;
mapping(address => Account) public accounts;
uint256 public passedPoint;
RewardVelocityPoint[] public registeredPoints;
struct Account {
uint256 deposited;
uint256 pointOnLane;
uint256 rewardAmount;
}
struct RewardVelocityPoint {
uint256 blockNumber;
uint256 rewardPerBlock;
uint256 decrementUnitPerBlock;
}
struct UpdateRewardLaneModel {
uint256 len; uint256 tmpBlockDelta;
uint256 memPassedPoint; uint256 tmpPassedPoint;
uint256 memThisBlockNum;
uint256 memLastBlockNum; uint256 tmpLastBlockNum;
uint256 memTotalDeposit;
uint256 memRewardLane; uint256 tmpRewardLane;
uint256 memRewardPerBlock; uint256 tmpRewardPerBlock;
uint256 memDecrementUnitPerBlock; uint256 tmpDecrementUnitPerBlock;
}
}
// File: contracts/module/eventModule.sol
/**
* @title BiFi's Reward Distribution Event Contract
* @notice Define the service Events
* @author BiFi(seinmyung25, Miller-kk, tlatkdgus1, dongchangYoo)
*/
contract eventModule {
/// @dev Events for user actions
event Deposit(address userAddr, uint256 amount, uint256 userDeposit, uint256 totalDeposit);
event Withdraw(address userAddr, uint256 amount, uint256 userDeposit, uint256 totalDeposit);
event Claim(address userAddr, uint256 amount);
event UpdateRewardParams(uint256 atBlockNumber, uint256 rewardPerBlock, uint256 decrementUnitPerBlock);
/// @dev Events for admin actions below
/// @dev Contracts Access Control
event ClaimLock(bool lock);
event WithdrawLock(bool lock);
event OwnershipTransfer(address from, address to);
/// @dev Distribution Model Parameter editer
event SetRewardParams(uint256 rewardPerBlock, uint256 decrementUnitPerBlock);
event RegisterRewardParams(uint256 atBlockNumber, uint256 rewardPerBlock, uint256 decrementUnitPerBlock);
event DeleteRegisterRewardParams(uint256 index, uint256 atBlockNumber, uint256 rewardPerBlock, uint256 decrementUnitPerBlock, uint256 arrayLen);
}
// File: contracts/module/internalModule.sol
/**
* @title BiFi's Reward Distribution Internal Contract
* @notice Implement the basic functions for staking and reward distribution
* @dev All functions are internal.
* @author BiFi(seinmyung25, Miller-kk, tlatkdgus1, dongchangYoo)
*/
contract internalModule is storageModule, eventModule, safeMathModule {
/**
* @notice Deposit the Contribution Tokens
* @param userAddr The user address of the Contribution Tokens
* @param amount The amount of the Contribution Tokens
*/
function _deposit(address userAddr, uint256 amount) internal {
Account memory user = accounts[userAddr];
uint256 totalDeposit = totalDeposited;
user.deposited = safeAdd(user.deposited, amount);
accounts[userAddr].deposited = user.deposited;
totalDeposit = safeAdd(totalDeposited, amount);
totalDeposited = totalDeposit;
if(amount > 0) {
/// @dev transfer the Contribution Toknes to this contract.
emit Deposit(userAddr, amount, user.deposited, totalDeposit);
require( lpErc.transferFrom(msg.sender, address(this), amount), "token error" );
}
}
/**
* @notice Withdraw the Contribution Tokens
* @param userAddr The user address of the Contribution Tokens
* @param amount The amount of the Contribution Tokens
*/
function _withdraw(address userAddr, uint256 amount) internal {
Account memory user = accounts[userAddr];
uint256 totalDeposit = totalDeposited;
require(user.deposited >= amount, "not enough user Deposit");
user.deposited = safeSub(user.deposited, amount);
accounts[userAddr].deposited = user.deposited;
totalDeposit = safeSub(totalDeposited, amount);
totalDeposited = totalDeposit;
if(amount > 0) {
/// @dev transfer the Contribution Tokens from this contact.
emit Withdraw(userAddr, amount, user.deposited, totalDeposit);
require( lpErc.transfer(userAddr, amount), "token error" );
}
}
/**
* @notice Calculate current reward
* @dev This function is called whenever the balance of the Contribution
Tokens of the user.
* @param userAddr The user address of the Contribution and Reward Tokens
*/
function _redeemAll(address userAddr) internal {
Account memory user = accounts[userAddr];
uint256 newRewardLane = _updateRewardLane();
uint256 distance = safeSub(newRewardLane, user.pointOnLane);
uint256 rewardAmount = expMul(user.deposited, distance);
if(user.pointOnLane != newRewardLane) accounts[userAddr].pointOnLane = newRewardLane;
if(rewardAmount != 0) accounts[userAddr].rewardAmount = safeAdd(user.rewardAmount, rewardAmount);
}
/**
* @notice Claim the Reward Tokens
* @dev Transfer all reward the user has earned at once.
* @param userAddr The user address of the Reward Tokens
*/
function _rewardClaim(address userAddr) internal {
Account memory user = accounts[userAddr];
if(user.rewardAmount != 0) {
uint256 amount = user.rewardAmount;
accounts[userAddr].rewardAmount = 0;
/// @dev transfer the Reward Tokens from this contract.
emit Claim(userAddr, amount);
require(rewardErc.transfer(userAddr, amount), "token error" );
}
}
/**
* @notice Update the reward lane value upto ths currnet moment (block)
* @dev This function should care the "reward velocity points," at which the
parameters of reward distribution are changed.
* @return The current (calculated) reward lane value
*/
function _updateRewardLane() internal returns (uint256) {
/// @dev Set up memory variables used for calculation temporarily.
UpdateRewardLaneModel memory vars;
vars.len = registeredPoints.length;
vars.memTotalDeposit = totalDeposited;
vars.tmpPassedPoint = vars.memPassedPoint = passedPoint;
vars.memThisBlockNum = block.number;
vars.tmpLastBlockNum = vars.memLastBlockNum = lastBlockNum;
vars.tmpRewardLane = vars.memRewardLane = rewardLane;
vars.tmpRewardPerBlock = vars.memRewardPerBlock = rewardPerBlock;
vars.tmpDecrementUnitPerBlock = vars.memDecrementUnitPerBlock = decrementUnitPerBlock;
for(uint256 i=vars.memPassedPoint; i<vars.len; i++) {
RewardVelocityPoint memory point = registeredPoints[i];
/**
* @dev Check whether this reward velocity point is valid and has
not applied yet.
*/
if(vars.tmpLastBlockNum < point.blockNumber && point.blockNumber <= vars.memThisBlockNum) {
vars.tmpPassedPoint = i+1;
/// @dev Update the reward lane with the tmp variables
vars.tmpBlockDelta = safeSub(point.blockNumber, vars.tmpLastBlockNum);
(vars.tmpRewardLane, vars.tmpRewardPerBlock) =
_calcNewRewardLane(
vars.tmpRewardLane,
vars.memTotalDeposit,
vars.tmpRewardPerBlock,
vars.tmpDecrementUnitPerBlock,
vars.tmpBlockDelta);
/// @dev Update the tmp variables with this reward velocity point.
vars.tmpLastBlockNum = point.blockNumber;
vars.tmpRewardPerBlock = point.rewardPerBlock;
vars.tmpDecrementUnitPerBlock = point.decrementUnitPerBlock;
/**
* @dev Notify the update of the parameters (by passing the
reward velocity points)
*/
emit UpdateRewardParams(point.blockNumber, point.rewardPerBlock, point.decrementUnitPerBlock);
} else {
/// @dev sorted array, exit eariler without accessing future points.
break;
}
}
/**
* @dev Update the reward lane for the remained period between the
latest velocity point and this moment (block)
*/
if( vars.tmpLastBlockNum < vars.memThisBlockNum ) {
vars.tmpBlockDelta = safeSub(vars.memThisBlockNum, vars.tmpLastBlockNum);
vars.tmpLastBlockNum = vars.memThisBlockNum;
(vars.tmpRewardLane, vars.tmpRewardPerBlock) =
_calcNewRewardLane(
vars.tmpRewardLane,
vars.memTotalDeposit,
vars.tmpRewardPerBlock,
vars.tmpDecrementUnitPerBlock,
vars.tmpBlockDelta);
}
/**
* @dev Update the reward lane parameters with the tmp variables.
*/
if(vars.memLastBlockNum != vars.tmpLastBlockNum) lastBlockNum = vars.tmpLastBlockNum;
if(vars.memPassedPoint != vars.tmpPassedPoint) passedPoint = vars.tmpPassedPoint;
if(vars.memRewardLane != vars.tmpRewardLane) rewardLane = vars.tmpRewardLane;
if(vars.memRewardPerBlock != vars.tmpRewardPerBlock) rewardPerBlock = vars.tmpRewardPerBlock;
if(vars.memDecrementUnitPerBlock != vars.tmpDecrementUnitPerBlock) decrementUnitPerBlock = vars.tmpDecrementUnitPerBlock;
return vars.tmpRewardLane;
}
/**
* @notice Calculate a new reward lane value with the given parameters
* @param _rewardLane The previous reward lane value
* @param _totalDeposit Thte total deposit amount of the Contribution Tokens
* @param _rewardPerBlock The reward token amount per a block
* @param _decrementUnitPerBlock The decerement amount of the reward token per a block
*/
function _calcNewRewardLane(
uint256 _rewardLane,
uint256 _totalDeposit,
uint256 _rewardPerBlock,
uint256 _decrementUnitPerBlock,
uint256 delta) internal pure returns (uint256, uint256) {
uint256 executableDelta;
if(_decrementUnitPerBlock != 0) {
executableDelta = safeDiv(_rewardPerBlock, _decrementUnitPerBlock);
if(delta > executableDelta) delta = executableDelta;
else executableDelta = 0;
}
uint256 distance;
if(_totalDeposit != 0) {
distance = expMul( _sequencePartialSumAverage(_rewardPerBlock, delta, _decrementUnitPerBlock), safeMul( expDiv(one, _totalDeposit), delta) );
_rewardLane = safeAdd(_rewardLane, distance);
}
if(executableDelta != 0) _rewardPerBlock = 0;
else _rewardPerBlock = _getNewRewardPerBlock(_rewardPerBlock, _decrementUnitPerBlock, delta);
return (_rewardLane, _rewardPerBlock);
}
/**
* @notice Register a new reward velocity point
* @dev We assume that reward velocity points are stored in order of block
number. Namely, registerPoints is always a sorted array.
* @param _blockNumber The block number for the point.
* @param _rewardPerBlock The reward token amount per a block
* @param _decrementUnitPerBlock The decerement amount of the reward token per a block
*/
function _registerRewardVelocity(uint256 _blockNumber, uint256 _rewardPerBlock, uint256 _decrementUnitPerBlock) internal {
RewardVelocityPoint memory varPoint = RewardVelocityPoint(_blockNumber, _rewardPerBlock, _decrementUnitPerBlock);
emit RegisterRewardParams(_blockNumber, _rewardPerBlock, _decrementUnitPerBlock);
registeredPoints.push(varPoint);
}
/**
* @notice Delete a existing reward velocity point
* @dev We assume that reward velocity points are stored in order of block
number. Namely, registerPoints is always a sorted array.
* @param _index The index number of deleting point in state array.
*/
function _deleteRegisteredRewardVelocity(uint256 _index) internal {
uint256 len = registeredPoints.length;
require(len != 0 && _index < len, "error: no elements in registeredPoints");
RewardVelocityPoint memory point = registeredPoints[_index];
emit DeleteRegisterRewardParams(_index, point.blockNumber, point.rewardPerBlock, point.decrementUnitPerBlock, len-1);
for(uint256 i=_index; i<len-1; i++) {
registeredPoints[i] = registeredPoints[i+1];
}
registeredPoints.pop();
}
/**
* @notice Set paramaters for the reward distribution
* @param _rewardPerBlock The reward token amount per a block
* @param _decrementUnitPerBlock The decerement amount of the reward token per a block
*/
function _setParams(uint256 _rewardPerBlock, uint256 _decrementUnitPerBlock) internal {
emit SetRewardParams(_rewardPerBlock, _decrementUnitPerBlock);
rewardPerBlock = _rewardPerBlock;
decrementUnitPerBlock = _decrementUnitPerBlock;
}
/**
* @return the avaerage of the RewardLance of the inactive (i.e., no-action)
periods.
*/
function _sequencePartialSumAverage(uint256 a, uint256 n, uint256 d) internal pure returns (uint256) {
/**
@dev return Sn / n,
where Sn = ( (n{2*a + (n-1)d}) / 2 )
== ( (2na + (n-1)d) / 2 ) / n
caveat: use safeSub() to avoid the case that d is negative
*/
if (n > 0)
return safeDiv(safeSub( safeMul(2,a), safeMul( safeSub(n,1), d) ), 2);
else
return 0;
}
function _getNewRewardPerBlock(uint256 before, uint256 dec, uint256 delta) internal pure returns (uint256) {
return safeSub(before, safeMul(dec, delta));
}
function _setClaimLock(bool lock) internal {
emit ClaimLock(lock);
claimLock = lock;
}
function _setWithdrawLock(bool lock) internal {
emit WithdrawLock(lock);
withdrawLock = lock;
}
function _ownershipTransfer(address to) internal {
emit OwnershipTransfer(msg.sender, to);
owner = to;
}
}
// File: contracts/module/viewModule.sol
/**
* @title BiFi's Reward Distribution View Contract
* @notice Implements the view functions for support front-end
* @author BiFi(seinmyung25, Miller-kk, tlatkdgus1, dongchangYoo)
*/
contract viewModule is internalModule {
function marketInformation(uint256 _fromBlockNumber, uint256 _toBlockNumber) external view returns (
uint256 rewardStartBlockNumber,
uint256 distributedAmount,
uint256 totalDeposit,
uint256 poolRate
)
{
if(rewardPerBlock == 0) rewardStartBlockNumber = registeredPoints[0].blockNumber;
else rewardStartBlockNumber = registeredPoints[0].blockNumber;
distributedAmount = _redeemAllView(address(0));
totalDeposit = totalDeposited;
poolRate = getPoolRate(address(0), _fromBlockNumber, _toBlockNumber);
return (
rewardStartBlockNumber,
distributedAmount,
totalDeposit,
poolRate
);
}
function userInformation(address userAddr, uint256 _fromBlockNumber, uint256 _toBlockNumber) external view returns (
uint256 stakedTokenAmount,
uint256 rewardStartBlockNumber,
uint256 claimStartBlockNumber,
uint256 earnedTokenAmount,
uint256 poolRate
)
{
Account memory user = accounts[userAddr];
stakedTokenAmount = user.deposited;
if(rewardPerBlock == 0) rewardStartBlockNumber = registeredPoints[0].blockNumber;
else rewardStartBlockNumber = registeredPoints[0].blockNumber;
earnedTokenAmount = _redeemAllView(userAddr);
poolRate = getPoolRate(userAddr, _fromBlockNumber, _toBlockNumber);
return (stakedTokenAmount, rewardStartBlockNumber, claimStartBlockNumber, earnedTokenAmount, poolRate);
}
function modelInfo() external view returns (uint256, uint256, uint256, uint256, uint256) {
return (rewardPerBlock, decrementUnitPerBlock, rewardLane, lastBlockNum, totalDeposited);
}
function getParams() external view returns (uint256, uint256, uint256, uint256) {
return (rewardPerBlock, rewardLane, lastBlockNum, totalDeposited);
}
function getRegisteredPointLength() external view returns (uint256) {
return registeredPoints.length;
}
function getRegisteredPoint(uint256 index) external view returns (uint256, uint256, uint256) {
RewardVelocityPoint memory point = registeredPoints[index];
return (point.blockNumber, point.rewardPerBlock, point.decrementUnitPerBlock);
}
function userInfo(address userAddr) external view returns (uint256, uint256, uint256) {
Account memory user = accounts[userAddr];
uint256 earnedRewardAmount = _redeemAllView(userAddr);
return (user.deposited, user.pointOnLane, earnedRewardAmount);
}
function distributionInfo() external view returns (uint256, uint256, uint256) {
uint256 totalDistributedRewardAmount_now = _distributedRewardAmountView();
return (rewardPerBlock, decrementUnitPerBlock, totalDistributedRewardAmount_now);
}
function _distributedRewardAmountView() internal view returns (uint256) {
return _redeemAllView( address(0) );
}
function _redeemAllView(address userAddr) internal view returns (uint256) {
Account memory user;
uint256 newRewardLane;
if( userAddr != address(0) ) {
user = accounts[userAddr];
newRewardLane = _updateRewardLaneView(lastBlockNum);
} else {
user = Account(totalDeposited, 0, 0);
newRewardLane = _updateRewardLaneView(0);
}
uint256 distance = safeSub(newRewardLane, user.pointOnLane);
uint256 rewardAmount = expMul(user.deposited, distance);
return safeAdd(user.rewardAmount, rewardAmount);
}
function _updateRewardLaneView(uint256 fromBlockNumber) internal view returns (uint256) {
/// @dev Set up memory variables used for calculation temporarily.
UpdateRewardLaneModel memory vars;
vars.len = registeredPoints.length;
vars.memTotalDeposit = totalDeposited;
if(fromBlockNumber == 0){
vars.tmpPassedPoint = vars.memPassedPoint = 0;
vars.memThisBlockNum = block.number;
vars.tmpLastBlockNum = vars.memLastBlockNum = 0;
vars.tmpRewardLane = vars.memRewardLane = 0;
vars.tmpRewardPerBlock = vars.memRewardPerBlock = 0;
vars.tmpDecrementUnitPerBlock = vars.memDecrementUnitPerBlock = 0;
} else {
vars.tmpPassedPoint = vars.memPassedPoint = passedPoint;
vars.memThisBlockNum = block.number;
vars.tmpLastBlockNum = vars.memLastBlockNum = fromBlockNumber;
vars.tmpRewardLane = vars.memRewardLane = rewardLane;
vars.tmpRewardPerBlock = vars.memRewardPerBlock = rewardPerBlock;
vars.tmpDecrementUnitPerBlock = vars.memDecrementUnitPerBlock = decrementUnitPerBlock;
}
for(uint256 i=vars.memPassedPoint; i<vars.len; i++) {
RewardVelocityPoint memory point = registeredPoints[i];
/**
* @dev Check whether this reward velocity point is valid and has
not applied yet.
*/
if(vars.tmpLastBlockNum < point.blockNumber && point.blockNumber <= vars.memThisBlockNum) {
vars.tmpPassedPoint = i+1;
/// @dev Update the reward lane with the tmp variables
vars.tmpBlockDelta = safeSub(point.blockNumber, vars.tmpLastBlockNum);
(vars.tmpRewardLane, vars.tmpRewardPerBlock) =
_calcNewRewardLane(
vars.tmpRewardLane,
vars.memTotalDeposit,
vars.tmpRewardPerBlock,
vars.tmpDecrementUnitPerBlock,
vars.tmpBlockDelta);
/// @dev Update the tmp variables with this reward velocity point.
vars.tmpLastBlockNum = point.blockNumber;
vars.tmpRewardPerBlock = point.rewardPerBlock;
vars.tmpDecrementUnitPerBlock = point.decrementUnitPerBlock;
/**
* @dev Notify the update of the parameters (by passing the
reward velocity points)
*/
} else {
/// @dev sorted array, exit eariler without accessing future points.
break;
}
}
/**
* @dev Update the reward lane for the remained period between the
latest velocity point and this moment (block)
*/
if(vars.memThisBlockNum > vars.tmpLastBlockNum) {
vars.tmpBlockDelta = safeSub(vars.memThisBlockNum, vars.tmpLastBlockNum);
vars.tmpLastBlockNum = vars.memThisBlockNum;
(vars.tmpRewardLane, vars.tmpRewardPerBlock) =
_calcNewRewardLane(
vars.tmpRewardLane,
vars.memTotalDeposit,
vars.tmpRewardPerBlock,
vars.tmpDecrementUnitPerBlock,
vars.tmpBlockDelta);
}
return vars.tmpRewardLane;
}
/**
* @notice Get The rewardPerBlock of user in suggested period(see params)
* @param userAddr The Address of user, 0 for total
* @param fromBlockNumber calculation start block number
* @param toBlockNumber calculation end block number
* @notice this function calculate based on current contract state
*/
function getPoolRate(address userAddr, uint256 fromBlockNumber, uint256 toBlockNumber) internal view returns (uint256) {
UpdateRewardLaneModel memory vars;
vars.len = registeredPoints.length;
vars.memTotalDeposit = totalDeposited;
vars.tmpLastBlockNum = vars.memLastBlockNum = fromBlockNumber;
(vars.memPassedPoint, vars.memRewardPerBlock, vars.memDecrementUnitPerBlock) = getParamsByBlockNumber(fromBlockNumber);
vars.tmpPassedPoint = vars.memPassedPoint;
vars.tmpRewardPerBlock = vars.memRewardPerBlock;
vars.tmpDecrementUnitPerBlock = vars.memDecrementUnitPerBlock;
vars.memThisBlockNum = toBlockNumber;
vars.tmpRewardLane = vars.memRewardLane = 0;
for(uint256 i=vars.memPassedPoint; i<vars.len; i++) {
RewardVelocityPoint memory point = registeredPoints[i];
if(vars.tmpLastBlockNum < point.blockNumber && point.blockNumber <= vars.memThisBlockNum) {
vars.tmpPassedPoint = i+1;
vars.tmpBlockDelta = safeSub(point.blockNumber, vars.tmpLastBlockNum);
(vars.tmpRewardLane, vars.tmpRewardPerBlock) =
_calcNewRewardLane(
vars.tmpRewardLane,
vars.memTotalDeposit,
vars.tmpRewardPerBlock,
vars.tmpDecrementUnitPerBlock,
vars.tmpBlockDelta);
vars.tmpLastBlockNum = point.blockNumber;
vars.tmpRewardPerBlock = point.rewardPerBlock;
vars.tmpDecrementUnitPerBlock = point.decrementUnitPerBlock;
} else {
break;
}
}
if(vars.memThisBlockNum > vars.tmpLastBlockNum) {
vars.tmpBlockDelta = safeSub(vars.memThisBlockNum, vars.tmpLastBlockNum);
vars.tmpLastBlockNum = vars.memThisBlockNum;
(vars.tmpRewardLane, vars.tmpRewardPerBlock) =
_calcNewRewardLane(
vars.tmpRewardLane,
vars.memTotalDeposit,
vars.tmpRewardPerBlock,
vars.tmpDecrementUnitPerBlock,
vars.tmpBlockDelta);
}
Account memory user;
if( userAddr != address(0) ) user = accounts[userAddr];
else user = Account(vars.memTotalDeposit, 0, 0);
return safeDiv(expMul(user.deposited, vars.tmpRewardLane), safeSub(toBlockNumber, fromBlockNumber));
}
function getParamsByBlockNumber(uint256 _blockNumber) internal view returns (uint256, uint256, uint256) {
uint256 _rewardPerBlock; uint256 _decrement;
uint256 i;
uint256 tmpthisPoint;
uint256 pointLength = registeredPoints.length;
if( pointLength > 0 ) {
for(i = 0; i < pointLength; i++) {
RewardVelocityPoint memory point = registeredPoints[i];
if(_blockNumber >= point.blockNumber && 0 != point.blockNumber) {
tmpthisPoint = i;
_rewardPerBlock = point.rewardPerBlock;
_decrement = point.decrementUnitPerBlock;
} else if( 0 == point.blockNumber ) continue;
else break;
}
}
RewardVelocityPoint memory point = registeredPoints[tmpthisPoint];
_rewardPerBlock = point.rewardPerBlock;
_decrement = point.decrementUnitPerBlock;
if(_blockNumber > point.blockNumber) {
_rewardPerBlock = safeSub(_rewardPerBlock, safeMul(_decrement, safeSub(_blockNumber, point.blockNumber) ) );
}
return (i, _rewardPerBlock, _decrement);
}
function getUserPoolRate(address userAddr, uint256 fromBlockNumber, uint256 toBlockNumber) external view returns (uint256) {
return getPoolRate(userAddr, fromBlockNumber, toBlockNumber);
}
function getModelPoolRate(uint256 fromBlockNumber, uint256 toBlockNumber) external view returns (uint256) {
return getPoolRate(address(0), fromBlockNumber, toBlockNumber);
}
}
// File: contracts/module/externalModule.sol
/**
* @title BiFi's Reward Distribution External Contract
* @notice Implements the service actions.
* @author BiFi(seinmyung25, Miller-kk, tlatkdgus1, dongchangYoo)
*/
contract externalModule is viewModule {
modifier onlyOwner() {
require(msg.sender == owner, "onlyOwner: external function access control!");
_;
}
modifier checkClaimLocked() {
require(!claimLock, "error: claim Locked");
_;
}
modifier checkWithdrawLocked() {
require(!withdrawLock, "error: withdraw Locked");
_;
}
/**
* @notice Set the Deposit-Token address
* @param erc20Addr The address of Deposit Token
*/
function setERC(address erc20Addr) external onlyOwner {
lpErc = ERC20(erc20Addr);
}
/**
* @notice Set the Contribution-Token address
* @param erc20Addr The address of Contribution Token
*/
function setRE(address erc20Addr) external onlyOwner {
rewardErc = ERC20(erc20Addr);
}
/**
* @notice Set the reward distribution parameters instantly
*/
function setParam(uint256 _rewardPerBlock, uint256 _decrementUnitPerBlock) onlyOwner external {
_setParams(_rewardPerBlock, _decrementUnitPerBlock);
}
/**
* @notice Terminate Contract Distribution
*/
function modelFinish(uint256 amount) external onlyOwner {
if( amount != 0) {
require( rewardErc.transfer(owner, amount), "token error" );
}
else {
require( rewardErc.transfer(owner, rewardErc.balanceOf(address(this))), "token error" );
}
delete totalDeposited;
delete rewardPerBlock;
delete decrementUnitPerBlock;
delete rewardLane;
delete totalDeposited;
delete registeredPoints;
}
/**
* @notice Transfer the Remaining Contribution Tokens
*/
function retrieveRewardAmount(uint256 amount) external onlyOwner {
if( amount != 0) {
require( rewardErc.transfer(owner, amount), "token error");
}
else {
require( rewardErc.transfer(owner, rewardErc.balanceOf(address(this))), "token error");
}
}
/**
* @notice Deposit the Contribution Tokens
* @param amount The amount of the Contribution Tokens
*/
function deposit(uint256 amount) external {
address userAddr = msg.sender;
_redeemAll(userAddr);
_deposit(userAddr, amount);
}
/**
* @notice Deposit the Contribution Tokens to target user
* @param userAddr The target user
* @param amount The amount of the Contribution Tokens
*/
function depositTo(address userAddr, uint256 amount) external {
_redeemAll(userAddr);
_deposit(userAddr, amount);
}
/**
* @notice Withdraw the Contribution Tokens
* @param amount The amount of the Contribution Tokens
*/
function withdraw(uint256 amount) checkWithdrawLocked external {
address userAddr = msg.sender;
_redeemAll(userAddr);
_withdraw(userAddr, amount);
}
/**
* @notice Claim the Reward Tokens
* @dev Transfer all reward the user has earned at once.
*/
function rewardClaim() checkClaimLocked external {
address userAddr = msg.sender;
_redeemAll(userAddr);
_rewardClaim(userAddr);
}
/**
* @notice Claim the Reward Tokens
* @param userAddr The targetUser
* @dev Transfer all reward the target user has earned at once.
*/
function rewardClaimTo(address userAddr) checkClaimLocked external {
_redeemAll(userAddr);
_rewardClaim(userAddr);
}
/// @dev Set locks & access control
function setClaimLock(bool lock) onlyOwner external {
_setClaimLock(lock);
}
function setWithdrawLock(bool lock) onlyOwner external {
_setWithdrawLock(lock);
}
function ownershipTransfer(address to) onlyOwner external {
_ownershipTransfer(to);
}
/**
* @notice Register a new future reward velocity point
*/
function registerRewardVelocity(uint256 _blockNumber, uint256 _rewardPerBlock, uint256 _decrementUnitPerBlock) onlyOwner public {
require(_blockNumber > block.number, "new Reward params should register earlier");
require(registeredPoints.length == 0 || _blockNumber > registeredPoints[registeredPoints.length-1].blockNumber, "Earilier velocity points are already set.");
_registerRewardVelocity(_blockNumber, _rewardPerBlock, _decrementUnitPerBlock);
}
function deleteRegisteredRewardVelocity(uint256 _index) onlyOwner external {
require(_index >= passedPoint, "Reward velocity point already passed.");
_deleteRegisteredRewardVelocity(_index);
}
/**
* @notice Set the reward distribution parameters
*/
function setRewardVelocity(uint256 _rewardPerBlock, uint256 _decrementUnitPerBlock) onlyOwner external {
_updateRewardLane();
_setParams(_rewardPerBlock, _decrementUnitPerBlock);
}
}
// File: contracts/DistributionModelV3.sol
/**
* @title BiFi's Reward Distribution Contract
* @notice Implements voting process along with vote delegation
* @author BiFi(seinmyung25, Miller-kk, tlatkdgus1, dongchangYoo)
*/
contract DistributionModelV3 is externalModule {
constructor(address _owner, address _lpErc, address _rewardErc) public {
owner = _owner;
lpErc = ERC20(_lpErc);
rewardErc = ERC20(_rewardErc);
lastBlockNum = block.number;
}
}
contract BFCModel is DistributionModelV3 {
constructor(address _owner, address _lpErc, address _rewardErc, uint256 _start)
DistributionModelV3(_owner, _lpErc, _rewardErc) public {
/*
_start: parameter start block nubmer
0x3935413a1cdd90ff: fixed point(1e18) reward per blocks
0x62e9bea75f: fixed point(1e18) decrement per blocks
*/
_registerRewardVelocity(_start, 0x3935413a1cdd90ff, 0x62e9bea75f);
}
}
contract BFCETHModel is DistributionModelV3 {
constructor(address _owner, address _lpErc, address _rewardErc, uint256 _start)
DistributionModelV3(_owner, _lpErc, _rewardErc) public {
/*
_start: parameter start block nubmer
0xe4d505786b744b3f: fixed point(1e18) reward per blocks
0x18ba6fb966b: fixed point(1e18) decrement per blocks
*/
_registerRewardVelocity(_start, 0xe4d505786b744b3f, 0x18ba6fb966b);
}
}
contract BiFiETHModel is DistributionModelV3 {
constructor(address _owner, address _lpErc, address _rewardErc, uint256 _start)
DistributionModelV3(_owner, _lpErc, _rewardErc) public {
/*
_start: parameter start block nubmer
0x11e0a46e285a68955: fixed point(1e18) reward per blocks
0x1ee90ba90c4: fixed point(1e18) decrement per blocks
*/
_registerRewardVelocity(_start, 0x11e0a46e285a68955, 0x1ee90ba90c4);
}
}
{
"compilationTarget": {
"BiFiETHModel.sol": "BiFiETHModel"
},
"evmVersion": "istanbul",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": []
}
[{"inputs":[{"internalType":"address","name":"_owner","type":"address"},{"internalType":"address","name":"_lpErc","type":"address"},{"internalType":"address","name":"_rewardErc","type":"address"},{"internalType":"uint256","name":"_start","type":"uint256"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"userAddr","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Claim","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"bool","name":"lock","type":"bool"}],"name":"ClaimLock","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"index","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"atBlockNumber","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"rewardPerBlock","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"decrementUnitPerBlock","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"arrayLen","type":"uint256"}],"name":"DeleteRegisterRewardParams","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"userAddr","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"userDeposit","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"totalDeposit","type":"uint256"}],"name":"Deposit","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"from","type":"address"},{"indexed":false,"internalType":"address","name":"to","type":"address"}],"name":"OwnershipTransfer","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"atBlockNumber","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"rewardPerBlock","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"decrementUnitPerBlock","type":"uint256"}],"name":"RegisterRewardParams","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"rewardPerBlock","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"decrementUnitPerBlock","type":"uint256"}],"name":"SetRewardParams","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"atBlockNumber","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"rewardPerBlock","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"decrementUnitPerBlock","type":"uint256"}],"name":"UpdateRewardParams","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"userAddr","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"userDeposit","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"totalDeposit","type":"uint256"}],"name":"Withdraw","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"bool","name":"lock","type":"bool"}],"name":"WithdrawLock","type":"event"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"accounts","outputs":[{"internalType":"uint256","name":"deposited","type":"uint256"},{"internalType":"uint256","name":"pointOnLane","type":"uint256"},{"internalType":"uint256","name":"rewardAmount","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"claimLock","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"decrementUnitPerBlock","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_index","type":"uint256"}],"name":"deleteRegisteredRewardVelocity","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"deposit","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"userAddr","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"depositTo","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"distributionInfo","outputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"fromBlockNumber","type":"uint256"},{"internalType":"uint256","name":"toBlockNumber","type":"uint256"}],"name":"getModelPoolRate","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getParams","outputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"index","type":"uint256"}],"name":"getRegisteredPoint","outputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getRegisteredPointLength","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"userAddr","type":"address"},{"internalType":"uint256","name":"fromBlockNumber","type":"uint256"},{"internalType":"uint256","name":"toBlockNumber","type":"uint256"}],"name":"getUserPoolRate","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"lastBlockNum","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"lpErc","outputs":[{"internalType":"contract ERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_fromBlockNumber","type":"uint256"},{"internalType":"uint256","name":"_toBlockNumber","type":"uint256"}],"name":"marketInformation","outputs":[{"internalType":"uint256","name":"rewardStartBlockNumber","type":"uint256"},{"internalType":"uint256","name":"distributedAmount","type":"uint256"},{"internalType":"uint256","name":"totalDeposit","type":"uint256"},{"internalType":"uint256","name":"poolRate","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"modelFinish","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"modelInfo","outputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"}],"name":"ownershipTransfer","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"passedPoint","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_blockNumber","type":"uint256"},{"internalType":"uint256","name":"_rewardPerBlock","type":"uint256"},{"internalType":"uint256","name":"_decrementUnitPerBlock","type":"uint256"}],"name":"registerRewardVelocity","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"registeredPoints","outputs":[{"internalType":"uint256","name":"blockNumber","type":"uint256"},{"internalType":"uint256","name":"rewardPerBlock","type":"uint256"},{"internalType":"uint256","name":"decrementUnitPerBlock","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"retrieveRewardAmount","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"rewardClaim","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"userAddr","type":"address"}],"name":"rewardClaimTo","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"rewardErc","outputs":[{"internalType":"contract ERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"rewardLane","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"rewardPerBlock","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bool","name":"lock","type":"bool"}],"name":"setClaimLock","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"erc20Addr","type":"address"}],"name":"setERC","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_rewardPerBlock","type":"uint256"},{"internalType":"uint256","name":"_decrementUnitPerBlock","type":"uint256"}],"name":"setParam","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"erc20Addr","type":"address"}],"name":"setRE","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_rewardPerBlock","type":"uint256"},{"internalType":"uint256","name":"_decrementUnitPerBlock","type":"uint256"}],"name":"setRewardVelocity","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"lock","type":"bool"}],"name":"setWithdrawLock","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"totalDeposited","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"userAddr","type":"address"}],"name":"userInfo","outputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"userAddr","type":"address"},{"internalType":"uint256","name":"_fromBlockNumber","type":"uint256"},{"internalType":"uint256","name":"_toBlockNumber","type":"uint256"}],"name":"userInformation","outputs":[{"internalType":"uint256","name":"stakedTokenAmount","type":"uint256"},{"internalType":"uint256","name":"rewardStartBlockNumber","type":"uint256"},{"internalType":"uint256","name":"claimStartBlockNumber","type":"uint256"},{"internalType":"uint256","name":"earnedTokenAmount","type":"uint256"},{"internalType":"uint256","name":"poolRate","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"withdrawLock","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"}]