编译器
0.8.19+commit.7dd6d404
文件 1 的 15:Address.sol
pragma solidity ^0.8.1;
library Address {
function isContract(address account) internal view returns (bool) {
return account.code.length > 0;
}
function sendValue(address payable recipient, uint256 amount) internal {
require(address(this).balance >= amount, "Address: insufficient balance");
(bool success, ) = recipient.call{value: amount}("");
require(success, "Address: unable to send value, recipient may have reverted");
}
function functionCall(address target, bytes memory data) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, "Address: low-level call failed");
}
function functionCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, errorMessage);
}
function functionCallWithValue(
address target,
bytes memory data,
uint256 value
) internal returns (bytes memory) {
return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
}
function functionCallWithValue(
address target,
bytes memory data,
uint256 value,
string memory errorMessage
) internal returns (bytes memory) {
require(address(this).balance >= value, "Address: insufficient balance for call");
(bool success, bytes memory returndata) = target.call{value: value}(data);
return verifyCallResultFromTarget(target, success, returndata, errorMessage);
}
function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
return functionStaticCall(target, data, "Address: low-level static call failed");
}
function functionStaticCall(
address target,
bytes memory data,
string memory errorMessage
) internal view returns (bytes memory) {
(bool success, bytes memory returndata) = target.staticcall(data);
return verifyCallResultFromTarget(target, success, returndata, errorMessage);
}
function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
return functionDelegateCall(target, data, "Address: low-level delegate call failed");
}
function functionDelegateCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
(bool success, bytes memory returndata) = target.delegatecall(data);
return verifyCallResultFromTarget(target, success, returndata, errorMessage);
}
function verifyCallResultFromTarget(
address target,
bool success,
bytes memory returndata,
string memory errorMessage
) internal view returns (bytes memory) {
if (success) {
if (returndata.length == 0) {
require(isContract(target), "Address: call to non-contract");
}
return returndata;
} else {
_revert(returndata, errorMessage);
}
}
function verifyCallResult(
bool success,
bytes memory returndata,
string memory errorMessage
) internal pure returns (bytes memory) {
if (success) {
return returndata;
} else {
_revert(returndata, errorMessage);
}
}
function _revert(bytes memory returndata, string memory errorMessage) private pure {
if (returndata.length > 0) {
assembly {
let returndata_size := mload(returndata)
revert(add(32, returndata), returndata_size)
}
} else {
revert(errorMessage);
}
}
}
文件 2 的 15:Context.sol
pragma solidity ^0.8.0;
abstract contract Context {
function _msgSender() internal view virtual returns (address) {
return msg.sender;
}
function _msgData() internal view virtual returns (bytes calldata) {
return msg.data;
}
}
文件 3 的 15:ERC20.sol
pragma solidity ^0.8.0;
import "./IERC20.sol";
import "./extensions/IERC20Metadata.sol";
import "../../utils/Context.sol";
contract ERC20 is Context, IERC20, IERC20Metadata {
mapping(address => uint256) private _balances;
mapping(address => mapping(address => uint256)) private _allowances;
uint256 private _totalSupply;
string private _name;
string private _symbol;
constructor(string memory name_, string memory symbol_) {
_name = name_;
_symbol = symbol_;
}
function name() public view virtual override returns (string memory) {
return _name;
}
function symbol() public view virtual override returns (string memory) {
return _symbol;
}
function decimals() public view virtual override returns (uint8) {
return 18;
}
function totalSupply() public view virtual override returns (uint256) {
return _totalSupply;
}
function balanceOf(address account) public view virtual override returns (uint256) {
return _balances[account];
}
function transfer(address to, uint256 amount) public virtual override returns (bool) {
address owner = _msgSender();
_transfer(owner, to, amount);
return true;
}
function allowance(address owner, address spender) public view virtual override returns (uint256) {
return _allowances[owner][spender];
}
function approve(address spender, uint256 amount) public virtual override returns (bool) {
address owner = _msgSender();
_approve(owner, spender, amount);
return true;
}
function transferFrom(
address from,
address to,
uint256 amount
) public virtual override returns (bool) {
address spender = _msgSender();
_spendAllowance(from, spender, amount);
_transfer(from, to, amount);
return true;
}
function increaseAllowance(address spender, uint256 addedValue) public virtual returns (bool) {
address owner = _msgSender();
_approve(owner, spender, allowance(owner, spender) + addedValue);
return true;
}
function decreaseAllowance(address spender, uint256 subtractedValue) public virtual returns (bool) {
address owner = _msgSender();
uint256 currentAllowance = allowance(owner, spender);
require(currentAllowance >= subtractedValue, "ERC20: decreased allowance below zero");
unchecked {
_approve(owner, spender, currentAllowance - subtractedValue);
}
return true;
}
function _transfer(
address from,
address to,
uint256 amount
) internal virtual {
require(from != address(0), "ERC20: transfer from the zero address");
require(to != address(0), "ERC20: transfer to the zero address");
_beforeTokenTransfer(from, to, amount);
uint256 fromBalance = _balances[from];
require(fromBalance >= amount, "ERC20: transfer amount exceeds balance");
unchecked {
_balances[from] = fromBalance - amount;
_balances[to] += amount;
}
emit Transfer(from, to, amount);
_afterTokenTransfer(from, to, amount);
}
function _mint(address account, uint256 amount) internal virtual {
require(account != address(0), "ERC20: mint to the zero address");
_beforeTokenTransfer(address(0), account, amount);
_totalSupply += amount;
unchecked {
_balances[account] += amount;
}
emit Transfer(address(0), account, amount);
_afterTokenTransfer(address(0), account, amount);
}
function _burn(address account, uint256 amount) internal virtual {
require(account != address(0), "ERC20: burn from the zero address");
_beforeTokenTransfer(account, address(0), amount);
uint256 accountBalance = _balances[account];
require(accountBalance >= amount, "ERC20: burn amount exceeds balance");
unchecked {
_balances[account] = accountBalance - amount;
_totalSupply -= amount;
}
emit Transfer(account, address(0), amount);
_afterTokenTransfer(account, address(0), amount);
}
function _approve(
address owner,
address spender,
uint256 amount
) internal virtual {
require(owner != address(0), "ERC20: approve from the zero address");
require(spender != address(0), "ERC20: approve to the zero address");
_allowances[owner][spender] = amount;
emit Approval(owner, spender, amount);
}
function _spendAllowance(
address owner,
address spender,
uint256 amount
) internal virtual {
uint256 currentAllowance = allowance(owner, spender);
if (currentAllowance != type(uint256).max) {
require(currentAllowance >= amount, "ERC20: insufficient allowance");
unchecked {
_approve(owner, spender, currentAllowance - amount);
}
}
}
function _beforeTokenTransfer(
address from,
address to,
uint256 amount
) internal virtual {}
function _afterTokenTransfer(
address from,
address to,
uint256 amount
) internal virtual {}
}
文件 4 的 15:IERC20.sol
pragma solidity ^0.8.0;
interface IERC20 {
event Transfer(address indexed from, address indexed to, uint256 value);
event Approval(address indexed owner, address indexed spender, uint256 value);
function totalSupply() external view returns (uint256);
function balanceOf(address account) external view returns (uint256);
function transfer(address to, uint256 amount) external returns (bool);
function allowance(address owner, address spender) external view returns (uint256);
function approve(address spender, uint256 amount) external returns (bool);
function transferFrom(
address from,
address to,
uint256 amount
) external returns (bool);
}
文件 5 的 15:IERC20Metadata.sol
pragma solidity ^0.8.0;
import "../IERC20.sol";
interface IERC20Metadata is IERC20 {
function name() external view returns (string memory);
function symbol() external view returns (string memory);
function decimals() external view returns (uint8);
}
文件 6 的 15:IFullProtec.sol
pragma solidity >=0.6.2;
interface IFullProtec {
function getPercentSupplyStaked() external view returns (uint256);
function setChef(address) external;
}
文件 7 的 15:IScramble.sol
pragma solidity 0.8.19;
import "../interfaces/IUniswapV2Pair.sol";
interface IScramble {
function mint(address to, uint256 amount) external;
function balanceOf(address account) external view returns (uint256);
function totalSupply() external view returns (uint256);
function transferUnderlying(address to, uint256 value) external returns (bool);
function fragmentToScramble(uint256 value) external view returns (uint256);
function scrambleToFragment(uint256 scramble) external view returns (uint256);
function balanceOfUnderlying(address who) external view returns (uint256);
function burn(uint256 amount) external;
function grantRole(bytes32 role, address account) external;
function revokeRole(bytes32 role, address account) external;
function INIT_SUPPLY() external view returns (uint);
function MINTER_ROLE() external view returns (bytes32);
function REBASER_ROLE() external view returns (bytes32);
function setPair(address _router, bool _bool) external;
function setFees(uint256 _fees) external;
function setMarketingAddress(address _marketing) external;
function approve(address spender, uint256 amount) external returns (bool);
function transfer(address, uint) external returns (bool);
function rebase(
uint256 epoch,
uint256 indexDelta,
bool positive
) external returns (uint256);
function setExcludedFromReflections(address, bool) external;
function uniswapV2Pair() external returns (IUniswapV2Pair);
function owner() external returns (address);
function reflectionsReceiver() external returns (address);
function tradingOpen() external returns (bool);
function setMaxWallet(uint) external;
function openTrading() external;
function manualSwap() external;
function transferFrom(address from, address to, uint256 amount) external returns (bool);
}
文件 8 的 15:IUniswapV2Pair.sol
pragma solidity >=0.5.0;
interface IUniswapV2Pair {
event Approval(address indexed owner, address indexed spender, uint value);
event Transfer(address indexed from, address indexed to, uint value);
function name() external pure returns (string memory);
function symbol() external pure returns (string memory);
function decimals() external pure returns (uint8);
function totalSupply() external view returns (uint);
function balanceOf(address owner) external view returns (uint);
function allowance(address owner, address spender) external view returns (uint);
function approve(address spender, uint value) external returns (bool);
function transfer(address to, uint value) external returns (bool);
function transferFrom(address from, address to, uint value) external returns (bool);
function DOMAIN_SEPARATOR() external view returns (bytes32);
function PERMIT_TYPEHASH() external pure returns (bytes32);
function nonces(address owner) external view returns (uint);
function permit(address owner, address spender, uint value, uint deadline, uint8 v, bytes32 r, bytes32 s) external;
event Mint(address indexed sender, uint amount0, uint amount1);
event Burn(address indexed sender, uint amount0, uint amount1, address indexed to);
event Swap(
address indexed sender,
uint amount0In,
uint amount1In,
uint amount0Out,
uint amount1Out,
address indexed to
);
event Sync(uint112 reserve0, uint112 reserve1);
function MINIMUM_LIQUIDITY() external pure returns (uint);
function factory() external view returns (address);
function token0() external view returns (address);
function token1() external view returns (address);
function getReserves() external view returns (uint112 reserve0, uint112 reserve1, uint32 blockTimestampLast);
function price0CumulativeLast() external view returns (uint);
function price1CumulativeLast() external view returns (uint);
function kLast() external view returns (uint);
function mint(address to) external returns (uint liquidity);
function burn(address to) external returns (uint amount0, uint amount1);
function swap(uint amount0Out, uint amount1Out, address to, bytes calldata data) external;
function skim(address to) external;
function sync() external;
function initialize(address, address) external;
}
文件 9 的 15:IUniswapV2Router01.sol
pragma solidity >=0.6.2;
interface IUniswapV2Router01 {
function factory() external pure returns (address);
function WETH() external pure returns (address);
function addLiquidity(
address tokenA,
address tokenB,
uint amountADesired,
uint amountBDesired,
uint amountAMin,
uint amountBMin,
address to,
uint deadline
) external returns (uint amountA, uint amountB, uint liquidity);
function addLiquidityETH(
address token,
uint amountTokenDesired,
uint amountTokenMin,
uint amountETHMin,
address to,
uint deadline
) external payable returns (uint amountToken, uint amountETH, uint liquidity);
function removeLiquidity(
address tokenA,
address tokenB,
uint liquidity,
uint amountAMin,
uint amountBMin,
address to,
uint deadline
) external returns (uint amountA, uint amountB);
function removeLiquidityETH(
address token,
uint liquidity,
uint amountTokenMin,
uint amountETHMin,
address to,
uint deadline
) external returns (uint amountToken, uint amountETH);
function removeLiquidityWithPermit(
address tokenA,
address tokenB,
uint liquidity,
uint amountAMin,
uint amountBMin,
address to,
uint deadline,
bool approveMax, uint8 v, bytes32 r, bytes32 s
) external returns (uint amountA, uint amountB);
function removeLiquidityETHWithPermit(
address token,
uint liquidity,
uint amountTokenMin,
uint amountETHMin,
address to,
uint deadline,
bool approveMax, uint8 v, bytes32 r, bytes32 s
) external returns (uint amountToken, uint amountETH);
function swapExactTokensForTokens(
uint amountIn,
uint amountOutMin,
address[] calldata path,
address to,
uint deadline
) external returns (uint[] memory amounts);
function swapTokensForExactTokens(
uint amountOut,
uint amountInMax,
address[] calldata path,
address to,
uint deadline
) external returns (uint[] memory amounts);
function swapExactETHForTokens(uint amountOutMin, address[] calldata path, address to, uint deadline)
external
payable
returns (uint[] memory amounts);
function swapTokensForExactETH(uint amountOut, uint amountInMax, address[] calldata path, address to, uint deadline)
external
returns (uint[] memory amounts);
function swapExactTokensForETH(uint amountIn, uint amountOutMin, address[] calldata path, address to, uint deadline)
external
returns (uint[] memory amounts);
function swapETHForExactTokens(uint amountOut, address[] calldata path, address to, uint deadline)
external
payable
returns (uint[] memory amounts);
function quote(uint amountA, uint reserveA, uint reserveB) external pure returns (uint amountB);
function getAmountOut(uint amountIn, uint reserveIn, uint reserveOut) external pure returns (uint amountOut);
function getAmountIn(uint amountOut, uint reserveIn, uint reserveOut) external pure returns (uint amountIn);
function getAmountsOut(uint amountIn, address[] calldata path) external view returns (uint[] memory amounts);
function getAmountsIn(uint amountOut, address[] calldata path) external view returns (uint[] memory amounts);
}
文件 10 的 15:IUniswapV2Router02.sol
pragma solidity >=0.6.2;
import './IUniswapV2Router01.sol';
interface IUniswapV2Router02 is IUniswapV2Router01 {
function removeLiquidityETHSupportingFeeOnTransferTokens(
address token,
uint liquidity,
uint amountTokenMin,
uint amountETHMin,
address to,
uint deadline
) external returns (uint amountETH);
function removeLiquidityETHWithPermitSupportingFeeOnTransferTokens(
address token,
uint liquidity,
uint amountTokenMin,
uint amountETHMin,
address to,
uint deadline,
bool approveMax, uint8 v, bytes32 r, bytes32 s
) external returns (uint amountETH);
function swapExactTokensForTokensSupportingFeeOnTransferTokens(
uint amountIn,
uint amountOutMin,
address[] calldata path,
address to,
uint deadline
) external;
function swapExactETHForTokensSupportingFeeOnTransferTokens(
uint amountOutMin,
address[] calldata path,
address to,
uint deadline
) external payable;
function swapExactTokensForETHSupportingFeeOnTransferTokens(
uint amountIn,
uint amountOutMin,
address[] calldata path,
address to,
uint deadline
) external;
}
文件 11 的 15:IWhite.sol
pragma solidity 0.8.19;
interface IWhite {
function mint(address to, uint256 amount) external;
function balanceOf(address account) external view returns (uint256);
function totalSupply() external view returns (uint256);
function burn(uint256 amount) external;
function approve(address spender, uint256 amount) external returns (bool);
function transfer(address, uint) external returns (bool);
function transferOwnership(address) external;
}
文件 12 的 15:Ownable.sol
pragma solidity ^0.8.0;
import "../utils/Context.sol";
abstract contract Ownable is Context {
address private _owner;
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
constructor() {
_transferOwnership(_msgSender());
}
modifier onlyOwner() {
_checkOwner();
_;
}
function owner() public view virtual returns (address) {
return _owner;
}
function _checkOwner() internal view virtual {
require(owner() == _msgSender(), "Ownable: caller is not the owner");
}
function renounceOwnership() public virtual onlyOwner {
_transferOwnership(address(0));
}
function transferOwnership(address newOwner) public virtual onlyOwner {
require(newOwner != address(0), "Ownable: new owner is the zero address");
_transferOwnership(newOwner);
}
function _transferOwnership(address newOwner) internal virtual {
address oldOwner = _owner;
_owner = newOwner;
emit OwnershipTransferred(oldOwner, newOwner);
}
}
文件 13 的 15:SafeERC20.sol
pragma solidity ^0.8.0;
import "../IERC20.sol";
import "../extensions/draft-IERC20Permit.sol";
import "../../../utils/Address.sol";
library SafeERC20 {
using Address for address;
function safeTransfer(
IERC20 token,
address to,
uint256 value
) internal {
_callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value));
}
function safeTransferFrom(
IERC20 token,
address from,
address to,
uint256 value
) internal {
_callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value));
}
function safeApprove(
IERC20 token,
address spender,
uint256 value
) internal {
require(
(value == 0) || (token.allowance(address(this), spender) == 0),
"SafeERC20: approve from non-zero to non-zero allowance"
);
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value));
}
function safeIncreaseAllowance(
IERC20 token,
address spender,
uint256 value
) internal {
uint256 newAllowance = token.allowance(address(this), spender) + value;
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance));
}
function safeDecreaseAllowance(
IERC20 token,
address spender,
uint256 value
) internal {
unchecked {
uint256 oldAllowance = token.allowance(address(this), spender);
require(oldAllowance >= value, "SafeERC20: decreased allowance below zero");
uint256 newAllowance = oldAllowance - value;
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance));
}
}
function safePermit(
IERC20Permit token,
address owner,
address spender,
uint256 value,
uint256 deadline,
uint8 v,
bytes32 r,
bytes32 s
) internal {
uint256 nonceBefore = token.nonces(owner);
token.permit(owner, spender, value, deadline, v, r, s);
uint256 nonceAfter = token.nonces(owner);
require(nonceAfter == nonceBefore + 1, "SafeERC20: permit did not succeed");
}
function _callOptionalReturn(IERC20 token, bytes memory data) private {
bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed");
if (returndata.length > 0) {
require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
}
}
}
文件 14 的 15:ScrambleChef.sol
import "@openzeppelin/contracts/access/Ownable.sol";
import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol";
import "../interfaces/IUniswapV2Router02.sol";
import "../interfaces/IScramble.sol";
import "../interfaces/IFullProtec.sol";
import "../interfaces/IWhite.sol";
import "interfaces/IUniswapV2Pair.sol";
pragma solidity 0.8.19;
library ABDKMath64x64 {
int128 private constant MIN_64x64 = -0x80000000000000000000000000000000;
int128 private constant MAX_64x64 = 0x7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF;
function fromInt(int256 x) internal pure returns (int128) {
unchecked {
require(x >= -0x8000000000000000 && x <= 0x7FFFFFFFFFFFFFFF);
return int128(x << 64);
}
}
function toInt(int128 x) internal pure returns (int64) {
unchecked {
return int64(x >> 64);
}
}
function fromUInt(uint256 x) internal pure returns (int128) {
unchecked {
require(x <= 0x7FFFFFFFFFFFFFFF);
return int128(int256(x << 64));
}
}
function toUInt(int128 x) internal pure returns (uint64) {
unchecked {
require(x >= 0);
return uint64(uint128(x >> 64));
}
}
function from128x128(int256 x) internal pure returns (int128) {
unchecked {
int256 result = x >> 64;
require(result >= MIN_64x64 && result <= MAX_64x64);
return int128(result);
}
}
function to128x128(int128 x) internal pure returns (int256) {
unchecked {
return int256(x) << 64;
}
}
function add(int128 x, int128 y) internal pure returns (int128) {
unchecked {
int256 result = int256(x) + y;
require(result >= MIN_64x64 && result <= MAX_64x64);
return int128(result);
}
}
function sub(int128 x, int128 y) internal pure returns (int128) {
unchecked {
int256 result = int256(x) - y;
require(result >= MIN_64x64 && result <= MAX_64x64);
return int128(result);
}
}
function mul(int128 x, int128 y) internal pure returns (int128) {
unchecked {
int256 result = (int256(x) * y) >> 64;
require(result >= MIN_64x64 && result <= MAX_64x64);
return int128(result);
}
}
function muli(int128 x, int256 y) internal pure returns (int256) {
unchecked {
if (x == MIN_64x64) {
require(
y >= -0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF
&& y <= 0x1000000000000000000000000000000000000000000000000
);
return -y << 63;
} else {
bool negativeResult = false;
if (x < 0) {
x = -x;
negativeResult = true;
}
if (y < 0) {
y = -y;
negativeResult = !negativeResult;
}
uint256 absoluteResult = mulu(x, uint256(y));
if (negativeResult) {
require(absoluteResult <= 0x8000000000000000000000000000000000000000000000000000000000000000);
return -int256(absoluteResult);
} else {
require(absoluteResult <= 0x7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF);
return int256(absoluteResult);
}
}
}
}
function mulu(int128 x, uint256 y) internal pure returns (uint256) {
unchecked {
if (y == 0) return 0;
require(x >= 0);
uint256 lo = (uint256(int256(x)) * (y & 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF)) >> 64;
uint256 hi = uint256(int256(x)) * (y >> 128);
require(hi <= 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF);
hi <<= 64;
require(hi <= 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF - lo);
return hi + lo;
}
}
function div(int128 x, int128 y) internal pure returns (int128) {
unchecked {
require(y != 0);
int256 result = (int256(x) << 64) / y;
require(result >= MIN_64x64 && result <= MAX_64x64);
return int128(result);
}
}
function divi(int256 x, int256 y) internal pure returns (int128) {
unchecked {
require(y != 0);
bool negativeResult = false;
if (x < 0) {
x = -x;
negativeResult = true;
}
if (y < 0) {
y = -y;
negativeResult = !negativeResult;
}
uint128 absoluteResult = divuu(uint256(x), uint256(y));
if (negativeResult) {
require(absoluteResult <= 0x80000000000000000000000000000000);
return -int128(absoluteResult);
} else {
require(absoluteResult <= 0x7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF);
return int128(absoluteResult);
}
}
}
function divu(uint256 x, uint256 y) internal pure returns (int128) {
unchecked {
require(y != 0);
uint128 result = divuu(x, y);
require(result <= uint128(MAX_64x64));
return int128(result);
}
}
function neg(int128 x) internal pure returns (int128) {
unchecked {
require(x != MIN_64x64);
return -x;
}
}
function abs(int128 x) internal pure returns (int128) {
unchecked {
require(x != MIN_64x64);
return x < 0 ? -x : x;
}
}
function inv(int128 x) internal pure returns (int128) {
unchecked {
require(x != 0);
int256 result = int256(0x100000000000000000000000000000000) / x;
require(result >= MIN_64x64 && result <= MAX_64x64);
return int128(result);
}
}
function avg(int128 x, int128 y) internal pure returns (int128) {
unchecked {
return int128((int256(x) + int256(y)) >> 1);
}
}
function gavg(int128 x, int128 y) internal pure returns (int128) {
unchecked {
int256 m = int256(x) * int256(y);
require(m >= 0);
require(m < 0x4000000000000000000000000000000000000000000000000000000000000000);
return int128(sqrtu(uint256(m)));
}
}
function pow(int128 x, uint256 y) internal pure returns (int128) {
unchecked {
bool negative = x < 0 && y & 1 == 1;
uint256 absX = uint128(x < 0 ? -x : x);
uint256 absResult;
absResult = 0x100000000000000000000000000000000;
if (absX <= 0x10000000000000000) {
absX <<= 63;
while (y != 0) {
if (y & 0x1 != 0) {
absResult = (absResult * absX) >> 127;
}
absX = (absX * absX) >> 127;
if (y & 0x2 != 0) {
absResult = (absResult * absX) >> 127;
}
absX = (absX * absX) >> 127;
if (y & 0x4 != 0) {
absResult = (absResult * absX) >> 127;
}
absX = (absX * absX) >> 127;
if (y & 0x8 != 0) {
absResult = (absResult * absX) >> 127;
}
absX = (absX * absX) >> 127;
y >>= 4;
}
absResult >>= 64;
} else {
uint256 absXShift = 63;
if (absX < 0x1000000000000000000000000) {
absX <<= 32;
absXShift -= 32;
}
if (absX < 0x10000000000000000000000000000) {
absX <<= 16;
absXShift -= 16;
}
if (absX < 0x1000000000000000000000000000000) {
absX <<= 8;
absXShift -= 8;
}
if (absX < 0x10000000000000000000000000000000) {
absX <<= 4;
absXShift -= 4;
}
if (absX < 0x40000000000000000000000000000000) {
absX <<= 2;
absXShift -= 2;
}
if (absX < 0x80000000000000000000000000000000) {
absX <<= 1;
absXShift -= 1;
}
uint256 resultShift = 0;
while (y != 0) {
require(absXShift < 64);
if (y & 0x1 != 0) {
absResult = (absResult * absX) >> 127;
resultShift += absXShift;
if (absResult > 0x100000000000000000000000000000000) {
absResult >>= 1;
resultShift += 1;
}
}
absX = (absX * absX) >> 127;
absXShift <<= 1;
if (absX >= 0x100000000000000000000000000000000) {
absX >>= 1;
absXShift += 1;
}
y >>= 1;
}
require(resultShift < 64);
absResult >>= 64 - resultShift;
}
int256 result = negative ? -int256(absResult) : int256(absResult);
require(result >= MIN_64x64 && result <= MAX_64x64);
return int128(result);
}
}
function sqrt(int128 x) internal pure returns (int128) {
unchecked {
require(x >= 0);
return int128(sqrtu(uint256(int256(x)) << 64));
}
}
function log_2(int128 x) internal pure returns (int128) {
unchecked {
require(x > 0);
int256 msb = 0;
int256 xc = x;
if (xc >= 0x10000000000000000) {
xc >>= 64;
msb += 64;
}
if (xc >= 0x100000000) {
xc >>= 32;
msb += 32;
}
if (xc >= 0x10000) {
xc >>= 16;
msb += 16;
}
if (xc >= 0x100) {
xc >>= 8;
msb += 8;
}
if (xc >= 0x10) {
xc >>= 4;
msb += 4;
}
if (xc >= 0x4) {
xc >>= 2;
msb += 2;
}
if (xc >= 0x2) msb += 1;
int256 result = (msb - 64) << 64;
uint256 ux = uint256(int256(x)) << uint256(127 - msb);
for (int256 bit = 0x8000000000000000; bit > 0; bit >>= 1) {
ux *= ux;
uint256 b = ux >> 255;
ux >>= 127 + b;
result += bit * int256(b);
}
return int128(result);
}
}
function ln(int128 x) internal pure returns (int128) {
unchecked {
require(x > 0);
return int128(int256((uint256(int256(log_2(x))) * 0xB17217F7D1CF79ABC9E3B39803F2F6AF) >> 128));
}
}
function exp_2(int128 x) internal pure returns (int128) {
unchecked {
require(x < 0x400000000000000000);
if (x < -0x400000000000000000) return 0;
uint256 result = 0x80000000000000000000000000000000;
if (x & 0x8000000000000000 > 0) {
result = (result * 0x16A09E667F3BCC908B2FB1366EA957D3E) >> 128;
}
if (x & 0x4000000000000000 > 0) {
result = (result * 0x1306FE0A31B7152DE8D5A46305C85EDEC) >> 128;
}
if (x & 0x2000000000000000 > 0) {
result = (result * 0x1172B83C7D517ADCDF7C8C50EB14A791F) >> 128;
}
if (x & 0x1000000000000000 > 0) {
result = (result * 0x10B5586CF9890F6298B92B71842A98363) >> 128;
}
if (x & 0x800000000000000 > 0) {
result = (result * 0x1059B0D31585743AE7C548EB68CA417FD) >> 128;
}
if (x & 0x400000000000000 > 0) {
result = (result * 0x102C9A3E778060EE6F7CACA4F7A29BDE8) >> 128;
}
if (x & 0x200000000000000 > 0) {
result = (result * 0x10163DA9FB33356D84A66AE336DCDFA3F) >> 128;
}
if (x & 0x100000000000000 > 0) {
result = (result * 0x100B1AFA5ABCBED6129AB13EC11DC9543) >> 128;
}
if (x & 0x80000000000000 > 0) {
result = (result * 0x10058C86DA1C09EA1FF19D294CF2F679B) >> 128;
}
if (x & 0x40000000000000 > 0) {
result = (result * 0x1002C605E2E8CEC506D21BFC89A23A00F) >> 128;
}
if (x & 0x20000000000000 > 0) {
result = (result * 0x100162F3904051FA128BCA9C55C31E5DF) >> 128;
}
if (x & 0x10000000000000 > 0) {
result = (result * 0x1000B175EFFDC76BA38E31671CA939725) >> 128;
}
if (x & 0x8000000000000 > 0) {
result = (result * 0x100058BA01FB9F96D6CACD4B180917C3D) >> 128;
}
if (x & 0x4000000000000 > 0) {
result = (result * 0x10002C5CC37DA9491D0985C348C68E7B3) >> 128;
}
if (x & 0x2000000000000 > 0) {
result = (result * 0x1000162E525EE054754457D5995292026) >> 128;
}
if (x & 0x1000000000000 > 0) {
result = (result * 0x10000B17255775C040618BF4A4ADE83FC) >> 128;
}
if (x & 0x800000000000 > 0) {
result = (result * 0x1000058B91B5BC9AE2EED81E9B7D4CFAB) >> 128;
}
if (x & 0x400000000000 > 0) {
result = (result * 0x100002C5C89D5EC6CA4D7C8ACC017B7C9) >> 128;
}
if (x & 0x200000000000 > 0) {
result = (result * 0x10000162E43F4F831060E02D839A9D16D) >> 128;
}
if (x & 0x100000000000 > 0) {
result = (result * 0x100000B1721BCFC99D9F890EA06911763) >> 128;
}
if (x & 0x80000000000 > 0) {
result = (result * 0x10000058B90CF1E6D97F9CA14DBCC1628) >> 128;
}
if (x & 0x40000000000 > 0) {
result = (result * 0x1000002C5C863B73F016468F6BAC5CA2B) >> 128;
}
if (x & 0x20000000000 > 0) {
result = (result * 0x100000162E430E5A18F6119E3C02282A5) >> 128;
}
if (x & 0x10000000000 > 0) {
result = (result * 0x1000000B1721835514B86E6D96EFD1BFE) >> 128;
}
if (x & 0x8000000000 > 0) {
result = (result * 0x100000058B90C0B48C6BE5DF846C5B2EF) >> 128;
}
if (x & 0x4000000000 > 0) {
result = (result * 0x10000002C5C8601CC6B9E94213C72737A) >> 128;
}
if (x & 0x2000000000 > 0) {
result = (result * 0x1000000162E42FFF037DF38AA2B219F06) >> 128;
}
if (x & 0x1000000000 > 0) {
result = (result * 0x10000000B17217FBA9C739AA5819F44F9) >> 128;
}
if (x & 0x800000000 > 0) {
result = (result * 0x1000000058B90BFCDEE5ACD3C1CEDC823) >> 128;
}
if (x & 0x400000000 > 0) {
result = (result * 0x100000002C5C85FE31F35A6A30DA1BE50) >> 128;
}
if (x & 0x200000000 > 0) {
result = (result * 0x10000000162E42FF0999CE3541B9FFFCF) >> 128;
}
if (x & 0x100000000 > 0) {
result = (result * 0x100000000B17217F80F4EF5AADDA45554) >> 128;
}
if (x & 0x80000000 > 0) {
result = (result * 0x10000000058B90BFBF8479BD5A81B51AD) >> 128;
}
if (x & 0x40000000 > 0) {
result = (result * 0x1000000002C5C85FDF84BD62AE30A74CC) >> 128;
}
if (x & 0x20000000 > 0) {
result = (result * 0x100000000162E42FEFB2FED257559BDAA) >> 128;
}
if (x & 0x10000000 > 0) {
result = (result * 0x1000000000B17217F7D5A7716BBA4A9AE) >> 128;
}
if (x & 0x8000000 > 0) {
result = (result * 0x100000000058B90BFBE9DDBAC5E109CCE) >> 128;
}
if (x & 0x4000000 > 0) {
result = (result * 0x10000000002C5C85FDF4B15DE6F17EB0D) >> 128;
}
if (x & 0x2000000 > 0) {
result = (result * 0x1000000000162E42FEFA494F1478FDE05) >> 128;
}
if (x & 0x1000000 > 0) {
result = (result * 0x10000000000B17217F7D20CF927C8E94C) >> 128;
}
if (x & 0x800000 > 0) {
result = (result * 0x1000000000058B90BFBE8F71CB4E4B33D) >> 128;
}
if (x & 0x400000 > 0) {
result = (result * 0x100000000002C5C85FDF477B662B26945) >> 128;
}
if (x & 0x200000 > 0) {
result = (result * 0x10000000000162E42FEFA3AE53369388C) >> 128;
}
if (x & 0x100000 > 0) {
result = (result * 0x100000000000B17217F7D1D351A389D40) >> 128;
}
if (x & 0x80000 > 0) {
result = (result * 0x10000000000058B90BFBE8E8B2D3D4EDE) >> 128;
}
if (x & 0x40000 > 0) {
result = (result * 0x1000000000002C5C85FDF4741BEA6E77E) >> 128;
}
if (x & 0x20000 > 0) {
result = (result * 0x100000000000162E42FEFA39FE95583C2) >> 128;
}
if (x & 0x10000 > 0) {
result = (result * 0x1000000000000B17217F7D1CFB72B45E1) >> 128;
}
if (x & 0x8000 > 0) {
result = (result * 0x100000000000058B90BFBE8E7CC35C3F0) >> 128;
}
if (x & 0x4000 > 0) {
result = (result * 0x10000000000002C5C85FDF473E242EA38) >> 128;
}
if (x & 0x2000 > 0) {
result = (result * 0x1000000000000162E42FEFA39F02B772C) >> 128;
}
if (x & 0x1000 > 0) {
result = (result * 0x10000000000000B17217F7D1CF7D83C1A) >> 128;
}
if (x & 0x800 > 0) {
result = (result * 0x1000000000000058B90BFBE8E7BDCBE2E) >> 128;
}
if (x & 0x400 > 0) {
result = (result * 0x100000000000002C5C85FDF473DEA871F) >> 128;
}
if (x & 0x200 > 0) {
result = (result * 0x10000000000000162E42FEFA39EF44D91) >> 128;
}
if (x & 0x100 > 0) {
result = (result * 0x100000000000000B17217F7D1CF79E949) >> 128;
}
if (x & 0x80 > 0) {
result = (result * 0x10000000000000058B90BFBE8E7BCE544) >> 128;
}
if (x & 0x40 > 0) {
result = (result * 0x1000000000000002C5C85FDF473DE6ECA) >> 128;
}
if (x & 0x20 > 0) {
result = (result * 0x100000000000000162E42FEFA39EF366F) >> 128;
}
if (x & 0x10 > 0) {
result = (result * 0x1000000000000000B17217F7D1CF79AFA) >> 128;
}
if (x & 0x8 > 0) {
result = (result * 0x100000000000000058B90BFBE8E7BCD6D) >> 128;
}
if (x & 0x4 > 0) {
result = (result * 0x10000000000000002C5C85FDF473DE6B2) >> 128;
}
if (x & 0x2 > 0) {
result = (result * 0x1000000000000000162E42FEFA39EF358) >> 128;
}
if (x & 0x1 > 0) {
result = (result * 0x10000000000000000B17217F7D1CF79AB) >> 128;
}
result >>= uint256(int256(63 - (x >> 64)));
require(result <= uint256(int256(MAX_64x64)));
return int128(int256(result));
}
}
function exp(int128 x) internal pure returns (int128) {
unchecked {
require(x < 0x400000000000000000);
if (x < -0x400000000000000000) return 0;
return exp_2(int128((int256(x) * 0x171547652B82FE1777D0FFDA0D23A7D12) >> 128));
}
}
function divuu(uint256 x, uint256 y) private pure returns (uint128) {
unchecked {
require(y != 0);
uint256 result;
if (x <= 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF) {
result = (x << 64) / y;
} else {
uint256 msb = 192;
uint256 xc = x >> 192;
if (xc >= 0x100000000) {
xc >>= 32;
msb += 32;
}
if (xc >= 0x10000) {
xc >>= 16;
msb += 16;
}
if (xc >= 0x100) {
xc >>= 8;
msb += 8;
}
if (xc >= 0x10) {
xc >>= 4;
msb += 4;
}
if (xc >= 0x4) {
xc >>= 2;
msb += 2;
}
if (xc >= 0x2) msb += 1;
result = (x << (255 - msb)) / (((y - 1) >> (msb - 191)) + 1);
require(result <= 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF);
uint256 hi = result * (y >> 128);
uint256 lo = result * (y & 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF);
uint256 xh = x >> 192;
uint256 xl = x << 64;
if (xl < lo) xh -= 1;
xl -= lo;
lo = hi << 128;
if (xl < lo) xh -= 1;
xl -= lo;
assert(xh == hi >> 128);
result += xl / y;
}
require(result <= 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF);
return uint128(result);
}
}
function sqrtu(uint256 x) private pure returns (uint128) {
unchecked {
if (x == 0) {
return 0;
} else {
uint256 xx = x;
uint256 r = 1;
if (xx >= 0x100000000000000000000000000000000) {
xx >>= 128;
r <<= 64;
}
if (xx >= 0x10000000000000000) {
xx >>= 64;
r <<= 32;
}
if (xx >= 0x100000000) {
xx >>= 32;
r <<= 16;
}
if (xx >= 0x10000) {
xx >>= 16;
r <<= 8;
}
if (xx >= 0x100) {
xx >>= 8;
r <<= 4;
}
if (xx >= 0x10) {
xx >>= 4;
r <<= 2;
}
if (xx >= 0x8) {
r <<= 1;
}
r = (r + x / r) >> 1;
r = (r + x / r) >> 1;
r = (r + x / r) >> 1;
r = (r + x / r) >> 1;
r = (r + x / r) >> 1;
r = (r + x / r) >> 1;
r = (r + x / r) >> 1;
uint256 r1 = x / r;
return uint128(r < r1 ? r : r1);
}
}
}
}
import "@openzeppelin/contracts/token/ERC20/ERC20.sol";
pragma solidity 0.8.19;
contract ScrambleChef is Ownable {
using SafeERC20 for IERC20;
using Address for address;
struct UserInfo {
uint256 amount;
uint256 rewardDebt;
uint256 lockEndedTimestamp;
}
struct PoolInfo {
IERC20 lpToken;
uint256 allocPoint;
uint256 lastRewardTimestamp;
uint256 accRewardPerShare;
}
IScramble public scramble;
IUniswapV2Pair public scrambleLp;
IFullProtec public fullProtec;
IWhite public white;
uint256 public rewardPerSecond;
uint256 public dailyDebaseRateHardCap = 50e18;
uint256 public lastTimestamp;
PoolInfo[] public poolInfo;
mapping(uint256 => mapping(address => UserInfo)) public userInfo;
mapping(uint256 => mapping(address => uint256)) private userRewards;
mapping(uint256 => uint256) public lockDurations;
mapping(address => uint256) public xpAccumulated;
mapping(address => uint256) public lastClaimedBlock;
uint256 public totalAllocPoint = 0;
uint256 public startTimestamp;
mapping(uint256 => uint256) public totalStakedInPool;
event Deposit(address indexed user, uint256 indexed pid, uint256 indexed amount);
event Withdraw(address indexed user, uint256 indexed pid, uint256 indexed amount);
event RewardPaid(address indexed user, uint256 indexed pid, uint256 indexed amount);
event LogRewardPerSecond(uint256 amount);
event LogPoolAddition(uint256 indexed pid, uint256 allocPoint, IERC20 indexed lpToken);
event LogSetPool(uint256 indexed pid, uint256 allocPoint);
event LogUpdatePool(uint256 indexed pid, uint256 lastRewardTimestamp, uint256 lpSupply, uint256 accRewardPerShare);
event LogSetLockDuration(uint256 indexed pid, uint256 lockDuration);
constructor() {
scramble = IScramble(0x63b420fb3294BA1d300CE5D3ba4BBCA0F4fe5e3b);
white = IWhite(0x7a38aFa395666799b3DbFe22C0d1467feC931Bb0);
fullProtec = IFullProtec(0x400aFbc1bBa6E8fF4462D161f7DC24e4873D4eBB);
scrambleLp = IUniswapV2Pair(0xeD7985385bF434F0815AA9C90450945aEE02d733);
rewardPerSecond = 10e18;
startTimestamp = block.timestamp;
lastTimestamp = block.timestamp;
}
function poolLength() external view returns (uint256) {
return poolInfo.length;
}
function setLockDuration(uint256 _pid, uint256 _lockDuration) external onlyOwner {
lockDurations[_pid] = _lockDuration;
emit LogSetLockDuration(_pid, _lockDuration);
}
function updateRewardPerSecond(uint256 _rewardPerSecond) external onlyOwner {
massUpdatePools();
rewardPerSecond = _rewardPerSecond;
emit LogRewardPerSecond(_rewardPerSecond);
}
function add(uint256 _allocPoint, IERC20 _lpToken, bool _withUpdate) external onlyOwner {
if (_withUpdate) {
massUpdatePools();
}
uint256 lastRewardTimestamp = block.timestamp > startTimestamp ? block.timestamp : startTimestamp;
totalAllocPoint = totalAllocPoint + _allocPoint;
poolInfo.push(
PoolInfo({
lpToken: _lpToken,
allocPoint: _allocPoint,
lastRewardTimestamp: lastRewardTimestamp,
accRewardPerShare: 0
})
);
emit LogPoolAddition(poolInfo.length - 1, _allocPoint, _lpToken);
}
function set(uint256 _pid, uint256 _allocPoint, bool _withUpdate) external onlyOwner {
if (_withUpdate) {
massUpdatePools();
}
totalAllocPoint = totalAllocPoint - poolInfo[_pid].allocPoint + _allocPoint;
poolInfo[_pid].allocPoint = _allocPoint;
emit LogSetPool(_pid, _allocPoint);
}
function pendingReward(uint256 _pid, address _user) public view returns (uint256) {
PoolInfo storage pool = poolInfo[_pid];
UserInfo storage user = userInfo[_pid][_user];
uint256 accRewardPerShare = pool.accRewardPerShare;
uint256 lpSupply = pool.lpToken.balanceOf(address(this));
if (address(pool.lpToken) == address(scramble)) {
lpSupply = scramble.balanceOfUnderlying(address(this));
}
if (block.timestamp > pool.lastRewardTimestamp && lpSupply != 0) {
uint256 scrambleReward =
((block.timestamp - pool.lastRewardTimestamp) * rewardPerSecond * pool.allocPoint) / totalAllocPoint;
accRewardPerShare += (scrambleReward * 1e12) / lpSupply;
}
return userRewards[_pid][_user] + (user.amount * accRewardPerShare) / 1e12 - user.rewardDebt;
}
function massUpdatePools() internal {
uint256 length = poolInfo.length;
for (uint256 pid = 0; pid < length; ++pid) {
updatePool(pid);
}
}
function updatePool(uint256 _pid) internal {
PoolInfo storage pool = poolInfo[_pid];
if (block.timestamp <= pool.lastRewardTimestamp) {
return;
}
uint256 lpSupply = pool.lpToken.balanceOf(address(this));
if (lpSupply == 0) {
pool.lastRewardTimestamp = block.timestamp;
return;
}
uint256 scrambleReward =
((block.timestamp - pool.lastRewardTimestamp) * rewardPerSecond * pool.allocPoint) / totalAllocPoint;
pool.accRewardPerShare += (scrambleReward * 1e12) / lpSupply;
pool.lastRewardTimestamp = block.timestamp;
emit LogUpdatePool(_pid, pool.lastRewardTimestamp, lpSupply, pool.accRewardPerShare);
}
function deposit(uint256 _pid, uint256 _amount, address _account) external {
if (_pid == 0) {
require(msg.sender == address(fullProtec), "Not allowed");
} else {
require(
msg.sender == _account || msg.sender == address(this) || msg.sender == address(fullProtec),
"You can't deposit for someone else"
);
}
require(_amount > 0, "Deposit amount can't be zero");
PoolInfo storage pool = poolInfo[_pid];
UserInfo storage user = userInfo[_pid][_account];
user.lockEndedTimestamp = block.timestamp + lockDurations[_pid];
updatePool(_pid);
queueRewards(_pid, _account);
if (address(pool.lpToken) == address(white)) {
pool.lpToken.safeTransferFrom(address(fullProtec), address(this), _amount);
totalStakedInPool[_pid] += _amount;
} else {
pool.lpToken.safeTransferFrom(_account, address(this), _amount);
totalStakedInPool[_pid] += _amount;
}
emit Deposit(_account, _pid, _amount);
user.amount += _amount;
user.rewardDebt = (user.amount * pool.accRewardPerShare) / 1e12;
}
function withdraw(uint256 _pid, uint256 _amount, address _account) external {
if (_pid == 0) {
require(msg.sender == address(fullProtec), "Not allowed");
} else {
require(
msg.sender == _account || msg.sender == address(this) || msg.sender == address(fullProtec),
"You can't withdraw for someone else"
);
}
require(_amount > 0, "Withdraw amount can't be zero");
PoolInfo storage pool = poolInfo[_pid];
UserInfo storage user = userInfo[_pid][_account];
require(user.lockEndedTimestamp <= block.timestamp, "Still locked");
require(user.amount >= _amount, "You can't withdraw that much");
updatePool(_pid);
queueRewards(_pid, _account);
user.amount -= _amount;
user.rewardDebt = (user.amount * pool.accRewardPerShare) / 1e12;
if (address(pool.lpToken) == address(white)) {
pool.lpToken.safeTransfer(address(fullProtec), _amount);
totalStakedInPool[_pid] -= _amount;
} else {
pool.lpToken.safeTransfer(address(_account), _amount);
totalStakedInPool[_pid] -= _amount;
}
emit Withdraw(_account, _pid, _amount);
}
function makeScramble(address _account) external {
require(msg.sender == _account || msg.sender == address(this), "You can't claim for someone else");
uint256 stakedPool0 = userInfo[0][_account].amount;
uint256 stakedPool1 = userInfo[1][_account].amount;
if (stakedPool0 == 0 && stakedPool1 == 0) {
require(stakedPool0 > 0 && stakedPool1 > 0, "Can't make scramble without WHITE or YOLK");
}
if (lastClaimedBlock[_account] == 0) {
lastClaimedBlock[_account] = block.number - 1000;
}
xpAccumulated[_account] += getPendingXp(_account);
xpAccumulated[_account] += getPendingBonusXp(_account);
lastClaimedBlock[_account] = block.number;
uint256 bonusRewards = 0;
if (getPendingBonusRewards(_account) > 0) {
bonusRewards = getPendingBonusRewards(_account);
}
claim(0, _account);
claim(1, _account);
if (bonusRewards > 0) {
scramble.mint(_account, bonusRewards);
}
}
function claim(uint256 _pid, address _account) internal returns (uint256) {
require(msg.sender == _account || msg.sender == address(this), "You can't claim for someone else");
updatePool(_pid);
queueRewards(_pid, _account);
uint256 pending = userRewards[_pid][_account];
if (pending > 0) {
UserInfo storage user = userInfo[_pid][_account];
user.lockEndedTimestamp = block.timestamp + lockDurations[_pid];
userRewards[_pid][_account] = 0;
userInfo[_pid][_account].rewardDebt =
(userInfo[_pid][_account].amount * poolInfo[_pid].accRewardPerShare) / (1e12);
if (lastTimestamp != block.timestamp) {
uint256 secs = block.timestamp - lastTimestamp;
if (block.timestamp - lastTimestamp > 1 days) {
secs = 1 days;
}
lastTimestamp = block.timestamp;
scramble.rebase(block.timestamp, getDebaseRate() * secs, false);
scrambleLp.sync();
}
scramble.mint(_account, pending);
emit RewardPaid(_account, _pid, pending);
return pending;
} else {
return 0;
}
}
function queueRewards(uint256 _pid, address _account) internal {
UserInfo memory user = userInfo[_pid][_account];
uint256 pending = (user.amount * poolInfo[_pid].accRewardPerShare) / (1e12) - user.rewardDebt;
if (pending > 0) {
userRewards[_pid][_account] += pending;
}
}
function getDebaseRate() public view returns (uint256) {
if (fullProtec.getPercentSupplyStaked() >= dailyDebaseRateHardCap) {
return (dailyDebaseRateHardCap * 1e16) / 1e18 / 86400;
} else {
return (fullProtec.getPercentSupplyStaked() * 1e16) / 1e18 / 86400;
}
}
function setDailyDebaseRateHardCap(uint256 _dailyDebaseRateHardCap) public onlyOwner {
dailyDebaseRateHardCap = _dailyDebaseRateHardCap;
}
function setWhiteVaultAddress(address _fullProtec) public onlyOwner {
fullProtec = IFullProtec(_fullProtec);
}
function emergencyWithdraw(address lpToken, uint amount) public onlyOwner {
IERC20(lpToken).transfer(owner(), amount);
}
mapping(address => uint256) public claimMultiplierLevel;
mapping(address => uint256) public xpMultiplierLevel;
function upgradeScrambleMultiplier(address _account) public {
require(msg.sender == _account || msg.sender == address(this), "You can't upgrade for someone else");
uint256 upgradePrice = getClaimMultiplierUpgradePrice(_account);
require(xpAccumulated[_account] >= upgradePrice, "Not enough XP");
xpAccumulated[_account] -= upgradePrice;
claimMultiplierLevel[_account] += 1;
}
function upgradeXpMultiplier(address _account) public {
require(msg.sender == _account || msg.sender == address(this), "You can't upgrade for someone else");
uint256 upgradePrice = getXpMultiplierUpgradePrice(_account);
require(xpAccumulated[_account] >= upgradePrice, "Not enough XP");
xpAccumulated[_account] -= upgradePrice;
xpMultiplierLevel[_account] += 1;
}
function getClaimMultiplierUpgradePrice(address _account) public view returns (uint256) {
return claimMultiplierLevel[_account] * 1000 + 1000;
}
function getXpMultiplierUpgradePrice(address _account) public view returns (uint256) {
return xpMultiplierLevel[_account] * 1000 + 1000;
}
function getPendingBonusRewards(address _account) public view returns (uint256) {
uint256 pending0 = pendingReward(0, _account);
uint256 pending1 = pendingReward(1, _account);
uint256 total = pending0 + pending1;
uint256 bonus = (total * claimMultiplierLevel[_account] * 10) / 100;
return bonus;
}
function getPendingXp(address _account) public view returns (uint256) {
uint256 _lastClaimedBlock;
if (lastClaimedBlock[_account] == 0) {
_lastClaimedBlock = block.number - 1000;
} else {
_lastClaimedBlock = lastClaimedBlock[_account];
}
return block.number - _lastClaimedBlock;
}
function getPendingBonusXp(address _account) public view returns (uint256) {
return (getPendingXp(_account) * (xpMultiplierLevel[_account] * 10)) / 100;
}
}
文件 15 的 15:draft-IERC20Permit.sol
pragma solidity ^0.8.0;
interface IERC20Permit {
function permit(
address owner,
address spender,
uint256 value,
uint256 deadline,
uint8 v,
bytes32 r,
bytes32 s
) external;
function nonces(address owner) external view returns (uint256);
function DOMAIN_SEPARATOR() external view returns (bytes32);
}
{
"compilationTarget": {
"src/ScrambleChef.sol": "ScrambleChef"
},
"evmVersion": "paris",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": [
":@openzeppelin/=lib/openzeppelin-contracts/",
":ds-test/=lib/forge-std/lib/ds-test/src/",
":forge-std/=lib/forge-std/src/",
":openzeppelin-contracts/=lib/openzeppelin-contracts/"
]
}
[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":true,"internalType":"uint256","name":"pid","type":"uint256"},{"indexed":true,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Deposit","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"pid","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"allocPoint","type":"uint256"},{"indexed":true,"internalType":"contract IERC20","name":"lpToken","type":"address"}],"name":"LogPoolAddition","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"LogRewardPerSecond","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"pid","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"lockDuration","type":"uint256"}],"name":"LogSetLockDuration","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"pid","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"allocPoint","type":"uint256"}],"name":"LogSetPool","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"pid","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"lastRewardTimestamp","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"lpSupply","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"accRewardPerShare","type":"uint256"}],"name":"LogUpdatePool","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":true,"internalType":"uint256","name":"pid","type":"uint256"},{"indexed":true,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"RewardPaid","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":true,"internalType":"uint256","name":"pid","type":"uint256"},{"indexed":true,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Withdraw","type":"event"},{"inputs":[{"internalType":"uint256","name":"_allocPoint","type":"uint256"},{"internalType":"contract IERC20","name":"_lpToken","type":"address"},{"internalType":"bool","name":"_withUpdate","type":"bool"}],"name":"add","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"claimMultiplierLevel","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"dailyDebaseRateHardCap","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_pid","type":"uint256"},{"internalType":"uint256","name":"_amount","type":"uint256"},{"internalType":"address","name":"_account","type":"address"}],"name":"deposit","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"lpToken","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"emergencyWithdraw","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"fullProtec","outputs":[{"internalType":"contract IFullProtec","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_account","type":"address"}],"name":"getClaimMultiplierUpgradePrice","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getDebaseRate","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_account","type":"address"}],"name":"getPendingBonusRewards","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_account","type":"address"}],"name":"getPendingBonusXp","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_account","type":"address"}],"name":"getPendingXp","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_account","type":"address"}],"name":"getXpMultiplierUpgradePrice","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"lastClaimedBlock","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"lastTimestamp","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"lockDurations","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_account","type":"address"}],"name":"makeScramble","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_pid","type":"uint256"},{"internalType":"address","name":"_user","type":"address"}],"name":"pendingReward","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"poolInfo","outputs":[{"internalType":"contract IERC20","name":"lpToken","type":"address"},{"internalType":"uint256","name":"allocPoint","type":"uint256"},{"internalType":"uint256","name":"lastRewardTimestamp","type":"uint256"},{"internalType":"uint256","name":"accRewardPerShare","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"poolLength","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"rewardPerSecond","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"scramble","outputs":[{"internalType":"contract IScramble","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"scrambleLp","outputs":[{"internalType":"contract IUniswapV2Pair","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_pid","type":"uint256"},{"internalType":"uint256","name":"_allocPoint","type":"uint256"},{"internalType":"bool","name":"_withUpdate","type":"bool"}],"name":"set","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_dailyDebaseRateHardCap","type":"uint256"}],"name":"setDailyDebaseRateHardCap","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_pid","type":"uint256"},{"internalType":"uint256","name":"_lockDuration","type":"uint256"}],"name":"setLockDuration","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_fullProtec","type":"address"}],"name":"setWhiteVaultAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"startTimestamp","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalAllocPoint","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"totalStakedInPool","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_rewardPerSecond","type":"uint256"}],"name":"updateRewardPerSecond","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_account","type":"address"}],"name":"upgradeScrambleMultiplier","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_account","type":"address"}],"name":"upgradeXpMultiplier","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"address","name":"","type":"address"}],"name":"userInfo","outputs":[{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"uint256","name":"rewardDebt","type":"uint256"},{"internalType":"uint256","name":"lockEndedTimestamp","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"white","outputs":[{"internalType":"contract IWhite","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_pid","type":"uint256"},{"internalType":"uint256","name":"_amount","type":"uint256"},{"internalType":"address","name":"_account","type":"address"}],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"xpAccumulated","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"xpMultiplierLevel","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"}]