Source Code
More Info
Private Name Tags
ContractCreator
TokenTracker
Latest 25 from a total of 151 transactions
| Transaction Hash |
Method
|
Block
|
From
|
|
To
|
||||
|---|---|---|---|---|---|---|---|---|---|
| Redeem | 24427032 | 12 days ago | IN | 0 ETH | 0.00001415 | ||||
| Deposit | 24414291 | 13 days ago | IN | 0 ETH | 0.00000521 | ||||
| Redeem | 24378239 | 19 days ago | IN | 0 ETH | 0.00025444 | ||||
| Deposit | 24242237 | 37 days ago | IN | 0 ETH | 0.00001092 | ||||
| Redeem | 24230044 | 39 days ago | IN | 0 ETH | 0.00000675 | ||||
| Redeem | 24230041 | 39 days ago | IN | 0 ETH | 0.00000367 | ||||
| Deposit | 24229472 | 39 days ago | IN | 0 ETH | 0.00000505 | ||||
| Remove Collatera... | 22639279 | 262 days ago | IN | 0 ETH | 0.00275346 | ||||
| Repay Asset | 22639275 | 262 days ago | IN | 0 ETH | 0.00027984 | ||||
| Remove Collatera... | 22044755 | 345 days ago | IN | 0 ETH | 0.00029943 | ||||
| Borrow Asset | 22044755 | 345 days ago | IN | 0 ETH | 0.00025922 | ||||
| Redeem | 22044755 | 345 days ago | IN | 0 ETH | 0.00008935 | ||||
| Redeem | 22044754 | 345 days ago | IN | 0 ETH | 0.00008886 | ||||
| Redeem | 22044752 | 345 days ago | IN | 0 ETH | 0.00009581 | ||||
| Repay Asset | 21973870 | 355 days ago | IN | 0 ETH | 0.00017233 | ||||
| Repay Asset | 21973856 | 355 days ago | IN | 0 ETH | 0.00014 | ||||
| Remove Collatera... | 21969114 | 355 days ago | IN | 0 ETH | 0.00129314 | ||||
| Repay Asset | 21969110 | 355 days ago | IN | 0 ETH | 0.00014485 | ||||
| Remove Collatera... | 21944529 | 359 days ago | IN | 0 ETH | 0.00119863 | ||||
| Repay Asset | 21944521 | 359 days ago | IN | 0 ETH | 0.00010979 | ||||
| Repay Asset | 21922010 | 362 days ago | IN | 0 ETH | 0.0013907 | ||||
| Repay Asset | 21921960 | 362 days ago | IN | 0 ETH | 0.00232022 | ||||
| Redeem | 21770645 | 383 days ago | IN | 0 ETH | 0.00021649 | ||||
| Repay Asset | 21761936 | 384 days ago | IN | 0 ETH | 0.00239798 | ||||
| Remove Collatera... | 21736709 | 388 days ago | IN | 0 ETH | 0.00176812 |
Latest 1 internal transaction
Advanced mode:
| Parent Transaction Hash | Method | Block |
From
|
|
To
|
||
|---|---|---|---|---|---|---|---|
| 0x61012080 | 17140950 | 1031 days ago | Contract Creation | 0 ETH |
Loading...
Loading
Loading...
Loading
Cross-Chain Transactions
Loading...
Loading
Contract Name:
FraxlendPair
Compiler Version
v0.8.19+commit.7dd6d404
Optimization Enabled:
Yes with 830 runs
Other Settings:
default evmVersion
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: ISC
pragma solidity ^0.8.19;
// ====================================================================
// | ______ _______ |
// | / _____________ __ __ / ____(_____ ____ _____ ________ |
// | / /_ / ___/ __ `| |/_/ / /_ / / __ \/ __ `/ __ \/ ___/ _ \ |
// | / __/ / / / /_/ _> < / __/ / / / / / /_/ / / / / /__/ __/ |
// | /_/ /_/ \__,_/_/|_| /_/ /_/_/ /_/\__,_/_/ /_/\___/\___/ |
// | |
// ====================================================================
// ========================== FraxlendPair ============================
// ====================================================================
// Frax Finance: https://github.com/FraxFinance
// Primary Author
// Drake Evans: https://github.com/DrakeEvans
// Reviewers
// Dennis: https://github.com/denett
// Sam Kazemian: https://github.com/samkazemian
// Travis Moore: https://github.com/FortisFortuna
// Jack Corddry: https://github.com/corddry
// Rich Gee: https://github.com/zer0blockchain
// ====================================================================
import { ERC20 } from "@openzeppelin/contracts/token/ERC20/ERC20.sol";
import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import { IERC20Metadata } from "@openzeppelin/contracts/token/ERC20/extensions/IERC20Metadata.sol";
import { ReentrancyGuard } from "@openzeppelin/contracts/security/ReentrancyGuard.sol";
import { SafeCast } from "@openzeppelin/contracts/utils/math/SafeCast.sol";
import { FraxlendPairConstants } from "./FraxlendPairConstants.sol";
import { FraxlendPairCore } from "./FraxlendPairCore.sol";
import { Timelock2Step } from "./Timelock2Step.sol";
import { SafeERC20 } from "./libraries/SafeERC20.sol";
import { VaultAccount, VaultAccountingLibrary } from "./libraries/VaultAccount.sol";
import { IRateCalculatorV2 } from "./interfaces/IRateCalculatorV2.sol";
import { ISwapper } from "./interfaces/ISwapper.sol";
/// @title FraxlendPair
/// @author Drake Evans (Frax Finance) https://github.com/drakeevans
/// @notice The FraxlendPair is a lending pair that allows users to engage in lending and borrowing activities
contract FraxlendPair is IERC20Metadata, FraxlendPairCore {
using VaultAccountingLibrary for VaultAccount;
using SafeERC20 for IERC20;
using SafeCast for uint256;
/// @param _configData abi.encode(address _asset, address _collateral, address _oracle, uint32 _maxOracleDeviation, address _rateContract, uint64 _fullUtilizationRate, uint256 _maxLTV, uint256 _cleanLiquidationFee, uint256 _dirtyLiquidationFee, uint256 _protocolLiquidationFee)
/// @param _immutables abi.encode(address _circuitBreakerAddress, address _comptrollerAddress, address _timelockAddress)
/// @param _customConfigData abi.encode(string memory _nameOfContract, string memory _symbolOfContract, uint8 _decimalsOfContract)
constructor(
bytes memory _configData,
bytes memory _immutables,
bytes memory _customConfigData
) FraxlendPairCore(_configData, _immutables, _customConfigData) {}
// ============================================================================================
// ERC20 Metadata
// ============================================================================================
function name() public view override(ERC20, IERC20Metadata) returns (string memory) {
return nameOfContract;
}
function symbol() public view override(ERC20, IERC20Metadata) returns (string memory) {
return symbolOfContract;
}
function decimals() public view override(ERC20, IERC20Metadata) returns (uint8) {
return decimalsOfContract;
}
// totalSupply for fToken ERC20 compatibility
function totalSupply() public view override(ERC20, IERC20) returns (uint256) {
return totalAsset.shares;
}
// ============================================================================================
// Functions: Helpers
// ============================================================================================
function asset() external view returns (address) {
return address(assetContract);
}
function getConstants()
external
pure
returns (
uint256 _LTV_PRECISION,
uint256 _LIQ_PRECISION,
uint256 _UTIL_PREC,
uint256 _FEE_PRECISION,
uint256 _EXCHANGE_PRECISION,
uint256 _DEVIATION_PRECISION,
uint256 _RATE_PRECISION,
uint256 _MAX_PROTOCOL_FEE
)
{
_LTV_PRECISION = LTV_PRECISION;
_LIQ_PRECISION = LIQ_PRECISION;
_UTIL_PREC = UTIL_PREC;
_FEE_PRECISION = FEE_PRECISION;
_EXCHANGE_PRECISION = EXCHANGE_PRECISION;
_DEVIATION_PRECISION = DEVIATION_PRECISION;
_RATE_PRECISION = RATE_PRECISION;
_MAX_PROTOCOL_FEE = MAX_PROTOCOL_FEE;
}
/// @notice The ```getUserSnapshot``` function gets user level accounting data
/// @param _address The user address
/// @return _userAssetShares The user fToken balance
/// @return _userBorrowShares The user borrow shares
/// @return _userCollateralBalance The user collateral balance
function getUserSnapshot(
address _address
) external view returns (uint256 _userAssetShares, uint256 _userBorrowShares, uint256 _userCollateralBalance) {
_userAssetShares = balanceOf(_address);
_userBorrowShares = userBorrowShares[_address];
_userCollateralBalance = userCollateralBalance[_address];
}
/// @notice The ```getPairAccounting``` function gets all pair level accounting numbers
/// @return _totalAssetAmount Total assets deposited and interest accrued, total claims
/// @return _totalAssetShares Total fTokens
/// @return _totalBorrowAmount Total borrows
/// @return _totalBorrowShares Total borrow shares
/// @return _totalCollateral Total collateral
function getPairAccounting()
external
view
returns (
uint128 _totalAssetAmount,
uint128 _totalAssetShares,
uint128 _totalBorrowAmount,
uint128 _totalBorrowShares,
uint256 _totalCollateral
)
{
(, , , , VaultAccount memory _totalAsset, VaultAccount memory _totalBorrow) = previewAddInterest();
_totalAssetAmount = _totalAsset.amount;
_totalAssetShares = _totalAsset.shares;
_totalBorrowAmount = _totalBorrow.amount;
_totalBorrowShares = _totalBorrow.shares;
_totalCollateral = totalCollateral;
}
/// @notice The ```toBorrowShares``` function converts a given amount of borrow debt into the number of shares
/// @param _amount Amount of borrow
/// @param _roundUp Whether to roundup during division
/// @param _previewInterest Whether to simulate interest accrual
/// @return _shares The number of shares
function toBorrowShares(
uint256 _amount,
bool _roundUp,
bool _previewInterest
) external view returns (uint256 _shares) {
if (_previewInterest) {
(, , , , , VaultAccount memory _totalBorrow) = previewAddInterest();
_shares = _totalBorrow.toShares(_amount, _roundUp);
} else {
_shares = totalBorrow.toShares(_amount, _roundUp);
}
}
/// @notice The ```toBorrowAmount``` function converts a given amount of borrow debt into the number of shares
/// @param _shares Shares of borrow
/// @param _roundUp Whether to roundup during division
/// @param _previewInterest Whether to simulate interest accrual
/// @return _amount The amount of asset
function toBorrowAmount(
uint256 _shares,
bool _roundUp,
bool _previewInterest
) external view returns (uint256 _amount) {
if (_previewInterest) {
(, , , , , VaultAccount memory _totalBorrow) = previewAddInterest();
_amount = _totalBorrow.toAmount(_shares, _roundUp);
} else {
_amount = totalBorrow.toAmount(_shares, _roundUp);
}
}
/// @notice The ```toAssetAmount``` function converts a given number of shares to an asset amount
/// @param _shares Shares of asset (fToken)
/// @param _roundUp Whether to round up after division
/// @param _previewInterest Whether to preview interest accrual before calculation
/// @return _amount The amount of asset
function toAssetAmount(
uint256 _shares,
bool _roundUp,
bool _previewInterest
) public view returns (uint256 _amount) {
if (_previewInterest) {
(, , , , VaultAccount memory _totalAsset, ) = previewAddInterest();
_amount = _totalAsset.toAmount(_shares, _roundUp);
} else {
_amount = totalAsset.toAmount(_shares, _roundUp);
}
}
/// @notice The ```toAssetShares``` function converts a given asset amount to a number of asset shares (fTokens)
/// @param _amount The amount of asset
/// @param _roundUp Whether to round up after division
/// @param _previewInterest Whether to preview interest accrual before calculation
/// @return _shares The number of shares (fTokens)
function toAssetShares(
uint256 _amount,
bool _roundUp,
bool _previewInterest
) public view returns (uint256 _shares) {
if (_previewInterest) {
(, , , , VaultAccount memory _totalAsset, ) = previewAddInterest();
_shares = _totalAsset.toShares(_amount, _roundUp);
} else {
_shares = totalAsset.toShares(_amount, _roundUp);
}
}
function convertToAssets(uint256 _shares) external view returns (uint256 _assets) {
_assets = toAssetAmount(_shares, false, true);
}
function convertToShares(uint256 _assets) external view returns (uint256 _shares) {
_shares = toAssetShares(_assets, false, true);
}
function pricePerShare() external view returns (uint256 _amount) {
_amount = toAssetAmount(1e18, false, true);
}
function totalAssets() external view returns (uint256) {
(, , , , VaultAccount memory _totalAsset, ) = previewAddInterest();
return _totalAsset.amount;
}
function maxDeposit(address _receiver) public view returns (uint256 _maxAssets) {
(, , , , VaultAccount memory _totalAsset, ) = previewAddInterest();
_maxAssets = _totalAsset.amount >= depositLimit ? 0 : depositLimit - _totalAsset.amount;
}
function maxMint(address _receiver) external view returns (uint256 _maxShares) {
(, , , , VaultAccount memory _totalAsset, ) = previewAddInterest();
uint256 _maxDeposit = _totalAsset.amount >= depositLimit ? 0 : depositLimit - _totalAsset.amount;
_maxShares = _totalAsset.toShares(_maxDeposit, false);
}
function maxWithdraw(address _owner) external view returns (uint256 _maxAssets) {
if (isWithdrawPaused) return 0;
(
,
,
uint256 _feesShare,
,
VaultAccount memory _totalAsset,
VaultAccount memory _totalBorrow
) = previewAddInterest();
// Get the owner balance and include the fees share if owner is this contract
uint256 _ownerBalance = _owner == address(this) ? balanceOf(_owner) + _feesShare : balanceOf(_owner);
// Return the lower of total assets in contract or total assets available to _owner
uint256 _totalAssetsAvailable = _totalAssetAvailable(_totalAsset, _totalBorrow);
uint256 _totalUserWithdraw = _totalAsset.toAmount(_ownerBalance, false);
_maxAssets = _totalAssetsAvailable < _totalUserWithdraw ? _totalAssetsAvailable : _totalUserWithdraw;
}
function maxRedeem(address _owner) external view returns (uint256 _maxShares) {
if (isWithdrawPaused) return 0;
(
,
,
uint256 _feesShare,
,
VaultAccount memory _totalAsset,
VaultAccount memory _totalBorrow
) = previewAddInterest();
// Calculate the total shares available
uint256 _totalAssetsAvailable = _totalAssetAvailable(_totalAsset, _totalBorrow);
uint256 _totalSharesAvailable = _totalAsset.toShares(_totalAssetsAvailable, false);
// Get the owner balance and include the fees share if owner is this contract
uint256 _ownerBalance = _owner == address(this) ? balanceOf(_owner) + _feesShare : balanceOf(_owner);
_maxShares = _totalSharesAvailable < _ownerBalance ? _totalSharesAvailable : _ownerBalance;
}
// ============================================================================================
// Functions: Configuration
// ============================================================================================
bool public isOracleSetterRevoked;
/// @notice The ```RevokeOracleSetter``` event is emitted when the oracle setter is revoked
event RevokeOracleInfoSetter();
/// @notice The ```revokeOracleSetter``` function revokes the oracle setter
function revokeOracleInfoSetter() external {
_requireTimelock();
isOracleSetterRevoked = true;
emit RevokeOracleInfoSetter();
}
/// @notice The ```SetOracleInfo``` event is emitted when the oracle info (address and max deviation) is set
/// @param oldOracle The old oracle address
/// @param oldMaxOracleDeviation The old max oracle deviation
/// @param newOracle The new oracle address
/// @param newMaxOracleDeviation The new max oracle deviation
event SetOracleInfo(
address oldOracle,
uint32 oldMaxOracleDeviation,
address newOracle,
uint32 newMaxOracleDeviation
);
/// @notice The ```setOracleInfo``` function sets the oracle data
/// @param _newOracle The new oracle address
/// @param _newMaxOracleDeviation The new max oracle deviation
function setOracle(address _newOracle, uint32 _newMaxOracleDeviation) external {
_requireTimelock();
if (isOracleSetterRevoked) revert SetterRevoked();
ExchangeRateInfo memory _exchangeRateInfo = exchangeRateInfo;
emit SetOracleInfo(
_exchangeRateInfo.oracle,
_exchangeRateInfo.maxOracleDeviation,
_newOracle,
_newMaxOracleDeviation
);
_exchangeRateInfo.oracle = _newOracle;
_exchangeRateInfo.maxOracleDeviation = _newMaxOracleDeviation;
exchangeRateInfo = _exchangeRateInfo;
}
bool public isMaxLTVSetterRevoked;
/// @notice The ```RevokeMaxLTVSetter``` event is emitted when the max LTV setter is revoked
event RevokeMaxLTVSetter();
/// @notice The ```revokeMaxLTVSetter``` function revokes the max LTV setter
function revokeMaxLTVSetter() external {
_requireTimelock();
isMaxLTVSetterRevoked = true;
emit RevokeMaxLTVSetter();
}
/// @notice The ```SetMaxLTV``` event is emitted when the max LTV is set
/// @param oldMaxLTV The old max LTV
/// @param newMaxLTV The new max LTV
event SetMaxLTV(uint256 oldMaxLTV, uint256 newMaxLTV);
/// @notice The ```setMaxLTV``` function sets the max LTV
/// @param _newMaxLTV The new max LTV
function setMaxLTV(uint256 _newMaxLTV) external {
_requireTimelock();
if (isMaxLTVSetterRevoked) revert SetterRevoked();
emit SetMaxLTV(maxLTV, _newMaxLTV);
maxLTV = _newMaxLTV;
}
bool public isRateContractSetterRevoked;
/// @notice The ```RevokeRateContractSetter``` event is emitted when the rate contract setter is revoked
event RevokeRateContractSetter();
/// @notice The ```revokeRateContractSetter``` function revokes the rate contract setter
function revokeRateContractSetter() external {
_requireTimelock();
isRateContractSetterRevoked = true;
emit RevokeRateContractSetter();
}
/// @notice The ```SetRateContract``` event is emitted when the rate contract is set
/// @param oldRateContract The old rate contract
/// @param newRateContract The new rate contract
event SetRateContract(address oldRateContract, address newRateContract);
/// @notice The ```setRateContract``` function sets the rate contract address
/// @param _newRateContract The new rate contract address
function setRateContract(address _newRateContract) external {
_requireTimelock();
if (isRateContractSetterRevoked) revert SetterRevoked();
emit SetRateContract(address(rateContract), _newRateContract);
rateContract = IRateCalculatorV2(_newRateContract);
}
bool public isLiquidationFeeSetterRevoked;
/// @notice The ```RevokeLiquidationFeeSetter``` event is emitted when the liquidation fee setter is revoked
event RevokeLiquidationFeeSetter();
/// @notice The ```revokeLiquidationFeeSetter``` function revokes the liquidation fee setter
function revokeLiquidationFeeSetter() external {
_requireTimelock();
isLiquidationFeeSetterRevoked = true;
emit RevokeLiquidationFeeSetter();
}
/// @notice The ```SetLiquidationFees``` event is emitted when the liquidation fees are set
/// @param oldCleanLiquidationFee The old clean liquidation fee
/// @param oldDirtyLiquidationFee The old dirty liquidation fee
/// @param oldProtocolLiquidationFee The old protocol liquidation fee
/// @param newCleanLiquidationFee The new clean liquidation fee
/// @param newDirtyLiquidationFee The new dirty liquidation fee
/// @param newProtocolLiquidationFee The new protocol liquidation fee
event SetLiquidationFees(
uint256 oldCleanLiquidationFee,
uint256 oldDirtyLiquidationFee,
uint256 oldProtocolLiquidationFee,
uint256 newCleanLiquidationFee,
uint256 newDirtyLiquidationFee,
uint256 newProtocolLiquidationFee
);
/// @notice The ```setLiquidationFees``` function sets the liquidation fees
/// @param _newCleanLiquidationFee The new clean liquidation fee
/// @param _newDirtyLiquidationFee The new dirty liquidation fee
function setLiquidationFees(
uint256 _newCleanLiquidationFee,
uint256 _newDirtyLiquidationFee,
uint256 _newProtocolLiquidationFee
) external {
_requireTimelock();
if (isLiquidationFeeSetterRevoked) revert SetterRevoked();
emit SetLiquidationFees(
cleanLiquidationFee,
dirtyLiquidationFee,
protocolLiquidationFee,
_newCleanLiquidationFee,
_newDirtyLiquidationFee,
_newProtocolLiquidationFee
);
cleanLiquidationFee = _newCleanLiquidationFee;
dirtyLiquidationFee = _newDirtyLiquidationFee;
protocolLiquidationFee = _newProtocolLiquidationFee;
}
/// @notice The ```ChangeFee``` event first when the fee is changed
/// @param newFee The new fee
event ChangeFee(uint32 newFee);
/// @notice The ```changeFee``` function changes the protocol fee, max 50%
/// @param _newFee The new fee
function changeFee(uint32 _newFee) external {
_requireTimelock();
if (isInterestPaused) revert InterestPaused();
if (_newFee > MAX_PROTOCOL_FEE) {
revert BadProtocolFee();
}
_addInterest();
currentRateInfo.feeToProtocolRate = _newFee;
emit ChangeFee(_newFee);
}
/// @notice The ```WithdrawFees``` event fires when the fees are withdrawn
/// @param shares Number of shares (fTokens) redeemed
/// @param recipient To whom the assets were sent
/// @param amountToTransfer The amount of fees redeemed
event WithdrawFees(uint128 shares, address recipient, uint256 amountToTransfer, uint256 collateralAmount);
/// @notice The ```withdrawFees``` function withdraws fees accumulated
/// @param _shares Number of fTokens to redeem
/// @param _recipient Address to send the assets
/// @return _amountToTransfer Amount of assets sent to recipient
function withdrawFees(uint128 _shares, address _recipient) external onlyOwner returns (uint256 _amountToTransfer) {
if (_recipient == address(0)) revert InvalidReceiver();
// Grab some data from state to save gas
VaultAccount memory _totalAsset = totalAsset;
// Take all available if 0 value passed
if (_shares == 0) _shares = uint128(balanceOf(address(this)));
// We must calculate this before we subtract from _totalAsset or invoke _burn
_amountToTransfer = _totalAsset.toAmount(_shares, true);
_approve(address(this), msg.sender, _shares);
_redeem(_totalAsset, _amountToTransfer.toUint128(), _shares, _recipient, address(this));
uint256 _collateralAmount = userCollateralBalance[address(this)];
_removeCollateral(_collateralAmount, _recipient, address(this));
emit WithdrawFees(_shares, _recipient, _amountToTransfer, _collateralAmount);
}
/// @notice The ```SetSwapper``` event fires whenever a swapper is black or whitelisted
/// @param swapper The swapper address
/// @param approval The approval
event SetSwapper(address swapper, bool approval);
/// @notice The ```setSwapper``` function is called to black or whitelist a given swapper address
/// @dev
/// @param _swapper The swapper address
/// @param _approval The approval
function setSwapper(address _swapper, bool _approval) external onlyOwner {
swappers[_swapper] = _approval;
emit SetSwapper(_swapper, _approval);
}
// ============================================================================================
// Functions: Access Control
// ============================================================================================
/// @notice The ```pause``` function is called to pause all contract functionality
function pause() external {
_requireProtocolOrOwner();
if (!isBorrowAccessControlRevoked) _setBorrowLimit(0);
if (!isDepositAccessControlRevoked) _setDepositLimit(0);
if (!isRepayAccessControlRevoked) _pauseRepay(true);
if (!isWithdrawAccessControlRevoked) _pauseWithdraw(true);
if (!isLiquidateAccessControlRevoked) _pauseLiquidate(true);
if (!isInterestAccessControlRevoked) {
_addInterest();
_pauseInterest(true);
}
}
/// @notice The ```unpause``` function is called to unpause all contract functionality
function unpause() external {
_requireTimelockOrOwner();
if (!isBorrowAccessControlRevoked) _setBorrowLimit(type(uint256).max);
if (!isDepositAccessControlRevoked) _setDepositLimit(type(uint256).max);
if (!isRepayAccessControlRevoked) _pauseRepay(false);
if (!isWithdrawAccessControlRevoked) _pauseWithdraw(false);
if (!isLiquidateAccessControlRevoked) _pauseLiquidate(false);
if (!isInterestAccessControlRevoked) {
_addInterest();
_pauseInterest(false);
}
}
/// @notice The ```pauseBorrow``` function sets borrow limit to 0
function pauseBorrow() external {
_requireProtocolOrOwner();
if (isBorrowAccessControlRevoked) revert AccessControlRevoked();
_setBorrowLimit(0);
}
/// @notice The ```setBorrowLimit``` function sets the borrow limit
/// @param _limit The new borrow limit
function setBorrowLimit(uint256 _limit) external {
_requireTimelockOrOwner();
if (isBorrowAccessControlRevoked) revert AccessControlRevoked();
_setBorrowLimit(_limit);
}
/// @notice The ```revokeBorrowLimitAccessControl``` function revokes borrow limit access control
/// @param _borrowLimit The new borrow limit
function revokeBorrowLimitAccessControl(uint256 _borrowLimit) external {
_requireTimelock();
_revokeBorrowAccessControl(_borrowLimit);
}
/// @notice The ```pauseDeposit``` function pauses deposit functionality
function pauseDeposit() external {
_requireProtocolOrOwner();
if (isDepositAccessControlRevoked) revert AccessControlRevoked();
_setDepositLimit(0);
}
/// @notice The ```setDepositLimit``` function sets the deposit limit
/// @param _limit The new deposit limit
function setDepositLimit(uint256 _limit) external {
_requireTimelockOrOwner();
if (isDepositAccessControlRevoked) revert AccessControlRevoked();
_setDepositLimit(_limit);
}
/// @notice The ```revokeDepositLimitAccessControl``` function revokes deposit limit access control
/// @param _depositLimit The new deposit limit
function revokeDepositLimitAccessControl(uint256 _depositLimit) external {
_requireTimelock();
_revokeDepositAccessControl(_depositLimit);
}
/// @notice The ```pauseRepay``` function pauses repay functionality
/// @param _isPaused The new pause state
function pauseRepay(bool _isPaused) external {
if (_isPaused) {
_requireProtocolOrOwner();
} else {
_requireTimelockOrOwner();
}
if (isRepayAccessControlRevoked) revert AccessControlRevoked();
_pauseRepay(_isPaused);
}
/// @notice The ```revokeRepayAccessControl``` function revokes repay access control
function revokeRepayAccessControl() external {
_requireTimelock();
_revokeRepayAccessControl();
}
/// @notice The ```pauseWithdraw``` function pauses withdraw functionality
/// @param _isPaused The new pause state
function pauseWithdraw(bool _isPaused) external {
if (_isPaused) {
_requireProtocolOrOwner();
} else {
_requireTimelockOrOwner();
}
if (isWithdrawAccessControlRevoked) revert AccessControlRevoked();
_pauseWithdraw(_isPaused);
}
/// @notice The ```revokeWithdrawAccessControl``` function revokes withdraw access control
function revokeWithdrawAccessControl() external {
_requireTimelock();
_revokeWithdrawAccessControl();
}
/// @notice The ```pauseLiquidate``` function pauses liquidate functionality
/// @param _isPaused The new pause state
function pauseLiquidate(bool _isPaused) external {
if (_isPaused) {
_requireProtocolOrOwner();
} else {
_requireTimelockOrOwner();
}
if (isLiquidateAccessControlRevoked) revert AccessControlRevoked();
_pauseLiquidate(_isPaused);
}
/// @notice The ```revokeLiquidateAccessControl``` function revokes liquidate access control
function revokeLiquidateAccessControl() external {
_requireTimelock();
_revokeLiquidateAccessControl();
}
/// @notice The ```pauseInterest``` function pauses interest functionality
/// @param _isPaused The new pause state
function pauseInterest(bool _isPaused) external {
if (_isPaused) {
_requireProtocolOrOwner();
} else {
_requireTimelockOrOwner();
}
if (isInterestAccessControlRevoked) revert AccessControlRevoked();
// Resets the lastTimestamp which has the effect of no interest accruing over the pause period
_addInterest();
_pauseInterest(_isPaused);
}
/// @notice The ```revokeInterestAccessControl``` function revokes interest access control
function revokeInterestAccessControl() external {
_requireTimelock();
_revokeInterestAccessControl();
}
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.7.0) (access/Ownable.sol)
pragma solidity ^0.8.0;
import "../utils/Context.sol";
/**
* @dev Contract module which provides a basic access control mechanism, where
* there is an account (an owner) that can be granted exclusive access to
* specific functions.
*
* By default, the owner account will be the one that deploys the contract. This
* can later be changed with {transferOwnership}.
*
* This module is used through inheritance. It will make available the modifier
* `onlyOwner`, which can be applied to your functions to restrict their use to
* the owner.
*/
abstract contract Ownable is Context {
address private _owner;
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
/**
* @dev Initializes the contract setting the deployer as the initial owner.
*/
constructor() {
_transferOwnership(_msgSender());
}
/**
* @dev Throws if called by any account other than the owner.
*/
modifier onlyOwner() {
_checkOwner();
_;
}
/**
* @dev Returns the address of the current owner.
*/
function owner() public view virtual returns (address) {
return _owner;
}
/**
* @dev Throws if the sender is not the owner.
*/
function _checkOwner() internal view virtual {
require(owner() == _msgSender(), "Ownable: caller is not the owner");
}
/**
* @dev Leaves the contract without owner. It will not be possible to call
* `onlyOwner` functions anymore. Can only be called by the current owner.
*
* NOTE: Renouncing ownership will leave the contract without an owner,
* thereby removing any functionality that is only available to the owner.
*/
function renounceOwnership() public virtual onlyOwner {
_transferOwnership(address(0));
}
/**
* @dev Transfers ownership of the contract to a new account (`newOwner`).
* Can only be called by the current owner.
*/
function transferOwnership(address newOwner) public virtual onlyOwner {
require(newOwner != address(0), "Ownable: new owner is the zero address");
_transferOwnership(newOwner);
}
/**
* @dev Transfers ownership of the contract to a new account (`newOwner`).
* Internal function without access restriction.
*/
function _transferOwnership(address newOwner) internal virtual {
address oldOwner = _owner;
_owner = newOwner;
emit OwnershipTransferred(oldOwner, newOwner);
}
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.8.0) (access/Ownable2Step.sol)
pragma solidity ^0.8.0;
import "./Ownable.sol";
/**
* @dev Contract module which provides access control mechanism, where
* there is an account (an owner) that can be granted exclusive access to
* specific functions.
*
* By default, the owner account will be the one that deploys the contract. This
* can later be changed with {transferOwnership} and {acceptOwnership}.
*
* This module is used through inheritance. It will make available all functions
* from parent (Ownable).
*/
abstract contract Ownable2Step is Ownable {
address private _pendingOwner;
event OwnershipTransferStarted(address indexed previousOwner, address indexed newOwner);
/**
* @dev Returns the address of the pending owner.
*/
function pendingOwner() public view virtual returns (address) {
return _pendingOwner;
}
/**
* @dev Starts the ownership transfer of the contract to a new account. Replaces the pending transfer if there is one.
* Can only be called by the current owner.
*/
function transferOwnership(address newOwner) public virtual override onlyOwner {
_pendingOwner = newOwner;
emit OwnershipTransferStarted(owner(), newOwner);
}
/**
* @dev Transfers ownership of the contract to a new account (`newOwner`) and deletes any pending owner.
* Internal function without access restriction.
*/
function _transferOwnership(address newOwner) internal virtual override {
delete _pendingOwner;
super._transferOwnership(newOwner);
}
/**
* @dev The new owner accepts the ownership transfer.
*/
function acceptOwnership() external {
address sender = _msgSender();
require(pendingOwner() == sender, "Ownable2Step: caller is not the new owner");
_transferOwnership(sender);
}
}// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (interfaces/IERC20.sol) pragma solidity ^0.8.0; import "../token/ERC20/IERC20.sol";
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.8.0) (security/ReentrancyGuard.sol)
pragma solidity ^0.8.0;
/**
* @dev Contract module that helps prevent reentrant calls to a function.
*
* Inheriting from `ReentrancyGuard` will make the {nonReentrant} modifier
* available, which can be applied to functions to make sure there are no nested
* (reentrant) calls to them.
*
* Note that because there is a single `nonReentrant` guard, functions marked as
* `nonReentrant` may not call one another. This can be worked around by making
* those functions `private`, and then adding `external` `nonReentrant` entry
* points to them.
*
* TIP: If you would like to learn more about reentrancy and alternative ways
* to protect against it, check out our blog post
* https://blog.openzeppelin.com/reentrancy-after-istanbul/[Reentrancy After Istanbul].
*/
abstract contract ReentrancyGuard {
// Booleans are more expensive than uint256 or any type that takes up a full
// word because each write operation emits an extra SLOAD to first read the
// slot's contents, replace the bits taken up by the boolean, and then write
// back. This is the compiler's defense against contract upgrades and
// pointer aliasing, and it cannot be disabled.
// The values being non-zero value makes deployment a bit more expensive,
// but in exchange the refund on every call to nonReentrant will be lower in
// amount. Since refunds are capped to a percentage of the total
// transaction's gas, it is best to keep them low in cases like this one, to
// increase the likelihood of the full refund coming into effect.
uint256 private constant _NOT_ENTERED = 1;
uint256 private constant _ENTERED = 2;
uint256 private _status;
constructor() {
_status = _NOT_ENTERED;
}
/**
* @dev Prevents a contract from calling itself, directly or indirectly.
* Calling a `nonReentrant` function from another `nonReentrant`
* function is not supported. It is possible to prevent this from happening
* by making the `nonReentrant` function external, and making it call a
* `private` function that does the actual work.
*/
modifier nonReentrant() {
_nonReentrantBefore();
_;
_nonReentrantAfter();
}
function _nonReentrantBefore() private {
// On the first call to nonReentrant, _status will be _NOT_ENTERED
require(_status != _ENTERED, "ReentrancyGuard: reentrant call");
// Any calls to nonReentrant after this point will fail
_status = _ENTERED;
}
function _nonReentrantAfter() private {
// By storing the original value once again, a refund is triggered (see
// https://eips.ethereum.org/EIPS/eip-2200)
_status = _NOT_ENTERED;
}
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.8.0) (token/ERC20/ERC20.sol)
pragma solidity ^0.8.0;
import "./IERC20.sol";
import "./extensions/IERC20Metadata.sol";
import "../../utils/Context.sol";
/**
* @dev Implementation of the {IERC20} interface.
*
* This implementation is agnostic to the way tokens are created. This means
* that a supply mechanism has to be added in a derived contract using {_mint}.
* For a generic mechanism see {ERC20PresetMinterPauser}.
*
* TIP: For a detailed writeup see our guide
* https://forum.openzeppelin.com/t/how-to-implement-erc20-supply-mechanisms/226[How
* to implement supply mechanisms].
*
* We have followed general OpenZeppelin Contracts guidelines: functions revert
* instead returning `false` on failure. This behavior is nonetheless
* conventional and does not conflict with the expectations of ERC20
* applications.
*
* Additionally, an {Approval} event is emitted on calls to {transferFrom}.
* This allows applications to reconstruct the allowance for all accounts just
* by listening to said events. Other implementations of the EIP may not emit
* these events, as it isn't required by the specification.
*
* Finally, the non-standard {decreaseAllowance} and {increaseAllowance}
* functions have been added to mitigate the well-known issues around setting
* allowances. See {IERC20-approve}.
*/
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;
/**
* @dev Sets the values for {name} and {symbol}.
*
* The default value of {decimals} is 18. To select a different value for
* {decimals} you should overload it.
*
* All two of these values are immutable: they can only be set once during
* construction.
*/
constructor(string memory name_, string memory symbol_) {
_name = name_;
_symbol = symbol_;
}
/**
* @dev Returns the name of the token.
*/
function name() public view virtual override returns (string memory) {
return _name;
}
/**
* @dev Returns the symbol of the token, usually a shorter version of the
* name.
*/
function symbol() public view virtual override returns (string memory) {
return _symbol;
}
/**
* @dev Returns the number of decimals used to get its user representation.
* For example, if `decimals` equals `2`, a balance of `505` tokens should
* be displayed to a user as `5.05` (`505 / 10 ** 2`).
*
* Tokens usually opt for a value of 18, imitating the relationship between
* Ether and Wei. This is the value {ERC20} uses, unless this function is
* overridden;
*
* NOTE: This information is only used for _display_ purposes: it in
* no way affects any of the arithmetic of the contract, including
* {IERC20-balanceOf} and {IERC20-transfer}.
*/
function decimals() public view virtual override returns (uint8) {
return 18;
}
/**
* @dev See {IERC20-totalSupply}.
*/
function totalSupply() public view virtual override returns (uint256) {
return _totalSupply;
}
/**
* @dev See {IERC20-balanceOf}.
*/
function balanceOf(address account) public view virtual override returns (uint256) {
return _balances[account];
}
/**
* @dev See {IERC20-transfer}.
*
* Requirements:
*
* - `to` cannot be the zero address.
* - the caller must have a balance of at least `amount`.
*/
function transfer(address to, uint256 amount) public virtual override returns (bool) {
address owner = _msgSender();
_transfer(owner, to, amount);
return true;
}
/**
* @dev See {IERC20-allowance}.
*/
function allowance(address owner, address spender) public view virtual override returns (uint256) {
return _allowances[owner][spender];
}
/**
* @dev See {IERC20-approve}.
*
* NOTE: If `amount` is the maximum `uint256`, the allowance is not updated on
* `transferFrom`. This is semantically equivalent to an infinite approval.
*
* Requirements:
*
* - `spender` cannot be the zero address.
*/
function approve(address spender, uint256 amount) public virtual override returns (bool) {
address owner = _msgSender();
_approve(owner, spender, amount);
return true;
}
/**
* @dev See {IERC20-transferFrom}.
*
* Emits an {Approval} event indicating the updated allowance. This is not
* required by the EIP. See the note at the beginning of {ERC20}.
*
* NOTE: Does not update the allowance if the current allowance
* is the maximum `uint256`.
*
* Requirements:
*
* - `from` and `to` cannot be the zero address.
* - `from` must have a balance of at least `amount`.
* - the caller must have allowance for ``from``'s tokens of at least
* `amount`.
*/
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;
}
/**
* @dev Atomically increases the allowance granted to `spender` by the caller.
*
* This is an alternative to {approve} that can be used as a mitigation for
* problems described in {IERC20-approve}.
*
* Emits an {Approval} event indicating the updated allowance.
*
* Requirements:
*
* - `spender` cannot be the zero address.
*/
function increaseAllowance(address spender, uint256 addedValue) public virtual returns (bool) {
address owner = _msgSender();
_approve(owner, spender, allowance(owner, spender) + addedValue);
return true;
}
/**
* @dev Atomically decreases the allowance granted to `spender` by the caller.
*
* This is an alternative to {approve} that can be used as a mitigation for
* problems described in {IERC20-approve}.
*
* Emits an {Approval} event indicating the updated allowance.
*
* Requirements:
*
* - `spender` cannot be the zero address.
* - `spender` must have allowance for the caller of at least
* `subtractedValue`.
*/
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;
}
/**
* @dev Moves `amount` of tokens from `from` to `to`.
*
* This internal function is equivalent to {transfer}, and can be used to
* e.g. implement automatic token fees, slashing mechanisms, etc.
*
* Emits a {Transfer} event.
*
* Requirements:
*
* - `from` cannot be the zero address.
* - `to` cannot be the zero address.
* - `from` must have a balance of at least `amount`.
*/
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;
// Overflow not possible: the sum of all balances is capped by totalSupply, and the sum is preserved by
// decrementing then incrementing.
_balances[to] += amount;
}
emit Transfer(from, to, amount);
_afterTokenTransfer(from, to, amount);
}
/** @dev Creates `amount` tokens and assigns them to `account`, increasing
* the total supply.
*
* Emits a {Transfer} event with `from` set to the zero address.
*
* Requirements:
*
* - `account` cannot be the zero address.
*/
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 {
// Overflow not possible: balance + amount is at most totalSupply + amount, which is checked above.
_balances[account] += amount;
}
emit Transfer(address(0), account, amount);
_afterTokenTransfer(address(0), account, amount);
}
/**
* @dev Destroys `amount` tokens from `account`, reducing the
* total supply.
*
* Emits a {Transfer} event with `to` set to the zero address.
*
* Requirements:
*
* - `account` cannot be the zero address.
* - `account` must have at least `amount` tokens.
*/
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;
// Overflow not possible: amount <= accountBalance <= totalSupply.
_totalSupply -= amount;
}
emit Transfer(account, address(0), amount);
_afterTokenTransfer(account, address(0), amount);
}
/**
* @dev Sets `amount` as the allowance of `spender` over the `owner` s tokens.
*
* This internal function is equivalent to `approve`, and can be used to
* e.g. set automatic allowances for certain subsystems, etc.
*
* Emits an {Approval} event.
*
* Requirements:
*
* - `owner` cannot be the zero address.
* - `spender` cannot be the zero address.
*/
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);
}
/**
* @dev Updates `owner` s allowance for `spender` based on spent `amount`.
*
* Does not update the allowance amount in case of infinite allowance.
* Revert if not enough allowance is available.
*
* Might emit an {Approval} event.
*/
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);
}
}
}
/**
* @dev Hook that is called before any transfer of tokens. This includes
* minting and burning.
*
* Calling conditions:
*
* - when `from` and `to` are both non-zero, `amount` of ``from``'s tokens
* will be transferred to `to`.
* - when `from` is zero, `amount` tokens will be minted for `to`.
* - when `to` is zero, `amount` of ``from``'s tokens will be burned.
* - `from` and `to` are never both zero.
*
* To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
*/
function _beforeTokenTransfer(
address from,
address to,
uint256 amount
) internal virtual {}
/**
* @dev Hook that is called after any transfer of tokens. This includes
* minting and burning.
*
* Calling conditions:
*
* - when `from` and `to` are both non-zero, `amount` of ``from``'s tokens
* has been transferred to `to`.
* - when `from` is zero, `amount` tokens have been minted for `to`.
* - when `to` is zero, `amount` of ``from``'s tokens have been burned.
* - `from` and `to` are never both zero.
*
* To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
*/
function _afterTokenTransfer(
address from,
address to,
uint256 amount
) internal virtual {}
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.6.0) (token/ERC20/IERC20.sol)
pragma solidity ^0.8.0;
/**
* @dev Interface of the ERC20 standard as defined in the EIP.
*/
interface IERC20 {
/**
* @dev Emitted when `value` tokens are moved from one account (`from`) to
* another (`to`).
*
* Note that `value` may be zero.
*/
event Transfer(address indexed from, address indexed to, uint256 value);
/**
* @dev Emitted when the allowance of a `spender` for an `owner` is set by
* a call to {approve}. `value` is the new allowance.
*/
event Approval(address indexed owner, address indexed spender, uint256 value);
/**
* @dev Returns the amount of tokens in existence.
*/
function totalSupply() external view returns (uint256);
/**
* @dev Returns the amount of tokens owned by `account`.
*/
function balanceOf(address account) external view returns (uint256);
/**
* @dev Moves `amount` tokens from the caller's account to `to`.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* Emits a {Transfer} event.
*/
function transfer(address to, uint256 amount) external returns (bool);
/**
* @dev Returns the remaining number of tokens that `spender` will be
* allowed to spend on behalf of `owner` through {transferFrom}. This is
* zero by default.
*
* This value changes when {approve} or {transferFrom} are called.
*/
function allowance(address owner, address spender) external view returns (uint256);
/**
* @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* IMPORTANT: Beware that changing an allowance with this method brings the risk
* that someone may use both the old and the new allowance by unfortunate
* transaction ordering. One possible solution to mitigate this race
* condition is to first reduce the spender's allowance to 0 and set the
* desired value afterwards:
* https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
*
* Emits an {Approval} event.
*/
function approve(address spender, uint256 amount) external returns (bool);
/**
* @dev Moves `amount` tokens from `from` to `to` using the
* allowance mechanism. `amount` is then deducted from the caller's
* allowance.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* Emits a {Transfer} event.
*/
function transferFrom(
address from,
address to,
uint256 amount
) external returns (bool);
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (token/ERC20/extensions/IERC20Metadata.sol)
pragma solidity ^0.8.0;
import "../IERC20.sol";
/**
* @dev Interface for the optional metadata functions from the ERC20 standard.
*
* _Available since v4.1._
*/
interface IERC20Metadata is IERC20 {
/**
* @dev Returns the name of the token.
*/
function name() external view returns (string memory);
/**
* @dev Returns the symbol of the token.
*/
function symbol() external view returns (string memory);
/**
* @dev Returns the decimals places of the token.
*/
function decimals() external view returns (uint8);
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (token/ERC20/extensions/draft-IERC20Permit.sol)
pragma solidity ^0.8.0;
/**
* @dev Interface of the ERC20 Permit extension allowing approvals to be made via signatures, as defined in
* https://eips.ethereum.org/EIPS/eip-2612[EIP-2612].
*
* Adds the {permit} method, which can be used to change an account's ERC20 allowance (see {IERC20-allowance}) by
* presenting a message signed by the account. By not relying on {IERC20-approve}, the token holder account doesn't
* need to send a transaction, and thus is not required to hold Ether at all.
*/
interface IERC20Permit {
/**
* @dev Sets `value` as the allowance of `spender` over ``owner``'s tokens,
* given ``owner``'s signed approval.
*
* IMPORTANT: The same issues {IERC20-approve} has related to transaction
* ordering also apply here.
*
* Emits an {Approval} event.
*
* Requirements:
*
* - `spender` cannot be the zero address.
* - `deadline` must be a timestamp in the future.
* - `v`, `r` and `s` must be a valid `secp256k1` signature from `owner`
* over the EIP712-formatted function arguments.
* - the signature must use ``owner``'s current nonce (see {nonces}).
*
* For more information on the signature format, see the
* https://eips.ethereum.org/EIPS/eip-2612#specification[relevant EIP
* section].
*/
function permit(
address owner,
address spender,
uint256 value,
uint256 deadline,
uint8 v,
bytes32 r,
bytes32 s
) external;
/**
* @dev Returns the current nonce for `owner`. This value must be
* included whenever a signature is generated for {permit}.
*
* Every successful call to {permit} increases ``owner``'s nonce by one. This
* prevents a signature from being used multiple times.
*/
function nonces(address owner) external view returns (uint256);
/**
* @dev Returns the domain separator used in the encoding of the signature for {permit}, as defined by {EIP712}.
*/
// solhint-disable-next-line func-name-mixedcase
function DOMAIN_SEPARATOR() external view returns (bytes32);
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.8.0) (token/ERC20/utils/SafeERC20.sol)
pragma solidity ^0.8.0;
import "../IERC20.sol";
import "../extensions/draft-IERC20Permit.sol";
import "../../../utils/Address.sol";
/**
* @title SafeERC20
* @dev Wrappers around ERC20 operations that throw on failure (when the token
* contract returns false). Tokens that return no value (and instead revert or
* throw on failure) are also supported, non-reverting calls are assumed to be
* successful.
* To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract,
* which allows you to call the safe operations as `token.safeTransfer(...)`, etc.
*/
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));
}
/**
* @dev Deprecated. This function has issues similar to the ones found in
* {IERC20-approve}, and its usage is discouraged.
*
* Whenever possible, use {safeIncreaseAllowance} and
* {safeDecreaseAllowance} instead.
*/
function safeApprove(
IERC20 token,
address spender,
uint256 value
) internal {
// safeApprove should only be called when setting an initial allowance,
// or when resetting it to zero. To increase and decrease it, use
// 'safeIncreaseAllowance' and 'safeDecreaseAllowance'
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");
}
/**
* @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement
* on the return value: the return value is optional (but if data is returned, it must not be false).
* @param token The token targeted by the call.
* @param data The call data (encoded using abi.encode or one of its variants).
*/
function _callOptionalReturn(IERC20 token, bytes memory data) private {
// We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since
// we're implementing it ourselves. We use {Address-functionCall} to perform this call, which verifies that
// the target address contains contract code and also asserts for success in the low-level call.
bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed");
if (returndata.length > 0) {
// Return data is optional
require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
}
}
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.8.0) (utils/Address.sol)
pragma solidity ^0.8.1;
/**
* @dev Collection of functions related to the address type
*/
library Address {
/**
* @dev Returns true if `account` is a contract.
*
* [IMPORTANT]
* ====
* It is unsafe to assume that an address for which this function returns
* false is an externally-owned account (EOA) and not a contract.
*
* Among others, `isContract` will return false for the following
* types of addresses:
*
* - an externally-owned account
* - a contract in construction
* - an address where a contract will be created
* - an address where a contract lived, but was destroyed
* ====
*
* [IMPORTANT]
* ====
* You shouldn't rely on `isContract` to protect against flash loan attacks!
*
* Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets
* like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract
* constructor.
* ====
*/
function isContract(address account) internal view returns (bool) {
// This method relies on extcodesize/address.code.length, which returns 0
// for contracts in construction, since the code is only stored at the end
// of the constructor execution.
return account.code.length > 0;
}
/**
* @dev Replacement for Solidity's `transfer`: sends `amount` wei to
* `recipient`, forwarding all available gas and reverting on errors.
*
* https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
* of certain opcodes, possibly making contracts go over the 2300 gas limit
* imposed by `transfer`, making them unable to receive funds via
* `transfer`. {sendValue} removes this limitation.
*
* https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more].
*
* IMPORTANT: because control is transferred to `recipient`, care must be
* taken to not create reentrancy vulnerabilities. Consider using
* {ReentrancyGuard} or the
* https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
*/
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");
}
/**
* @dev Performs a Solidity function call using a low level `call`. A
* plain `call` is an unsafe replacement for a function call: use this
* function instead.
*
* If `target` reverts with a revert reason, it is bubbled up by this
* function (like regular Solidity function calls).
*
* Returns the raw returned data. To convert to the expected return value,
* use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
*
* Requirements:
*
* - `target` must be a contract.
* - calling `target` with `data` must not revert.
*
* _Available since v3.1._
*/
function functionCall(address target, bytes memory data) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, "Address: low-level call failed");
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
* `errorMessage` as a fallback revert reason when `target` reverts.
*
* _Available since v3.1._
*/
function functionCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, errorMessage);
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
* but also transferring `value` wei to `target`.
*
* Requirements:
*
* - the calling contract must have an ETH balance of at least `value`.
* - the called Solidity function must be `payable`.
*
* _Available since v3.1._
*/
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");
}
/**
* @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
* with `errorMessage` as a fallback revert reason when `target` reverts.
*
* _Available since v3.1._
*/
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);
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
* but performing a static call.
*
* _Available since v3.3._
*/
function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
return functionStaticCall(target, data, "Address: low-level static call failed");
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
* but performing a static call.
*
* _Available since v3.3._
*/
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);
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
* but performing a delegate call.
*
* _Available since v3.4._
*/
function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
return functionDelegateCall(target, data, "Address: low-level delegate call failed");
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
* but performing a delegate call.
*
* _Available since v3.4._
*/
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);
}
/**
* @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling
* the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract.
*
* _Available since v4.8._
*/
function verifyCallResultFromTarget(
address target,
bool success,
bytes memory returndata,
string memory errorMessage
) internal view returns (bytes memory) {
if (success) {
if (returndata.length == 0) {
// only check isContract if the call was successful and the return data is empty
// otherwise we already know that it was a contract
require(isContract(target), "Address: call to non-contract");
}
return returndata;
} else {
_revert(returndata, errorMessage);
}
}
/**
* @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the
* revert reason or using the provided one.
*
* _Available since v4.3._
*/
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 {
// Look for revert reason and bubble it up if present
if (returndata.length > 0) {
// The easiest way to bubble the revert reason is using memory via assembly
/// @solidity memory-safe-assembly
assembly {
let returndata_size := mload(returndata)
revert(add(32, returndata), returndata_size)
}
} else {
revert(errorMessage);
}
}
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (utils/Context.sol)
pragma solidity ^0.8.0;
/**
* @dev Provides information about the current execution context, including the
* sender of the transaction and its data. While these are generally available
* via msg.sender and msg.data, they should not be accessed in such a direct
* manner, since when dealing with meta-transactions the account sending and
* paying for execution may not be the actual sender (as far as an application
* is concerned).
*
* This contract is only required for intermediate, library-like contracts.
*/
abstract contract Context {
function _msgSender() internal view virtual returns (address) {
return msg.sender;
}
function _msgData() internal view virtual returns (bytes calldata) {
return msg.data;
}
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (utils/introspection/IERC165.sol)
pragma solidity ^0.8.0;
/**
* @dev Interface of the ERC165 standard, as defined in the
* https://eips.ethereum.org/EIPS/eip-165[EIP].
*
* Implementers can declare support of contract interfaces, which can then be
* queried by others ({ERC165Checker}).
*
* For an implementation, see {ERC165}.
*/
interface IERC165 {
/**
* @dev Returns true if this contract implements the interface defined by
* `interfaceId`. See the corresponding
* https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section]
* to learn more about how these ids are created.
*
* This function call must use less than 30 000 gas.
*/
function supportsInterface(bytes4 interfaceId) external view returns (bool);
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.8.0) (utils/math/SafeCast.sol)
// This file was procedurally generated from scripts/generate/templates/SafeCast.js.
pragma solidity ^0.8.0;
/**
* @dev Wrappers over Solidity's uintXX/intXX casting operators with added overflow
* checks.
*
* Downcasting from uint256/int256 in Solidity does not revert on overflow. This can
* easily result in undesired exploitation or bugs, since developers usually
* assume that overflows raise errors. `SafeCast` restores this intuition by
* reverting the transaction when such an operation overflows.
*
* Using this library instead of the unchecked operations eliminates an entire
* class of bugs, so it's recommended to use it always.
*
* Can be combined with {SafeMath} and {SignedSafeMath} to extend it to smaller types, by performing
* all math on `uint256` and `int256` and then downcasting.
*/
library SafeCast {
/**
* @dev Returns the downcasted uint248 from uint256, reverting on
* overflow (when the input is greater than largest uint248).
*
* Counterpart to Solidity's `uint248` operator.
*
* Requirements:
*
* - input must fit into 248 bits
*
* _Available since v4.7._
*/
function toUint248(uint256 value) internal pure returns (uint248) {
require(value <= type(uint248).max, "SafeCast: value doesn't fit in 248 bits");
return uint248(value);
}
/**
* @dev Returns the downcasted uint240 from uint256, reverting on
* overflow (when the input is greater than largest uint240).
*
* Counterpart to Solidity's `uint240` operator.
*
* Requirements:
*
* - input must fit into 240 bits
*
* _Available since v4.7._
*/
function toUint240(uint256 value) internal pure returns (uint240) {
require(value <= type(uint240).max, "SafeCast: value doesn't fit in 240 bits");
return uint240(value);
}
/**
* @dev Returns the downcasted uint232 from uint256, reverting on
* overflow (when the input is greater than largest uint232).
*
* Counterpart to Solidity's `uint232` operator.
*
* Requirements:
*
* - input must fit into 232 bits
*
* _Available since v4.7._
*/
function toUint232(uint256 value) internal pure returns (uint232) {
require(value <= type(uint232).max, "SafeCast: value doesn't fit in 232 bits");
return uint232(value);
}
/**
* @dev Returns the downcasted uint224 from uint256, reverting on
* overflow (when the input is greater than largest uint224).
*
* Counterpart to Solidity's `uint224` operator.
*
* Requirements:
*
* - input must fit into 224 bits
*
* _Available since v4.2._
*/
function toUint224(uint256 value) internal pure returns (uint224) {
require(value <= type(uint224).max, "SafeCast: value doesn't fit in 224 bits");
return uint224(value);
}
/**
* @dev Returns the downcasted uint216 from uint256, reverting on
* overflow (when the input is greater than largest uint216).
*
* Counterpart to Solidity's `uint216` operator.
*
* Requirements:
*
* - input must fit into 216 bits
*
* _Available since v4.7._
*/
function toUint216(uint256 value) internal pure returns (uint216) {
require(value <= type(uint216).max, "SafeCast: value doesn't fit in 216 bits");
return uint216(value);
}
/**
* @dev Returns the downcasted uint208 from uint256, reverting on
* overflow (when the input is greater than largest uint208).
*
* Counterpart to Solidity's `uint208` operator.
*
* Requirements:
*
* - input must fit into 208 bits
*
* _Available since v4.7._
*/
function toUint208(uint256 value) internal pure returns (uint208) {
require(value <= type(uint208).max, "SafeCast: value doesn't fit in 208 bits");
return uint208(value);
}
/**
* @dev Returns the downcasted uint200 from uint256, reverting on
* overflow (when the input is greater than largest uint200).
*
* Counterpart to Solidity's `uint200` operator.
*
* Requirements:
*
* - input must fit into 200 bits
*
* _Available since v4.7._
*/
function toUint200(uint256 value) internal pure returns (uint200) {
require(value <= type(uint200).max, "SafeCast: value doesn't fit in 200 bits");
return uint200(value);
}
/**
* @dev Returns the downcasted uint192 from uint256, reverting on
* overflow (when the input is greater than largest uint192).
*
* Counterpart to Solidity's `uint192` operator.
*
* Requirements:
*
* - input must fit into 192 bits
*
* _Available since v4.7._
*/
function toUint192(uint256 value) internal pure returns (uint192) {
require(value <= type(uint192).max, "SafeCast: value doesn't fit in 192 bits");
return uint192(value);
}
/**
* @dev Returns the downcasted uint184 from uint256, reverting on
* overflow (when the input is greater than largest uint184).
*
* Counterpart to Solidity's `uint184` operator.
*
* Requirements:
*
* - input must fit into 184 bits
*
* _Available since v4.7._
*/
function toUint184(uint256 value) internal pure returns (uint184) {
require(value <= type(uint184).max, "SafeCast: value doesn't fit in 184 bits");
return uint184(value);
}
/**
* @dev Returns the downcasted uint176 from uint256, reverting on
* overflow (when the input is greater than largest uint176).
*
* Counterpart to Solidity's `uint176` operator.
*
* Requirements:
*
* - input must fit into 176 bits
*
* _Available since v4.7._
*/
function toUint176(uint256 value) internal pure returns (uint176) {
require(value <= type(uint176).max, "SafeCast: value doesn't fit in 176 bits");
return uint176(value);
}
/**
* @dev Returns the downcasted uint168 from uint256, reverting on
* overflow (when the input is greater than largest uint168).
*
* Counterpart to Solidity's `uint168` operator.
*
* Requirements:
*
* - input must fit into 168 bits
*
* _Available since v4.7._
*/
function toUint168(uint256 value) internal pure returns (uint168) {
require(value <= type(uint168).max, "SafeCast: value doesn't fit in 168 bits");
return uint168(value);
}
/**
* @dev Returns the downcasted uint160 from uint256, reverting on
* overflow (when the input is greater than largest uint160).
*
* Counterpart to Solidity's `uint160` operator.
*
* Requirements:
*
* - input must fit into 160 bits
*
* _Available since v4.7._
*/
function toUint160(uint256 value) internal pure returns (uint160) {
require(value <= type(uint160).max, "SafeCast: value doesn't fit in 160 bits");
return uint160(value);
}
/**
* @dev Returns the downcasted uint152 from uint256, reverting on
* overflow (when the input is greater than largest uint152).
*
* Counterpart to Solidity's `uint152` operator.
*
* Requirements:
*
* - input must fit into 152 bits
*
* _Available since v4.7._
*/
function toUint152(uint256 value) internal pure returns (uint152) {
require(value <= type(uint152).max, "SafeCast: value doesn't fit in 152 bits");
return uint152(value);
}
/**
* @dev Returns the downcasted uint144 from uint256, reverting on
* overflow (when the input is greater than largest uint144).
*
* Counterpart to Solidity's `uint144` operator.
*
* Requirements:
*
* - input must fit into 144 bits
*
* _Available since v4.7._
*/
function toUint144(uint256 value) internal pure returns (uint144) {
require(value <= type(uint144).max, "SafeCast: value doesn't fit in 144 bits");
return uint144(value);
}
/**
* @dev Returns the downcasted uint136 from uint256, reverting on
* overflow (when the input is greater than largest uint136).
*
* Counterpart to Solidity's `uint136` operator.
*
* Requirements:
*
* - input must fit into 136 bits
*
* _Available since v4.7._
*/
function toUint136(uint256 value) internal pure returns (uint136) {
require(value <= type(uint136).max, "SafeCast: value doesn't fit in 136 bits");
return uint136(value);
}
/**
* @dev Returns the downcasted uint128 from uint256, reverting on
* overflow (when the input is greater than largest uint128).
*
* Counterpart to Solidity's `uint128` operator.
*
* Requirements:
*
* - input must fit into 128 bits
*
* _Available since v2.5._
*/
function toUint128(uint256 value) internal pure returns (uint128) {
require(value <= type(uint128).max, "SafeCast: value doesn't fit in 128 bits");
return uint128(value);
}
/**
* @dev Returns the downcasted uint120 from uint256, reverting on
* overflow (when the input is greater than largest uint120).
*
* Counterpart to Solidity's `uint120` operator.
*
* Requirements:
*
* - input must fit into 120 bits
*
* _Available since v4.7._
*/
function toUint120(uint256 value) internal pure returns (uint120) {
require(value <= type(uint120).max, "SafeCast: value doesn't fit in 120 bits");
return uint120(value);
}
/**
* @dev Returns the downcasted uint112 from uint256, reverting on
* overflow (when the input is greater than largest uint112).
*
* Counterpart to Solidity's `uint112` operator.
*
* Requirements:
*
* - input must fit into 112 bits
*
* _Available since v4.7._
*/
function toUint112(uint256 value) internal pure returns (uint112) {
require(value <= type(uint112).max, "SafeCast: value doesn't fit in 112 bits");
return uint112(value);
}
/**
* @dev Returns the downcasted uint104 from uint256, reverting on
* overflow (when the input is greater than largest uint104).
*
* Counterpart to Solidity's `uint104` operator.
*
* Requirements:
*
* - input must fit into 104 bits
*
* _Available since v4.7._
*/
function toUint104(uint256 value) internal pure returns (uint104) {
require(value <= type(uint104).max, "SafeCast: value doesn't fit in 104 bits");
return uint104(value);
}
/**
* @dev Returns the downcasted uint96 from uint256, reverting on
* overflow (when the input is greater than largest uint96).
*
* Counterpart to Solidity's `uint96` operator.
*
* Requirements:
*
* - input must fit into 96 bits
*
* _Available since v4.2._
*/
function toUint96(uint256 value) internal pure returns (uint96) {
require(value <= type(uint96).max, "SafeCast: value doesn't fit in 96 bits");
return uint96(value);
}
/**
* @dev Returns the downcasted uint88 from uint256, reverting on
* overflow (when the input is greater than largest uint88).
*
* Counterpart to Solidity's `uint88` operator.
*
* Requirements:
*
* - input must fit into 88 bits
*
* _Available since v4.7._
*/
function toUint88(uint256 value) internal pure returns (uint88) {
require(value <= type(uint88).max, "SafeCast: value doesn't fit in 88 bits");
return uint88(value);
}
/**
* @dev Returns the downcasted uint80 from uint256, reverting on
* overflow (when the input is greater than largest uint80).
*
* Counterpart to Solidity's `uint80` operator.
*
* Requirements:
*
* - input must fit into 80 bits
*
* _Available since v4.7._
*/
function toUint80(uint256 value) internal pure returns (uint80) {
require(value <= type(uint80).max, "SafeCast: value doesn't fit in 80 bits");
return uint80(value);
}
/**
* @dev Returns the downcasted uint72 from uint256, reverting on
* overflow (when the input is greater than largest uint72).
*
* Counterpart to Solidity's `uint72` operator.
*
* Requirements:
*
* - input must fit into 72 bits
*
* _Available since v4.7._
*/
function toUint72(uint256 value) internal pure returns (uint72) {
require(value <= type(uint72).max, "SafeCast: value doesn't fit in 72 bits");
return uint72(value);
}
/**
* @dev Returns the downcasted uint64 from uint256, reverting on
* overflow (when the input is greater than largest uint64).
*
* Counterpart to Solidity's `uint64` operator.
*
* Requirements:
*
* - input must fit into 64 bits
*
* _Available since v2.5._
*/
function toUint64(uint256 value) internal pure returns (uint64) {
require(value <= type(uint64).max, "SafeCast: value doesn't fit in 64 bits");
return uint64(value);
}
/**
* @dev Returns the downcasted uint56 from uint256, reverting on
* overflow (when the input is greater than largest uint56).
*
* Counterpart to Solidity's `uint56` operator.
*
* Requirements:
*
* - input must fit into 56 bits
*
* _Available since v4.7._
*/
function toUint56(uint256 value) internal pure returns (uint56) {
require(value <= type(uint56).max, "SafeCast: value doesn't fit in 56 bits");
return uint56(value);
}
/**
* @dev Returns the downcasted uint48 from uint256, reverting on
* overflow (when the input is greater than largest uint48).
*
* Counterpart to Solidity's `uint48` operator.
*
* Requirements:
*
* - input must fit into 48 bits
*
* _Available since v4.7._
*/
function toUint48(uint256 value) internal pure returns (uint48) {
require(value <= type(uint48).max, "SafeCast: value doesn't fit in 48 bits");
return uint48(value);
}
/**
* @dev Returns the downcasted uint40 from uint256, reverting on
* overflow (when the input is greater than largest uint40).
*
* Counterpart to Solidity's `uint40` operator.
*
* Requirements:
*
* - input must fit into 40 bits
*
* _Available since v4.7._
*/
function toUint40(uint256 value) internal pure returns (uint40) {
require(value <= type(uint40).max, "SafeCast: value doesn't fit in 40 bits");
return uint40(value);
}
/**
* @dev Returns the downcasted uint32 from uint256, reverting on
* overflow (when the input is greater than largest uint32).
*
* Counterpart to Solidity's `uint32` operator.
*
* Requirements:
*
* - input must fit into 32 bits
*
* _Available since v2.5._
*/
function toUint32(uint256 value) internal pure returns (uint32) {
require(value <= type(uint32).max, "SafeCast: value doesn't fit in 32 bits");
return uint32(value);
}
/**
* @dev Returns the downcasted uint24 from uint256, reverting on
* overflow (when the input is greater than largest uint24).
*
* Counterpart to Solidity's `uint24` operator.
*
* Requirements:
*
* - input must fit into 24 bits
*
* _Available since v4.7._
*/
function toUint24(uint256 value) internal pure returns (uint24) {
require(value <= type(uint24).max, "SafeCast: value doesn't fit in 24 bits");
return uint24(value);
}
/**
* @dev Returns the downcasted uint16 from uint256, reverting on
* overflow (when the input is greater than largest uint16).
*
* Counterpart to Solidity's `uint16` operator.
*
* Requirements:
*
* - input must fit into 16 bits
*
* _Available since v2.5._
*/
function toUint16(uint256 value) internal pure returns (uint16) {
require(value <= type(uint16).max, "SafeCast: value doesn't fit in 16 bits");
return uint16(value);
}
/**
* @dev Returns the downcasted uint8 from uint256, reverting on
* overflow (when the input is greater than largest uint8).
*
* Counterpart to Solidity's `uint8` operator.
*
* Requirements:
*
* - input must fit into 8 bits
*
* _Available since v2.5._
*/
function toUint8(uint256 value) internal pure returns (uint8) {
require(value <= type(uint8).max, "SafeCast: value doesn't fit in 8 bits");
return uint8(value);
}
/**
* @dev Converts a signed int256 into an unsigned uint256.
*
* Requirements:
*
* - input must be greater than or equal to 0.
*
* _Available since v3.0._
*/
function toUint256(int256 value) internal pure returns (uint256) {
require(value >= 0, "SafeCast: value must be positive");
return uint256(value);
}
/**
* @dev Returns the downcasted int248 from int256, reverting on
* overflow (when the input is less than smallest int248 or
* greater than largest int248).
*
* Counterpart to Solidity's `int248` operator.
*
* Requirements:
*
* - input must fit into 248 bits
*
* _Available since v4.7._
*/
function toInt248(int256 value) internal pure returns (int248 downcasted) {
downcasted = int248(value);
require(downcasted == value, "SafeCast: value doesn't fit in 248 bits");
}
/**
* @dev Returns the downcasted int240 from int256, reverting on
* overflow (when the input is less than smallest int240 or
* greater than largest int240).
*
* Counterpart to Solidity's `int240` operator.
*
* Requirements:
*
* - input must fit into 240 bits
*
* _Available since v4.7._
*/
function toInt240(int256 value) internal pure returns (int240 downcasted) {
downcasted = int240(value);
require(downcasted == value, "SafeCast: value doesn't fit in 240 bits");
}
/**
* @dev Returns the downcasted int232 from int256, reverting on
* overflow (when the input is less than smallest int232 or
* greater than largest int232).
*
* Counterpart to Solidity's `int232` operator.
*
* Requirements:
*
* - input must fit into 232 bits
*
* _Available since v4.7._
*/
function toInt232(int256 value) internal pure returns (int232 downcasted) {
downcasted = int232(value);
require(downcasted == value, "SafeCast: value doesn't fit in 232 bits");
}
/**
* @dev Returns the downcasted int224 from int256, reverting on
* overflow (when the input is less than smallest int224 or
* greater than largest int224).
*
* Counterpart to Solidity's `int224` operator.
*
* Requirements:
*
* - input must fit into 224 bits
*
* _Available since v4.7._
*/
function toInt224(int256 value) internal pure returns (int224 downcasted) {
downcasted = int224(value);
require(downcasted == value, "SafeCast: value doesn't fit in 224 bits");
}
/**
* @dev Returns the downcasted int216 from int256, reverting on
* overflow (when the input is less than smallest int216 or
* greater than largest int216).
*
* Counterpart to Solidity's `int216` operator.
*
* Requirements:
*
* - input must fit into 216 bits
*
* _Available since v4.7._
*/
function toInt216(int256 value) internal pure returns (int216 downcasted) {
downcasted = int216(value);
require(downcasted == value, "SafeCast: value doesn't fit in 216 bits");
}
/**
* @dev Returns the downcasted int208 from int256, reverting on
* overflow (when the input is less than smallest int208 or
* greater than largest int208).
*
* Counterpart to Solidity's `int208` operator.
*
* Requirements:
*
* - input must fit into 208 bits
*
* _Available since v4.7._
*/
function toInt208(int256 value) internal pure returns (int208 downcasted) {
downcasted = int208(value);
require(downcasted == value, "SafeCast: value doesn't fit in 208 bits");
}
/**
* @dev Returns the downcasted int200 from int256, reverting on
* overflow (when the input is less than smallest int200 or
* greater than largest int200).
*
* Counterpart to Solidity's `int200` operator.
*
* Requirements:
*
* - input must fit into 200 bits
*
* _Available since v4.7._
*/
function toInt200(int256 value) internal pure returns (int200 downcasted) {
downcasted = int200(value);
require(downcasted == value, "SafeCast: value doesn't fit in 200 bits");
}
/**
* @dev Returns the downcasted int192 from int256, reverting on
* overflow (when the input is less than smallest int192 or
* greater than largest int192).
*
* Counterpart to Solidity's `int192` operator.
*
* Requirements:
*
* - input must fit into 192 bits
*
* _Available since v4.7._
*/
function toInt192(int256 value) internal pure returns (int192 downcasted) {
downcasted = int192(value);
require(downcasted == value, "SafeCast: value doesn't fit in 192 bits");
}
/**
* @dev Returns the downcasted int184 from int256, reverting on
* overflow (when the input is less than smallest int184 or
* greater than largest int184).
*
* Counterpart to Solidity's `int184` operator.
*
* Requirements:
*
* - input must fit into 184 bits
*
* _Available since v4.7._
*/
function toInt184(int256 value) internal pure returns (int184 downcasted) {
downcasted = int184(value);
require(downcasted == value, "SafeCast: value doesn't fit in 184 bits");
}
/**
* @dev Returns the downcasted int176 from int256, reverting on
* overflow (when the input is less than smallest int176 or
* greater than largest int176).
*
* Counterpart to Solidity's `int176` operator.
*
* Requirements:
*
* - input must fit into 176 bits
*
* _Available since v4.7._
*/
function toInt176(int256 value) internal pure returns (int176 downcasted) {
downcasted = int176(value);
require(downcasted == value, "SafeCast: value doesn't fit in 176 bits");
}
/**
* @dev Returns the downcasted int168 from int256, reverting on
* overflow (when the input is less than smallest int168 or
* greater than largest int168).
*
* Counterpart to Solidity's `int168` operator.
*
* Requirements:
*
* - input must fit into 168 bits
*
* _Available since v4.7._
*/
function toInt168(int256 value) internal pure returns (int168 downcasted) {
downcasted = int168(value);
require(downcasted == value, "SafeCast: value doesn't fit in 168 bits");
}
/**
* @dev Returns the downcasted int160 from int256, reverting on
* overflow (when the input is less than smallest int160 or
* greater than largest int160).
*
* Counterpart to Solidity's `int160` operator.
*
* Requirements:
*
* - input must fit into 160 bits
*
* _Available since v4.7._
*/
function toInt160(int256 value) internal pure returns (int160 downcasted) {
downcasted = int160(value);
require(downcasted == value, "SafeCast: value doesn't fit in 160 bits");
}
/**
* @dev Returns the downcasted int152 from int256, reverting on
* overflow (when the input is less than smallest int152 or
* greater than largest int152).
*
* Counterpart to Solidity's `int152` operator.
*
* Requirements:
*
* - input must fit into 152 bits
*
* _Available since v4.7._
*/
function toInt152(int256 value) internal pure returns (int152 downcasted) {
downcasted = int152(value);
require(downcasted == value, "SafeCast: value doesn't fit in 152 bits");
}
/**
* @dev Returns the downcasted int144 from int256, reverting on
* overflow (when the input is less than smallest int144 or
* greater than largest int144).
*
* Counterpart to Solidity's `int144` operator.
*
* Requirements:
*
* - input must fit into 144 bits
*
* _Available since v4.7._
*/
function toInt144(int256 value) internal pure returns (int144 downcasted) {
downcasted = int144(value);
require(downcasted == value, "SafeCast: value doesn't fit in 144 bits");
}
/**
* @dev Returns the downcasted int136 from int256, reverting on
* overflow (when the input is less than smallest int136 or
* greater than largest int136).
*
* Counterpart to Solidity's `int136` operator.
*
* Requirements:
*
* - input must fit into 136 bits
*
* _Available since v4.7._
*/
function toInt136(int256 value) internal pure returns (int136 downcasted) {
downcasted = int136(value);
require(downcasted == value, "SafeCast: value doesn't fit in 136 bits");
}
/**
* @dev Returns the downcasted int128 from int256, reverting on
* overflow (when the input is less than smallest int128 or
* greater than largest int128).
*
* Counterpart to Solidity's `int128` operator.
*
* Requirements:
*
* - input must fit into 128 bits
*
* _Available since v3.1._
*/
function toInt128(int256 value) internal pure returns (int128 downcasted) {
downcasted = int128(value);
require(downcasted == value, "SafeCast: value doesn't fit in 128 bits");
}
/**
* @dev Returns the downcasted int120 from int256, reverting on
* overflow (when the input is less than smallest int120 or
* greater than largest int120).
*
* Counterpart to Solidity's `int120` operator.
*
* Requirements:
*
* - input must fit into 120 bits
*
* _Available since v4.7._
*/
function toInt120(int256 value) internal pure returns (int120 downcasted) {
downcasted = int120(value);
require(downcasted == value, "SafeCast: value doesn't fit in 120 bits");
}
/**
* @dev Returns the downcasted int112 from int256, reverting on
* overflow (when the input is less than smallest int112 or
* greater than largest int112).
*
* Counterpart to Solidity's `int112` operator.
*
* Requirements:
*
* - input must fit into 112 bits
*
* _Available since v4.7._
*/
function toInt112(int256 value) internal pure returns (int112 downcasted) {
downcasted = int112(value);
require(downcasted == value, "SafeCast: value doesn't fit in 112 bits");
}
/**
* @dev Returns the downcasted int104 from int256, reverting on
* overflow (when the input is less than smallest int104 or
* greater than largest int104).
*
* Counterpart to Solidity's `int104` operator.
*
* Requirements:
*
* - input must fit into 104 bits
*
* _Available since v4.7._
*/
function toInt104(int256 value) internal pure returns (int104 downcasted) {
downcasted = int104(value);
require(downcasted == value, "SafeCast: value doesn't fit in 104 bits");
}
/**
* @dev Returns the downcasted int96 from int256, reverting on
* overflow (when the input is less than smallest int96 or
* greater than largest int96).
*
* Counterpart to Solidity's `int96` operator.
*
* Requirements:
*
* - input must fit into 96 bits
*
* _Available since v4.7._
*/
function toInt96(int256 value) internal pure returns (int96 downcasted) {
downcasted = int96(value);
require(downcasted == value, "SafeCast: value doesn't fit in 96 bits");
}
/**
* @dev Returns the downcasted int88 from int256, reverting on
* overflow (when the input is less than smallest int88 or
* greater than largest int88).
*
* Counterpart to Solidity's `int88` operator.
*
* Requirements:
*
* - input must fit into 88 bits
*
* _Available since v4.7._
*/
function toInt88(int256 value) internal pure returns (int88 downcasted) {
downcasted = int88(value);
require(downcasted == value, "SafeCast: value doesn't fit in 88 bits");
}
/**
* @dev Returns the downcasted int80 from int256, reverting on
* overflow (when the input is less than smallest int80 or
* greater than largest int80).
*
* Counterpart to Solidity's `int80` operator.
*
* Requirements:
*
* - input must fit into 80 bits
*
* _Available since v4.7._
*/
function toInt80(int256 value) internal pure returns (int80 downcasted) {
downcasted = int80(value);
require(downcasted == value, "SafeCast: value doesn't fit in 80 bits");
}
/**
* @dev Returns the downcasted int72 from int256, reverting on
* overflow (when the input is less than smallest int72 or
* greater than largest int72).
*
* Counterpart to Solidity's `int72` operator.
*
* Requirements:
*
* - input must fit into 72 bits
*
* _Available since v4.7._
*/
function toInt72(int256 value) internal pure returns (int72 downcasted) {
downcasted = int72(value);
require(downcasted == value, "SafeCast: value doesn't fit in 72 bits");
}
/**
* @dev Returns the downcasted int64 from int256, reverting on
* overflow (when the input is less than smallest int64 or
* greater than largest int64).
*
* Counterpart to Solidity's `int64` operator.
*
* Requirements:
*
* - input must fit into 64 bits
*
* _Available since v3.1._
*/
function toInt64(int256 value) internal pure returns (int64 downcasted) {
downcasted = int64(value);
require(downcasted == value, "SafeCast: value doesn't fit in 64 bits");
}
/**
* @dev Returns the downcasted int56 from int256, reverting on
* overflow (when the input is less than smallest int56 or
* greater than largest int56).
*
* Counterpart to Solidity's `int56` operator.
*
* Requirements:
*
* - input must fit into 56 bits
*
* _Available since v4.7._
*/
function toInt56(int256 value) internal pure returns (int56 downcasted) {
downcasted = int56(value);
require(downcasted == value, "SafeCast: value doesn't fit in 56 bits");
}
/**
* @dev Returns the downcasted int48 from int256, reverting on
* overflow (when the input is less than smallest int48 or
* greater than largest int48).
*
* Counterpart to Solidity's `int48` operator.
*
* Requirements:
*
* - input must fit into 48 bits
*
* _Available since v4.7._
*/
function toInt48(int256 value) internal pure returns (int48 downcasted) {
downcasted = int48(value);
require(downcasted == value, "SafeCast: value doesn't fit in 48 bits");
}
/**
* @dev Returns the downcasted int40 from int256, reverting on
* overflow (when the input is less than smallest int40 or
* greater than largest int40).
*
* Counterpart to Solidity's `int40` operator.
*
* Requirements:
*
* - input must fit into 40 bits
*
* _Available since v4.7._
*/
function toInt40(int256 value) internal pure returns (int40 downcasted) {
downcasted = int40(value);
require(downcasted == value, "SafeCast: value doesn't fit in 40 bits");
}
/**
* @dev Returns the downcasted int32 from int256, reverting on
* overflow (when the input is less than smallest int32 or
* greater than largest int32).
*
* Counterpart to Solidity's `int32` operator.
*
* Requirements:
*
* - input must fit into 32 bits
*
* _Available since v3.1._
*/
function toInt32(int256 value) internal pure returns (int32 downcasted) {
downcasted = int32(value);
require(downcasted == value, "SafeCast: value doesn't fit in 32 bits");
}
/**
* @dev Returns the downcasted int24 from int256, reverting on
* overflow (when the input is less than smallest int24 or
* greater than largest int24).
*
* Counterpart to Solidity's `int24` operator.
*
* Requirements:
*
* - input must fit into 24 bits
*
* _Available since v4.7._
*/
function toInt24(int256 value) internal pure returns (int24 downcasted) {
downcasted = int24(value);
require(downcasted == value, "SafeCast: value doesn't fit in 24 bits");
}
/**
* @dev Returns the downcasted int16 from int256, reverting on
* overflow (when the input is less than smallest int16 or
* greater than largest int16).
*
* Counterpart to Solidity's `int16` operator.
*
* Requirements:
*
* - input must fit into 16 bits
*
* _Available since v3.1._
*/
function toInt16(int256 value) internal pure returns (int16 downcasted) {
downcasted = int16(value);
require(downcasted == value, "SafeCast: value doesn't fit in 16 bits");
}
/**
* @dev Returns the downcasted int8 from int256, reverting on
* overflow (when the input is less than smallest int8 or
* greater than largest int8).
*
* Counterpart to Solidity's `int8` operator.
*
* Requirements:
*
* - input must fit into 8 bits
*
* _Available since v3.1._
*/
function toInt8(int256 value) internal pure returns (int8 downcasted) {
downcasted = int8(value);
require(downcasted == value, "SafeCast: value doesn't fit in 8 bits");
}
/**
* @dev Converts an unsigned uint256 into a signed int256.
*
* Requirements:
*
* - input must be less than or equal to maxInt256.
*
* _Available since v3.0._
*/
function toInt256(uint256 value) internal pure returns (int256) {
// Note: Unsafe cast below is okay because `type(int256).max` is guaranteed to be positive
require(value <= uint256(type(int256).max), "SafeCast: value doesn't fit in an int256");
return int256(value);
}
}// SPDX-License-Identifier: ISC
pragma solidity ^0.8.19;
// ====================================================================
// | ______ _______ |
// | / _____________ __ __ / ____(_____ ____ _____ ________ |
// | / /_ / ___/ __ `| |/_/ / /_ / / __ \/ __ `/ __ \/ ___/ _ \ |
// | / __/ / / / /_/ _> < / __/ / / / / / /_/ / / / / /__/ __/ |
// | /_/ /_/ \__,_/_/|_| /_/ /_/_/ /_/\__,_/_/ /_/\___/\___/ |
// | |
// ====================================================================
// ==================== FraxlendPairAccessControl =====================
// ====================================================================
// Frax Finance: https://github.com/FraxFinance
// Primary Author
// Drake Evans: https://github.com/DrakeEvans
// Reviewers
// Dennis: https://github.com/denett
// ====================================================================
import { Ownable2Step, Ownable } from "@openzeppelin/contracts/access/Ownable2Step.sol";
import { Timelock2Step } from "./Timelock2Step.sol";
import { FraxlendPairAccessControlErrors } from "./FraxlendPairAccessControlErrors.sol";
/// @title FraxlendPairAccessControl
/// @author Drake Evans (Frax Finance) https://github.com/drakeevans
/// @notice An abstract contract which contains the access control logic for FraxlendPair
abstract contract FraxlendPairAccessControl is Timelock2Step, Ownable2Step, FraxlendPairAccessControlErrors {
// Deployer
address public immutable DEPLOYER_ADDRESS;
// Admin contracts
address public circuitBreakerAddress;
// access control
uint256 public borrowLimit = type(uint256).max;
bool public isBorrowAccessControlRevoked;
uint256 public depositLimit = type(uint256).max;
bool public isDepositAccessControlRevoked;
bool public isRepayPaused;
bool public isRepayAccessControlRevoked;
bool public isWithdrawPaused;
bool public isWithdrawAccessControlRevoked;
bool public isLiquidatePaused;
bool public isLiquidateAccessControlRevoked;
bool public isInterestPaused;
bool public isInterestAccessControlRevoked;
/// @param _immutables abi.encode(address _circuitBreakerAddress, address _comptrollerAddress, address _timelockAddress)
constructor(bytes memory _immutables) Timelock2Step() Ownable2Step() {
// Handle Immutables Configuration
(address _circuitBreakerAddress, address _comptrollerAddress, address _timelockAddress) = abi.decode(
_immutables,
(address, address, address)
);
_setTimelock(_timelockAddress);
_transferOwnership(_comptrollerAddress);
// Deployer contract
DEPLOYER_ADDRESS = msg.sender;
circuitBreakerAddress = _circuitBreakerAddress;
}
// ============================================================================================
// Functions: Access Control
// ============================================================================================
function _requireProtocolOrOwner() internal view {
if (
msg.sender != circuitBreakerAddress &&
msg.sender != owner() &&
msg.sender != DEPLOYER_ADDRESS &&
msg.sender != timelockAddress
) {
revert OnlyProtocolOrOwner();
}
}
function _requireTimelockOrOwner() internal view {
if (msg.sender != owner() && msg.sender != timelockAddress) {
revert OnlyTimelockOrOwner();
}
}
/// @notice The ```RevokeBorrowAccessControl``` event is emitted when access to borrow limit is revoked
/// @param borrowLimit The final permanent borrow limit
event RevokeBorrowAccessControl(uint256 borrowLimit);
function _revokeBorrowAccessControl(uint256 _borrowLimit) internal {
isBorrowAccessControlRevoked = true;
borrowLimit = _borrowLimit;
emit RevokeBorrowAccessControl(_borrowLimit);
}
/// @notice The ```SetBorrowLimit``` event is emitted when the borrow limit is set
/// @param limit The new borrow limit
event SetBorrowLimit(uint256 limit);
function _setBorrowLimit(uint256 _limit) internal {
borrowLimit = _limit;
emit SetBorrowLimit(_limit);
}
/// @notice The ```RevokeDepositAccessControl``` event is emitted when access to deposit limit is revoked
/// @param depositLimit The final permanent deposit limit
event RevokeDepositAccessControl(uint256 depositLimit);
function _revokeDepositAccessControl(uint256 _depositLimit) internal {
isDepositAccessControlRevoked = true;
depositLimit = _depositLimit;
emit RevokeDepositAccessControl(_depositLimit);
}
/// @notice The ```SetDepositLimit``` event is emitted when the deposit limit is set
/// @param limit The new deposit limit
event SetDepositLimit(uint256 limit);
function _setDepositLimit(uint256 _limit) internal {
depositLimit = _limit;
emit SetDepositLimit(_limit);
}
/// @notice The ```RevokeRepayAccessControl``` event is emitted when repay access control is revoked
event RevokeRepayAccessControl();
function _revokeRepayAccessControl() internal {
isRepayAccessControlRevoked = true;
emit RevokeRepayAccessControl();
}
/// @notice The ```PauseRepay``` event is emitted when repay is paused or unpaused
/// @param isPaused The new paused state
event PauseRepay(bool isPaused);
function _pauseRepay(bool _isPaused) internal {
isRepayPaused = _isPaused;
emit PauseRepay(_isPaused);
}
/// @notice The ```RevokeWithdrawAccessControl``` event is emitted when withdraw access control is revoked
event RevokeWithdrawAccessControl();
function _revokeWithdrawAccessControl() internal {
isWithdrawAccessControlRevoked = true;
emit RevokeWithdrawAccessControl();
}
/// @notice The ```PauseWithdraw``` event is emitted when withdraw is paused or unpaused
/// @param isPaused The new paused state
event PauseWithdraw(bool isPaused);
function _pauseWithdraw(bool _isPaused) internal {
isWithdrawPaused = _isPaused;
emit PauseWithdraw(_isPaused);
}
/// @notice The ```RevokeLiquidateAccessControl``` event is emitted when liquidate access control is revoked
event RevokeLiquidateAccessControl();
function _revokeLiquidateAccessControl() internal {
isLiquidateAccessControlRevoked = true;
emit RevokeLiquidateAccessControl();
}
/// @notice The ```PauseLiquidate``` event is emitted when liquidate is paused or unpaused
/// @param isPaused The new paused state
event PauseLiquidate(bool isPaused);
function _pauseLiquidate(bool _isPaused) internal {
isLiquidatePaused = _isPaused;
emit PauseLiquidate(_isPaused);
}
/// @notice The ```RevokeInterestAccessControl``` event is emitted when interest access control is revoked
event RevokeInterestAccessControl();
function _revokeInterestAccessControl() internal {
isInterestAccessControlRevoked = true;
emit RevokeInterestAccessControl();
}
/// @notice The ```PauseInterest``` event is emitted when interest is paused or unpaused
/// @param isPaused The new paused state
event PauseInterest(bool isPaused);
function _pauseInterest(bool _isPaused) internal {
isInterestPaused = _isPaused;
emit PauseInterest(_isPaused);
}
/// @notice The ```SetCircuitBreaker``` event is emitted when the circuit breaker address is set
/// @param oldCircuitBreaker The old circuit breaker address
/// @param newCircuitBreaker The new circuit breaker address
event SetCircuitBreaker(address oldCircuitBreaker, address newCircuitBreaker);
/// @notice The ```_setCircuitBreaker``` function is called to set the circuit breaker address
/// @param _newCircuitBreaker The new circuit breaker address
function _setCircuitBreaker(address _newCircuitBreaker) internal {
address oldCircuitBreaker = circuitBreakerAddress;
circuitBreakerAddress = _newCircuitBreaker;
emit SetCircuitBreaker(oldCircuitBreaker, _newCircuitBreaker);
}
/// @notice The ```setCircuitBreaker``` function is called to set the circuit breaker address
/// @param _newCircuitBreaker The new circuit breaker address
function setCircuitBreaker(address _newCircuitBreaker) external virtual {
_requireTimelock();
_setCircuitBreaker(_newCircuitBreaker);
}
}// SPDX-License-Identifier: ISC
pragma solidity ^0.8.19;
// ====================================================================
// | ______ _______ |
// | / _____________ __ __ / ____(_____ ____ _____ ________ |
// | / /_ / ___/ __ `| |/_/ / /_ / / __ \/ __ `/ __ \/ ___/ _ \ |
// | / __/ / / / /_/ _> < / __/ / / / / / /_/ / / / / /__/ __/ |
// | /_/ /_/ \__,_/_/|_| /_/ /_/_/ /_/\__,_/_/ /_/\___/\___/ |
// | |
// ====================================================================
// ================ FraxlendPairAccessControlErrors ===================
// ====================================================================
// Frax Finance: https://github.com/FraxFinance
// Primary Author
// Drake Evans: https://github.com/DrakeEvans
// Reviewers
// Dennis: https://github.com/denett
// ====================================================================
/// @title FraxlendPairAccessControlErrors
/// @author Drake Evans (Frax Finance) https://github.com/drakeevans
/// @notice An abstract contract which contains the errors for the Access Control contract
abstract contract FraxlendPairAccessControlErrors {
error OnlyProtocolOrOwner();
error OnlyTimelockOrOwner();
error ExceedsBorrowLimit();
error AccessControlRevoked();
error RepayPaused();
error ExceedsDepositLimit();
error WithdrawPaused();
error LiquidatePaused();
error InterestPaused();
}// SPDX-License-Identifier: ISC
pragma solidity ^0.8.19;
// ====================================================================
// | ______ _______ |
// | / _____________ __ __ / ____(_____ ____ _____ ________ |
// | / /_ / ___/ __ `| |/_/ / /_ / / __ \/ __ `/ __ \/ ___/ _ \ |
// | / __/ / / / /_/ _> < / __/ / / / / / /_/ / / / / /__/ __/ |
// | /_/ /_/ \__,_/_/|_| /_/ /_/_/ /_/\__,_/_/ /_/\___/\___/ |
// | |
// ====================================================================
// ===================== FraxlendPairConstants ========================
// ====================================================================
// Frax Finance: https://github.com/FraxFinance
// Primary Author
// Drake Evans: https://github.com/DrakeEvans
// Reviewers
// Dennis: https://github.com/denett
// Sam Kazemian: https://github.com/samkazemian
// Travis Moore: https://github.com/FortisFortuna
// Jack Corddry: https://github.com/corddry
// Rich Gee: https://github.com/zer0blockchain
// ====================================================================
/// @title FraxlendPairConstants
/// @author Drake Evans (Frax Finance) https://github.com/drakeevans
/// @notice An abstract contract which contains the errors and constants for the FraxlendPair contract
abstract contract FraxlendPairConstants {
// ============================================================================================
// Constants
// ============================================================================================
// Precision settings
uint256 public constant LTV_PRECISION = 1e5; // 5 decimals
uint256 public constant LIQ_PRECISION = 1e5;
uint256 public constant UTIL_PREC = 1e5;
uint256 public constant FEE_PRECISION = 1e5;
uint256 public constant EXCHANGE_PRECISION = 1e18;
uint256 public constant DEVIATION_PRECISION = 1e5;
uint256 public constant RATE_PRECISION = 1e18;
// Protocol Fee
uint256 public constant MAX_PROTOCOL_FEE = 5e4; // 50% 1e5 precision
error Insolvent(uint256 _borrow, uint256 _collateral, uint256 _exchangeRate);
error BorrowerSolvent();
error InsufficientAssetsInContract(uint256 _assets, uint256 _request);
error SlippageTooHigh(uint256 _minOut, uint256 _actual);
error BadSwapper();
error InvalidPath(address _expected, address _actual);
error BadProtocolFee();
error PastDeadline(uint256 _blockTimestamp, uint256 _deadline);
error SetterRevoked();
error ExceedsMaxOracleDeviation();
error InvalidReceiver();
}// SPDX-License-Identifier: ISC
pragma solidity ^0.8.19;
// ====================================================================
// | ______ _______ |
// | / _____________ __ __ / ____(_____ ____ _____ ________ |
// | / /_ / ___/ __ `| |/_/ / /_ / / __ \/ __ `/ __ \/ ___/ _ \ |
// | / __/ / / / /_/ _> < / __/ / / / / / /_/ / / / / /__/ __/ |
// | /_/ /_/ \__,_/_/|_| /_/ /_/_/ /_/\__,_/_/ /_/\___/\___/ |
// | |
// ====================================================================
// ========================= FraxlendPairCore =========================
// ====================================================================
// Frax Finance: https://github.com/FraxFinance
// Primary Author
// Drake Evans: https://github.com/DrakeEvans
// Reviewers
// Dennis: https://github.com/denett
// Sam Kazemian: https://github.com/samkazemian
// Travis Moore: https://github.com/FortisFortuna
// Jack Corddry: https://github.com/corddry
// Rich Gee: https://github.com/zer0blockchain
// ====================================================================
import { ERC20 } from "@openzeppelin/contracts/token/ERC20/ERC20.sol";
import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import { ReentrancyGuard } from "@openzeppelin/contracts/security/ReentrancyGuard.sol";
import { SafeCast } from "@openzeppelin/contracts/utils/math/SafeCast.sol";
import { FraxlendPairAccessControl } from "./FraxlendPairAccessControl.sol";
import { FraxlendPairConstants } from "./FraxlendPairConstants.sol";
import { VaultAccount, VaultAccountingLibrary } from "./libraries/VaultAccount.sol";
import { SafeERC20 } from "./libraries/SafeERC20.sol";
import { IConvexStakingWrapperFraxlend } from "./interfaces/IConvexStakingWrapperFraxlend.sol";
import { IDualOracle } from "./interfaces/IDualOracle.sol";
import { IRateCalculatorV2 } from "./interfaces/IRateCalculatorV2.sol";
import { ISwapper } from "./interfaces/ISwapper.sol";
/// @title FraxlendPairCore
/// @author Drake Evans (Frax Finance) https://github.com/drakeevans
/// @notice An abstract contract which contains the core logic and storage for the FraxlendPair
abstract contract FraxlendPairCore is FraxlendPairAccessControl, FraxlendPairConstants, ERC20, ReentrancyGuard {
using VaultAccountingLibrary for VaultAccount;
using SafeERC20 for IERC20;
using SafeCast for uint256;
function version() external pure returns (uint256 _major, uint256 _minor, uint256 _patch) {
_major = 3;
_minor = 0;
_patch = 0;
}
// ============================================================================================
// Settings set by constructor()
// ============================================================================================
// Asset and collateral contracts
IERC20 internal immutable assetContract;
IERC20 public immutable collateralContract;
// Checkpoint Contract
IConvexStakingWrapperFraxlend public immutable checkPointContract;
// LTV Settings
/// @notice The maximum LTV allowed for this pair
/// @dev 1e5 precision
uint256 public maxLTV;
// Liquidation Fees
/// @notice The liquidation fee, given as a % of repayment amount, when all collateral is consumed in liquidation
/// @dev 1e5 precision
uint256 public cleanLiquidationFee;
/// @notice The liquidation fee, given as % of repayment amount, when some collateral remains for borrower
/// @dev 1e5 precision
uint256 public dirtyLiquidationFee;
/// @notice The portion of the liquidation fee given to protocol
/// @dev 1e5 precision
uint256 public protocolLiquidationFee;
// Interest Rate Calculator Contract
IRateCalculatorV2 public rateContract; // For complex rate calculations
// Swapper
mapping(address => bool) public swappers; // approved swapper addresses
// ERC20 Metadata
string internal nameOfContract;
string internal symbolOfContract;
uint8 internal immutable decimalsOfContract;
// ============================================================================================
// Storage
// ============================================================================================
/// @notice Stores information about the current interest rate
/// @dev struct is packed to reduce SLOADs. feeToProtocolRate is 1e5 precision, ratePerSec & fullUtilizationRate is 1e18 precision
CurrentRateInfo public currentRateInfo;
struct CurrentRateInfo {
uint32 lastBlock;
uint32 feeToProtocolRate; // Fee amount 1e5 precision
uint64 lastTimestamp;
uint64 ratePerSec;
uint64 fullUtilizationRate;
}
/// @notice Stores information about the current exchange rate. Collateral:Asset ratio
/// @dev Struct packed to save SLOADs. Amount of Collateral Token to buy 1e18 Asset Token
ExchangeRateInfo public exchangeRateInfo;
struct ExchangeRateInfo {
address oracle;
uint32 maxOracleDeviation; // % of larger number, 1e5 precision
uint184 lastTimestamp;
uint256 lowExchangeRate;
uint256 highExchangeRate;
}
// Contract Level Accounting
VaultAccount public totalAsset; // amount = total amount of assets, shares = total shares outstanding
VaultAccount public totalBorrow; // amount = total borrow amount with interest accrued, shares = total shares outstanding
uint256 public totalCollateral; // total amount of collateral in contract
// User Level Accounting
/// @notice Stores the balance of collateral for each user
mapping(address => uint256) public userCollateralBalance; // amount of collateral each user is backed
/// @notice Stores the balance of borrow shares for each user
mapping(address => uint256) public userBorrowShares; // represents the shares held by individuals
// NOTE: user shares of assets are represented as ERC-20 tokens and accessible via balanceOf()
// ============================================================================================
// Constructor
// ============================================================================================
/// @notice The ```constructor``` function is called on deployment
/// @param _configData abi.encode(address _asset, address _collateral, address _oracle, uint32 _maxOracleDeviation, address _rateContract, uint64 _fullUtilizationRate, uint256 _maxLTV, uint256 _cleanLiquidationFee, uint256 _dirtyLiquidationFee, uint256 _protocolLiquidationFee)
/// @param _immutables abi.encode(address _circuitBreakerAddress, address _comptrollerAddress, address _timelockAddress)
/// @param _customConfigData abi.encode(string memory _nameOfContract, string memory _symbolOfContract, uint8 _decimalsOfContract)
constructor(
bytes memory _configData,
bytes memory _immutables,
bytes memory _customConfigData
) FraxlendPairAccessControl(_immutables) ERC20("", "") {
{
(
address _asset,
address _collateral,
address _oracle,
uint32 _maxOracleDeviation,
address _rateContract,
uint64 _fullUtilizationRate,
uint256 _maxLTV,
uint256 _liquidationFee,
uint256 _protocolLiquidationFee,
address _checkPointContract
) = abi.decode(
_configData,
(address, address, address, uint32, address, uint64, uint256, uint256, uint256, address)
);
// Pair Settings
assetContract = IERC20(_asset);
collateralContract = IERC20(_collateral);
// approve the checkpoint contract, unlimited because all collateral will be sent to it
collateralContract.approve(_checkPointContract, type(uint256).max);
// Set the checkpoint contract
checkPointContract = IConvexStakingWrapperFraxlend(_checkPointContract);
currentRateInfo.feeToProtocolRate = 0;
currentRateInfo.fullUtilizationRate = _fullUtilizationRate;
currentRateInfo.lastTimestamp = uint64(block.timestamp - 1);
currentRateInfo.lastBlock = uint32(block.number - 1);
exchangeRateInfo.oracle = _oracle;
exchangeRateInfo.maxOracleDeviation = _maxOracleDeviation;
rateContract = IRateCalculatorV2(_rateContract);
//Liquidation Fee Settings
cleanLiquidationFee = _liquidationFee;
dirtyLiquidationFee = (_liquidationFee * 90_000) / LIQ_PRECISION; // 90% of clean fee
protocolLiquidationFee = _protocolLiquidationFee;
// set maxLTV
maxLTV = _maxLTV;
}
{
(string memory _nameOfContract, string memory _symbolOfContract, uint8 _decimalsOfContract) = abi.decode(
_customConfigData,
(string, string, uint8)
);
// ERC20 Metadata
nameOfContract = _nameOfContract;
symbolOfContract = _symbolOfContract;
decimalsOfContract = _decimalsOfContract;
// Instantiate Interest
_addInterest();
// Instantiate Exchange Rate
_updateExchangeRate();
}
}
// ============================================================================================
// Internal Helpers
// ============================================================================================
/// @notice The ```_totalAssetAvailable``` function returns the total balance of Asset Tokens in the contract
/// @param _totalAsset VaultAccount struct which stores total amount and shares for assets
/// @param _totalBorrow VaultAccount struct which stores total amount and shares for borrows
/// @return The balance of Asset Tokens held by contract
function _totalAssetAvailable(
VaultAccount memory _totalAsset,
VaultAccount memory _totalBorrow
) internal pure returns (uint256) {
return _totalAsset.amount - _totalBorrow.amount;
}
/// @notice The ```_isSolvent``` function determines if a given borrower is solvent given an exchange rate
/// @param _borrower The borrower address to check
/// @param _exchangeRate The exchange rate, i.e. the amount of collateral to buy 1e18 asset
/// @return Whether borrower is solvent
function _isSolvent(address _borrower, uint256 _exchangeRate) internal view returns (bool) {
if (maxLTV == 0) return true;
uint256 _borrowerAmount = totalBorrow.toAmount(userBorrowShares[_borrower], true);
if (_borrowerAmount == 0) return true;
uint256 _collateralAmount = userCollateralBalance[_borrower];
if (_collateralAmount == 0) return false;
uint256 _ltv = (((_borrowerAmount * _exchangeRate) / EXCHANGE_PRECISION) * LTV_PRECISION) / _collateralAmount;
return _ltv <= maxLTV;
}
// ============================================================================================
// Modifiers
// ============================================================================================
/// @notice Checks for solvency AFTER executing contract code
/// @param _borrower The borrower whose solvency we will check
modifier isSolvent(address _borrower) {
_;
ExchangeRateInfo memory _exchangeRateInfo = exchangeRateInfo;
if (!_isSolvent(_borrower, exchangeRateInfo.highExchangeRate)) {
revert Insolvent(
totalBorrow.toAmount(userBorrowShares[_borrower], true),
userCollateralBalance[_borrower],
exchangeRateInfo.highExchangeRate
);
}
}
// ============================================================================================
// Functions: Interest Accumulation and Adjustment
// ============================================================================================
/// @notice The ```AddInterest``` event is emitted when interest is accrued by borrowers
/// @param interestEarned The total interest accrued by all borrowers
/// @param rate The interest rate used to calculate accrued interest
/// @param feesAmount The amount of fees paid to protocol
/// @param feesShare The amount of shares distributed to protocol
event AddInterest(uint256 interestEarned, uint256 rate, uint256 feesAmount, uint256 feesShare);
/// @notice The ```UpdateRate``` event is emitted when the interest rate is updated
/// @param oldRatePerSec The old interest rate (per second)
/// @param oldFullUtilizationRate The old full utilization rate
/// @param newRatePerSec The new interest rate (per second)
/// @param newFullUtilizationRate The new full utilization rate
event UpdateRate(
uint256 oldRatePerSec,
uint256 oldFullUtilizationRate,
uint256 newRatePerSec,
uint256 newFullUtilizationRate
);
/// @notice The ```addInterest``` function is a public implementation of _addInterest and allows 3rd parties to trigger interest accrual
/// @return _interestEarned The amount of interest accrued by all borrowers
/// @return _feesAmount The amount of fees paid to protocol
/// @return _feesShare The amount of shares distributed to protocol
/// @return _currentRateInfo The new rate info struct
/// @return _totalAsset The new total asset struct
/// @return _totalBorrow The new total borrow struct
function addInterest(
bool _returnAccounting
)
external
nonReentrant
returns (
uint256 _interestEarned,
uint256 _feesAmount,
uint256 _feesShare,
CurrentRateInfo memory _currentRateInfo,
VaultAccount memory _totalAsset,
VaultAccount memory _totalBorrow
)
{
(, _interestEarned, _feesAmount, _feesShare, _currentRateInfo) = _addInterest();
if (_returnAccounting) {
_totalAsset = totalAsset;
_totalBorrow = totalBorrow;
}
}
/// @notice The ```previewAddInterest``` function
/// @return _interestEarned The amount of interest accrued by all borrowers
/// @return _feesAmount The amount of fees paid to protocol
/// @return _feesShare The amount of shares distributed to protocol
/// @return _newCurrentRateInfo The new rate info struct
/// @return _totalAsset The new total asset struct
/// @return _totalBorrow The new total borrow struct
function previewAddInterest()
public
view
returns (
uint256 _interestEarned,
uint256 _feesAmount,
uint256 _feesShare,
CurrentRateInfo memory _newCurrentRateInfo,
VaultAccount memory _totalAsset,
VaultAccount memory _totalBorrow
)
{
_newCurrentRateInfo = currentRateInfo;
// Write return values
InterestCalculationResults memory _results = _calculateInterest(_newCurrentRateInfo);
if (_results.isInterestUpdated) {
_interestEarned = _results.interestEarned;
_feesAmount = _results.feesAmount;
_feesShare = _results.feesShare;
_newCurrentRateInfo.ratePerSec = _results.newRate;
_newCurrentRateInfo.fullUtilizationRate = _results.newFullUtilizationRate;
_totalAsset = _results.totalAsset;
_totalBorrow = _results.totalBorrow;
} else {
_totalAsset = totalAsset;
_totalBorrow = totalBorrow;
}
}
struct InterestCalculationResults {
bool isInterestUpdated;
uint64 newRate;
uint64 newFullUtilizationRate;
uint256 interestEarned;
uint256 feesAmount;
uint256 feesShare;
VaultAccount totalAsset;
VaultAccount totalBorrow;
}
/// @notice The ```_calculateInterest``` function calculates the interest to be accrued and the new interest rate info
/// @param _currentRateInfo The current rate info
/// @return _results The results of the interest calculation
function _calculateInterest(
CurrentRateInfo memory _currentRateInfo
) internal view returns (InterestCalculationResults memory _results) {
// Short circuit if interest already calculated this block OR if interest is paused
if (_currentRateInfo.lastTimestamp != block.timestamp && !isInterestPaused) {
// Indicate that interest is updated and calculated
_results.isInterestUpdated = true;
// Write return values and use these to save gas
_results.totalAsset = totalAsset;
_results.totalBorrow = totalBorrow;
// Time elapsed since last interest update
uint256 _deltaTime = block.timestamp - _currentRateInfo.lastTimestamp;
// Get the utilization rate
uint256 _utilizationRate = _results.totalAsset.amount == 0
? 0
: (UTIL_PREC * _results.totalBorrow.amount) / _results.totalAsset.amount;
// Request new interest rate and full utilization rate from the rate calculator
(_results.newRate, _results.newFullUtilizationRate) = IRateCalculatorV2(rateContract).getNewRate(
_deltaTime,
_utilizationRate,
_currentRateInfo.fullUtilizationRate
);
// Calculate interest accrued
_results.interestEarned = (_deltaTime * _results.totalBorrow.amount * _results.newRate) / RATE_PRECISION;
// Accrue interest (if any) and fees iff no overflow
if (
_results.interestEarned > 0 &&
_results.interestEarned + _results.totalBorrow.amount <= type(uint128).max &&
_results.interestEarned + _results.totalAsset.amount <= type(uint128).max
) {
// Increment totalBorrow and totalAsset by interestEarned
_results.totalBorrow.amount += uint128(_results.interestEarned);
_results.totalAsset.amount += uint128(_results.interestEarned);
if (_currentRateInfo.feeToProtocolRate > 0) {
_results.feesAmount =
(_results.interestEarned * _currentRateInfo.feeToProtocolRate) /
FEE_PRECISION;
_results.feesShare =
(_results.feesAmount * _results.totalAsset.shares) /
(_results.totalAsset.amount - _results.feesAmount);
// Effects: Give new shares to this contract, effectively diluting lenders an amount equal to the fees
// We can safely cast because _feesShare < _feesAmount < interestEarned which is always less than uint128
_results.totalAsset.shares += uint128(_results.feesShare);
}
}
}
}
/// @notice The ```_addInterest``` function is invoked prior to every external function and is used to accrue interest and update interest rate
/// @dev Can only called once per block
/// @return _isInterestUpdated True if interest was calculated
/// @return _interestEarned The amount of interest accrued by all borrowers
/// @return _feesAmount The amount of fees paid to protocol
/// @return _feesShare The amount of shares distributed to protocol
/// @return _currentRateInfo The new rate info struct
function _addInterest()
internal
returns (
bool _isInterestUpdated,
uint256 _interestEarned,
uint256 _feesAmount,
uint256 _feesShare,
CurrentRateInfo memory _currentRateInfo
)
{
// Pull from storage and set default return values
_currentRateInfo = currentRateInfo;
// Calc interest
InterestCalculationResults memory _results = _calculateInterest(_currentRateInfo);
// Write return values only if interest was updated and calculated
if (_results.isInterestUpdated) {
_isInterestUpdated = _results.isInterestUpdated;
_interestEarned = _results.interestEarned;
_feesAmount = _results.feesAmount;
_feesShare = _results.feesShare;
// emit here so that we have access to the old values
emit UpdateRate(
_currentRateInfo.ratePerSec,
_currentRateInfo.fullUtilizationRate,
_results.newRate,
_results.newFullUtilizationRate
);
emit AddInterest(_interestEarned, _results.newRate, _feesAmount, _feesShare);
// overwrite original values
_currentRateInfo.ratePerSec = _results.newRate;
_currentRateInfo.fullUtilizationRate = _results.newFullUtilizationRate;
_currentRateInfo.lastTimestamp = uint64(block.timestamp);
_currentRateInfo.lastBlock = uint32(block.number);
// Effects: write to state
currentRateInfo = _currentRateInfo;
totalAsset = _results.totalAsset;
totalBorrow = _results.totalBorrow;
if (_feesShare > 0) _mint(address(this), _feesShare);
}
}
// ============================================================================================
// Functions: ExchangeRate
// ============================================================================================
/// @notice The ```UpdateExchangeRate``` event is emitted when the Collateral:Asset exchange rate is updated
/// @param lowExchangeRate The low exchange rate
/// @param highExchangeRate The high exchange rate
event UpdateExchangeRate(uint256 lowExchangeRate, uint256 highExchangeRate);
/// @notice The ```WarnOracleData``` event is emitted when one of the oracles has stale or otherwise problematic data
/// @param oracle The oracle address
event WarnOracleData(address oracle);
/// @notice The ```updateExchangeRate``` function is the external implementation of _updateExchangeRate.
/// @dev This function is invoked at most once per block as these queries can be expensive
/// @return _isBorrowAllowed True if deviation is within bounds
/// @return _lowExchangeRate The low exchange rate
/// @return _highExchangeRate The high exchange rate
function updateExchangeRate()
external
nonReentrant
returns (bool _isBorrowAllowed, uint256 _lowExchangeRate, uint256 _highExchangeRate)
{
return _updateExchangeRate();
}
/// @notice The ```_updateExchangeRate``` function retrieves the latest exchange rate. i.e how much collateral to buy 1e18 asset.
/// @dev This function is invoked at most once per block as these queries can be expensive
/// @return _isBorrowAllowed True if deviation is within bounds
/// @return _lowExchangeRate The low exchange rate
/// @return _highExchangeRate The high exchange rate
function _updateExchangeRate()
internal
returns (bool _isBorrowAllowed, uint256 _lowExchangeRate, uint256 _highExchangeRate)
{
// Pull from storage to save gas and set default return values
ExchangeRateInfo memory _exchangeRateInfo = exchangeRateInfo;
// Short circuit if already updated this block
if (_exchangeRateInfo.lastTimestamp != block.timestamp) {
// Get the latest exchange rate from the dual oracle
bool _oneOracleBad;
(_oneOracleBad, _lowExchangeRate, _highExchangeRate) = IDualOracle(_exchangeRateInfo.oracle).getPrices();
// If one oracle is bad data, emit an event for off-chain monitoring
if (_oneOracleBad) emit WarnOracleData(_exchangeRateInfo.oracle);
// Effects: Bookkeeping and write to storage
_exchangeRateInfo.lastTimestamp = uint184(block.timestamp);
_exchangeRateInfo.lowExchangeRate = _lowExchangeRate;
_exchangeRateInfo.highExchangeRate = _highExchangeRate;
exchangeRateInfo = _exchangeRateInfo;
emit UpdateExchangeRate(_lowExchangeRate, _highExchangeRate);
} else {
// Use default return values if already updated this block
_lowExchangeRate = _exchangeRateInfo.lowExchangeRate;
_highExchangeRate = _exchangeRateInfo.highExchangeRate;
}
uint256 _deviation = (DEVIATION_PRECISION *
(_exchangeRateInfo.highExchangeRate - _exchangeRateInfo.lowExchangeRate)) /
_exchangeRateInfo.highExchangeRate;
if (_deviation <= _exchangeRateInfo.maxOracleDeviation) {
_isBorrowAllowed = true;
}
}
// ============================================================================================
// Functions: Lending
// ============================================================================================
/// @notice The ```Deposit``` event fires when a user deposits assets to the pair
/// @param caller the msg.sender
/// @param owner the account the fTokens are sent to
/// @param assets the amount of assets deposited
/// @param shares the number of fTokens minted
event Deposit(address indexed caller, address indexed owner, uint256 assets, uint256 shares);
/// @notice The ```_deposit``` function is the internal implementation for lending assets
/// @dev Caller must invoke ```ERC20.approve``` on the Asset Token contract prior to calling function
/// @param _totalAsset An in memory VaultAccount struct representing the total amounts and shares for the Asset Token
/// @param _amount The amount of Asset Token to be transferred
/// @param _shares The amount of Asset Shares (fTokens) to be minted
/// @param _receiver The address to receive the Asset Shares (fTokens)
function _deposit(VaultAccount memory _totalAsset, uint128 _amount, uint128 _shares, address _receiver) internal {
// Effects: bookkeeping
_totalAsset.amount += _amount;
_totalAsset.shares += _shares;
// Effects: write back to storage
_mint(_receiver, _shares);
totalAsset = _totalAsset;
// Interactions
assetContract.safeTransferFrom(msg.sender, address(this), _amount);
emit Deposit(msg.sender, _receiver, _amount, _shares);
}
function previewDeposit(uint256 _assets) external view returns (uint256 _sharesReceived) {
(, , , , VaultAccount memory _totalAsset, ) = previewAddInterest();
_sharesReceived = _totalAsset.toShares(_assets, false);
}
/// @notice The ```deposit``` function allows a user to Lend Assets by specifying the amount of Asset Tokens to lend
/// @dev Caller must invoke ```ERC20.approve``` on the Asset Token contract prior to calling function
/// @param _amount The amount of Asset Token to transfer to Pair
/// @param _receiver The address to receive the Asset Shares (fTokens)
/// @return _sharesReceived The number of fTokens received for the deposit
function deposit(uint256 _amount, address _receiver) external nonReentrant returns (uint256 _sharesReceived) {
if (_receiver == address(0)) revert InvalidReceiver();
// Accrue interest if necessary
_addInterest();
// Pull from storage to save gas
VaultAccount memory _totalAsset = totalAsset;
// Check if this deposit will violate the deposit limit
if (depositLimit < _totalAsset.amount + _amount) revert ExceedsDepositLimit();
// Calculate the number of fTokens to mint
_sharesReceived = _totalAsset.toShares(_amount, false);
// Execute the deposit effects
_deposit(_totalAsset, _amount.toUint128(), _sharesReceived.toUint128(), _receiver);
}
function previewMint(uint256 _shares) external view returns (uint256 _amount) {
(, , , , VaultAccount memory _totalAsset, ) = previewAddInterest();
_amount = _totalAsset.toAmount(_shares, false);
}
function mint(uint256 _shares, address _receiver) external nonReentrant returns (uint256 _amount) {
if (_receiver == address(0)) revert InvalidReceiver();
// Accrue interest if necessary
_addInterest();
// Pull from storage to save gas
VaultAccount memory _totalAsset = totalAsset;
// Calculate the number of assets to transfer based on the shares to mint
_amount = _totalAsset.toAmount(_shares, false);
// Check if this deposit will violate the deposit limit
if (depositLimit < _totalAsset.amount + _amount) revert ExceedsDepositLimit();
// Execute the deposit effects
_deposit(_totalAsset, _amount.toUint128(), _shares.toUint128(), _receiver);
}
/// @notice The ```Withdraw``` event fires when a user redeems their fTokens for the underlying asset
/// @param caller the msg.sender
/// @param receiver The address to which the underlying asset will be transferred to
/// @param owner The owner of the fTokens
/// @param assets The assets transferred
/// @param shares The number of fTokens burned
event Withdraw(
address indexed caller,
address indexed receiver,
address indexed owner,
uint256 assets,
uint256 shares
);
/// @notice The ```_redeem``` function is an internal implementation which allows a Lender to pull their Asset Tokens out of the Pair
/// @dev Caller must invoke ```ERC20.approve``` on the Asset Token contract prior to calling function
/// @param _totalAsset An in-memory VaultAccount struct which holds the total amount of Asset Tokens and the total number of Asset Shares (fTokens)
/// @param _amountToReturn The number of Asset Tokens to return
/// @param _shares The number of Asset Shares (fTokens) to burn
/// @param _receiver The address to which the Asset Tokens will be transferred
/// @param _owner The owner of the Asset Shares (fTokens)
function _redeem(
VaultAccount memory _totalAsset,
uint128 _amountToReturn,
uint128 _shares,
address _receiver,
address _owner
) internal {
// Check for sufficient allowance/approval if necessary
if (msg.sender != _owner) {
uint256 allowed = allowance(_owner, msg.sender);
// NOTE: This will revert on underflow ensuring that allowance > shares
if (allowed != type(uint256).max) _approve(_owner, msg.sender, allowed - _shares);
}
// Check for sufficient withdraw liquidity (not strictly necessary because balance will underflow)
uint256 _assetsAvailable = _totalAssetAvailable(_totalAsset, totalBorrow);
if (_assetsAvailable < _amountToReturn) {
revert InsufficientAssetsInContract(_assetsAvailable, _amountToReturn);
}
// Effects: bookkeeping
_totalAsset.amount -= _amountToReturn;
_totalAsset.shares -= _shares;
// Effects: write to storage
totalAsset = _totalAsset;
_burn(_owner, _shares);
// Interactions
assetContract.safeTransfer(_receiver, _amountToReturn);
emit Withdraw(msg.sender, _receiver, _owner, _amountToReturn, _shares);
}
function previewRedeem(uint256 _shares) external view returns (uint256 _assets) {
(, , , , VaultAccount memory _totalAsset, ) = previewAddInterest();
_assets = _totalAsset.toAmount(_shares, false);
}
/// @notice The ```redeem``` function allows the caller to redeem their Asset Shares for Asset Tokens
/// @param _shares The number of Asset Shares (fTokens) to burn for Asset Tokens
/// @param _receiver The address to which the Asset Tokens will be transferred
/// @param _owner The owner of the Asset Shares (fTokens)
/// @return _amountToReturn The amount of Asset Tokens to be transferred
function redeem(
uint256 _shares,
address _receiver,
address _owner
) external nonReentrant returns (uint256 _amountToReturn) {
if (_receiver == address(0)) revert InvalidReceiver();
// Check if withdraw is paused and revert if necessary
if (isWithdrawPaused) revert WithdrawPaused();
// Accrue interest if necessary
_addInterest();
// Pull from storage to save gas
VaultAccount memory _totalAsset = totalAsset;
// Calculate the number of assets to transfer based on the shares to burn
_amountToReturn = _totalAsset.toAmount(_shares, false);
// Execute the withdraw effects
_redeem(_totalAsset, _amountToReturn.toUint128(), _shares.toUint128(), _receiver, _owner);
}
/// @notice The ```previewWithdraw``` function returns the number of Asset Shares (fTokens) that would be burned for a given amount of Asset Tokens
/// @param _amount The amount of Asset Tokens to be withdrawn
/// @return _sharesToBurn The number of shares that would be burned
function previewWithdraw(uint256 _amount) external view returns (uint256 _sharesToBurn) {
(, , , , VaultAccount memory _totalAsset, ) = previewAddInterest();
_sharesToBurn = _totalAsset.toShares(_amount, true);
}
/// @notice The ```withdraw``` function allows the caller to withdraw their Asset Tokens for a given amount of fTokens
/// @param _amount The amount to withdraw
/// @param _receiver The address to which the Asset Tokens will be transferred
/// @param _owner The owner of the Asset Shares (fTokens)
/// @return _sharesToBurn The number of shares (fTokens) that were burned
function withdraw(
uint256 _amount,
address _receiver,
address _owner
) external nonReentrant returns (uint256 _sharesToBurn) {
if (_receiver == address(0)) revert InvalidReceiver();
// Check if withdraw is paused and revert if necessary
if (isWithdrawPaused) revert WithdrawPaused();
// Accrue interest if necessary
_addInterest();
// Pull from storage to save gas
VaultAccount memory _totalAsset = totalAsset;
// Calculate the number of shares to burn based on the amount to withdraw
_sharesToBurn = _totalAsset.toShares(_amount, true);
// Execute the withdraw effects
_redeem(_totalAsset, _amount.toUint128(), _sharesToBurn.toUint128(), _receiver, _owner);
}
// ============================================================================================
// Functions: Borrowing
// ============================================================================================
/// @notice The ```BorrowAsset``` event is emitted when a borrower increases their position
/// @param _borrower The borrower whose account was debited
/// @param _receiver The address to which the Asset Tokens were transferred
/// @param _borrowAmount The amount of Asset Tokens transferred
/// @param _sharesAdded The number of Borrow Shares the borrower was debited
event BorrowAsset(
address indexed _borrower,
address indexed _receiver,
uint256 _borrowAmount,
uint256 _sharesAdded
);
/// @notice The ```_borrowAsset``` function is the internal implementation for borrowing assets
/// @param _borrowAmount The amount of the Asset Token to borrow
/// @param _receiver The address to receive the Asset Tokens
/// @return _sharesAdded The amount of borrow shares the msg.sender will be debited
function _borrowAsset(uint128 _borrowAmount, address _receiver) internal returns (uint256 _sharesAdded) {
// Get borrow accounting from storage to save gas
VaultAccount memory _totalBorrow = totalBorrow;
// Check available capital (not strictly necessary because balance will underflow, but better revert message)
uint256 _assetsAvailable = _totalAssetAvailable(totalAsset, _totalBorrow);
if (_assetsAvailable < _borrowAmount) {
revert InsufficientAssetsInContract(_assetsAvailable, _borrowAmount);
}
// Calculate the number of shares to add based on the amount to borrow
_sharesAdded = _totalBorrow.toShares(_borrowAmount, true);
// Effects: Bookkeeping to add shares & amounts to total Borrow accounting
_totalBorrow.amount += _borrowAmount;
_totalBorrow.shares += uint128(_sharesAdded);
// NOTE: we can safely cast here because shares are always less than amount and _borrowAmount is uint128
// Effects: write back to storage
totalBorrow = _totalBorrow;
userBorrowShares[msg.sender] += _sharesAdded;
// Interactions
if (_receiver != address(this)) {
assetContract.safeTransfer(_receiver, _borrowAmount);
}
emit BorrowAsset(msg.sender, _receiver, _borrowAmount, _sharesAdded);
}
/// @notice The ```borrowAsset``` function allows a user to open/increase a borrow position
/// @dev Borrower must call ```ERC20.approve``` on the Collateral Token contract if applicable
/// @param _borrowAmount The amount of Asset Token to borrow
/// @param _collateralAmount The amount of Collateral Token to transfer to Pair
/// @param _receiver The address which will receive the Asset Tokens
/// @return _shares The number of borrow Shares the msg.sender will be debited
function borrowAsset(
uint256 _borrowAmount,
uint256 _collateralAmount,
address _receiver
) external nonReentrant isSolvent(msg.sender) returns (uint256 _shares) {
if (_receiver == address(0)) revert InvalidReceiver();
// Accrue interest if necessary
_addInterest();
// Check if borrow will violate the borrow limit and revert if necessary
if (borrowLimit < totalBorrow.amount + _borrowAmount) revert ExceedsBorrowLimit();
// Update _exchangeRate and check if borrow is allowed based on deviation
(bool _isBorrowAllowed, , ) = _updateExchangeRate();
if (!_isBorrowAllowed) revert ExceedsMaxOracleDeviation();
// Only add collateral if necessary
if (_collateralAmount > 0) {
_addCollateral(msg.sender, _collateralAmount, msg.sender);
}
// Effects: Call internal borrow function
_shares = _borrowAsset(_borrowAmount.toUint128(), _receiver);
}
/// @notice The ```AddCollateral``` event is emitted when a borrower adds collateral to their position
/// @param sender The source of funds for the new collateral
/// @param borrower The borrower account for which the collateral should be credited
/// @param collateralAmount The amount of Collateral Token to be transferred
event AddCollateral(address indexed sender, address indexed borrower, uint256 collateralAmount);
/// @notice The ```_addCollateral``` function is an internal implementation for adding collateral to a borrowers position
/// @param _sender The source of funds for the new collateral
/// @param _collateralAmount The amount of Collateral Token to be transferred
/// @param _borrower The borrower account for which the collateral should be credited
function _addCollateral(address _sender, uint256 _collateralAmount, address _borrower) internal {
// NOTE: violates checks-effects-interactions pattern. Mark function NONREENTRANT
IConvexStakingWrapperFraxlend _checkPointContract = checkPointContract;
_checkPointContract.user_checkpoint(_borrower);
// Effects: write to state
userCollateralBalance[_borrower] += _collateralAmount;
totalCollateral += _collateralAmount;
// Interactions
if (_sender != address(this)) {
collateralContract.safeTransferFrom(_sender, address(this), _collateralAmount);
_checkPointContract.deposit(_collateralAmount, address(this));
}
emit AddCollateral(_sender, _borrower, _collateralAmount);
}
/// @notice The ```addCollateral``` function allows the caller to add Collateral Token to a borrowers position
/// @dev msg.sender must call ERC20.approve() on the Collateral Token contract prior to invocation
/// @param _collateralAmount The amount of Collateral Token to be added to borrower's position
/// @param _borrower The account to be credited
function addCollateral(uint256 _collateralAmount, address _borrower) external nonReentrant {
if (_borrower == address(0)) revert InvalidReceiver();
_addInterest();
_addCollateral(msg.sender, _collateralAmount, _borrower);
}
/// @notice The ```RemoveCollateral``` event is emitted when collateral is removed from a borrower's position
/// @param _sender The account from which funds are transferred
/// @param _collateralAmount The amount of Collateral Token to be transferred
/// @param _receiver The address to which Collateral Tokens will be transferred
event RemoveCollateral(
address indexed _sender,
uint256 _collateralAmount,
address indexed _receiver,
address indexed _borrower
);
/// @notice The ```_removeCollateral``` function is the internal implementation for removing collateral from a borrower's position
/// @param _collateralAmount The amount of Collateral Token to remove from the borrower's position
/// @param _receiver The address to receive the Collateral Token transferred
/// @param _borrower The borrower whose account will be debited the Collateral amount
function _removeCollateral(uint256 _collateralAmount, address _receiver, address _borrower) internal {
// NOTE: violates checks-effects-interactions pattern. Mark function NONREENTRANT
IConvexStakingWrapperFraxlend _checkPointContract = checkPointContract;
_checkPointContract.user_checkpoint(_borrower);
// Effects: write to state
// NOTE: Following line will revert on underflow if _collateralAmount > userCollateralBalance
userCollateralBalance[_borrower] -= _collateralAmount;
// NOTE: Following line will revert on underflow if totalCollateral < _collateralAmount
totalCollateral -= _collateralAmount;
// Interactions
if (_receiver != address(this)) {
_checkPointContract.withdrawAndUnwrap(_collateralAmount);
collateralContract.safeTransfer(_receiver, _collateralAmount);
}
emit RemoveCollateral(msg.sender, _collateralAmount, _receiver, _borrower);
}
/// @notice The ```removeCollateral``` function is used to remove collateral from msg.sender's borrow position
/// @dev msg.sender must be solvent after invocation or transaction will revert
/// @param _collateralAmount The amount of Collateral Token to transfer
/// @param _receiver The address to receive the transferred funds
function removeCollateral(
uint256 _collateralAmount,
address _receiver
) external nonReentrant isSolvent(msg.sender) {
if (_receiver == address(0)) revert InvalidReceiver();
_addInterest();
// Note: exchange rate is irrelevant when borrower has no debt shares
if (userBorrowShares[msg.sender] > 0) {
(bool _isBorrowAllowed, , ) = _updateExchangeRate();
if (!_isBorrowAllowed) revert ExceedsMaxOracleDeviation();
}
_removeCollateral(_collateralAmount, _receiver, msg.sender);
}
/// @notice The ```RepayAsset``` event is emitted whenever a debt position is repaid
/// @param payer The address paying for the repayment
/// @param borrower The borrower whose account will be credited
/// @param amountToRepay The amount of Asset token to be transferred
/// @param shares The amount of Borrow Shares which will be debited from the borrower after repayment
event RepayAsset(address indexed payer, address indexed borrower, uint256 amountToRepay, uint256 shares);
/// @notice The ```_repayAsset``` function is the internal implementation for repaying a borrow position
/// @dev The payer must have called ERC20.approve() on the Asset Token contract prior to invocation
/// @param _totalBorrow An in memory copy of the totalBorrow VaultAccount struct
/// @param _amountToRepay The amount of Asset Token to transfer
/// @param _shares The number of Borrow Shares the sender is repaying
/// @param _payer The address from which funds will be transferred
/// @param _borrower The borrower account which will be credited
function _repayAsset(
VaultAccount memory _totalBorrow,
uint128 _amountToRepay,
uint128 _shares,
address _payer,
address _borrower
) internal {
// Effects: Bookkeeping
_totalBorrow.amount -= _amountToRepay;
_totalBorrow.shares -= _shares;
// Effects: write to state
userBorrowShares[_borrower] -= _shares;
totalBorrow = _totalBorrow;
// Interactions
if (_payer != address(this)) {
assetContract.safeTransferFrom(_payer, address(this), _amountToRepay);
}
emit RepayAsset(_payer, _borrower, _amountToRepay, _shares);
}
/// @notice The ```repayAsset``` function allows the caller to pay down the debt for a given borrower.
/// @dev Caller must first invoke ```ERC20.approve()``` for the Asset Token contract
/// @param _shares The number of Borrow Shares which will be repaid by the call
/// @param _borrower The account for which the debt will be reduced
/// @return _amountToRepay The amount of Asset Tokens which were transferred in order to repay the Borrow Shares
function repayAsset(uint256 _shares, address _borrower) external nonReentrant returns (uint256 _amountToRepay) {
if (_borrower == address(0)) revert InvalidReceiver();
// Check if repay is paused revert if necessary
if (isRepayPaused) revert RepayPaused();
// Accrue interest if necessary
_addInterest();
// Calculate amount to repay based on shares
VaultAccount memory _totalBorrow = totalBorrow;
_amountToRepay = _totalBorrow.toAmount(_shares, true);
// Execute repayment effects
_repayAsset(_totalBorrow, _amountToRepay.toUint128(), _shares.toUint128(), msg.sender, _borrower);
}
// ============================================================================================
// Functions: Liquidations
// ============================================================================================
/// @notice The ```Liquidate``` event is emitted when a liquidation occurs
/// @param _borrower The borrower account for which the liquidation occurred
/// @param _collateralForLiquidator The amount of Collateral Token transferred to the liquidator
/// @param _sharesToLiquidate The number of Borrow Shares the liquidator repaid on behalf of the borrower
/// @param _sharesToAdjust The number of Borrow Shares that were adjusted on liabilities and assets (a writeoff)
event Liquidate(
address indexed _borrower,
uint256 _collateralForLiquidator,
uint256 _sharesToLiquidate,
uint256 _amountLiquidatorToRepay,
uint256 _feesAmount,
uint256 _sharesToAdjust,
uint256 _amountToAdjust
);
/// @notice The ```liquidate``` function allows a third party to repay a borrower's debt if they have become insolvent
/// @dev Caller must invoke ```ERC20.approve``` on the Asset Token contract prior to calling ```Liquidate()```
/// @param _sharesToLiquidate The number of Borrow Shares repaid by the liquidator
/// @param _deadline The timestamp after which tx will revert
/// @param _borrower The account for which the repayment is credited and from whom collateral will be taken
/// @return _collateralForLiquidator The amount of Collateral Token transferred to the liquidator
function liquidate(
uint128 _sharesToLiquidate,
uint256 _deadline,
address _borrower
) external nonReentrant returns (uint256 _collateralForLiquidator) {
if (_borrower == address(0)) revert InvalidReceiver();
// Check if liquidate is paused revert if necessary
if (isLiquidatePaused) revert LiquidatePaused();
// Ensure deadline has not passed
if (block.timestamp > _deadline) revert PastDeadline(block.timestamp, _deadline);
// accrue interest if necessary
_addInterest();
// Update exchange rate and use the lower rate for liquidations
(, uint256 _exchangeRate, ) = _updateExchangeRate();
// Check if borrower is solvent, revert if they are
if (_isSolvent(_borrower, _exchangeRate)) {
revert BorrowerSolvent();
}
// Read from state
VaultAccount memory _totalBorrow = totalBorrow;
uint256 _userCollateralBalance = userCollateralBalance[_borrower];
uint128 _borrowerShares = userBorrowShares[_borrower].toUint128();
// Prevent stack-too-deep
int256 _leftoverCollateral;
uint256 _feesAmount;
{
// Checks & Calculations
// Determine the liquidation amount in collateral units (i.e. how much debt liquidator is going to repay)
uint256 _liquidationAmountInCollateralUnits = ((_totalBorrow.toAmount(_sharesToLiquidate, false) *
_exchangeRate) / EXCHANGE_PRECISION);
// We first optimistically calculate the amount of collateral to give the liquidator based on the higher clean liquidation fee
// This fee only applies if the liquidator does a full liquidation
uint256 _optimisticCollateralForLiquidator = (_liquidationAmountInCollateralUnits *
(LIQ_PRECISION + cleanLiquidationFee)) / LIQ_PRECISION;
// Because interest accrues every block, _liquidationAmountInCollateralUnits from a few lines up is an ever increasing value
// This means that leftoverCollateral can occasionally go negative by a few hundred wei (cleanLiqFee premium covers this for liquidator)
_leftoverCollateral = (_userCollateralBalance.toInt256() - _optimisticCollateralForLiquidator.toInt256());
// If cleanLiquidation fee results in no leftover collateral, give liquidator all the collateral
// This will only be true when there liquidator is cleaning out the position
_collateralForLiquidator = _leftoverCollateral <= 0
? _userCollateralBalance
: (_liquidationAmountInCollateralUnits * (LIQ_PRECISION + dirtyLiquidationFee)) / LIQ_PRECISION;
if (protocolLiquidationFee > 0) {
_feesAmount = (protocolLiquidationFee * _collateralForLiquidator) / LIQ_PRECISION;
_collateralForLiquidator = _collateralForLiquidator - _feesAmount;
}
}
// Calculated here for use during repayment, grouped with other calcs before effects start
uint128 _amountLiquidatorToRepay = (_totalBorrow.toAmount(_sharesToLiquidate, true)).toUint128();
// Determine if and how much debt to adjust
uint128 _sharesToAdjust = 0;
{
uint128 _amountToAdjust = 0;
if (_leftoverCollateral <= 0) {
// Determine if we need to adjust any shares
_sharesToAdjust = _borrowerShares - _sharesToLiquidate;
if (_sharesToAdjust > 0) {
// Write off bad debt
_amountToAdjust = (_totalBorrow.toAmount(_sharesToAdjust, false)).toUint128();
// Note: Ensure this memory struct will be passed to _repayAsset for write to state
_totalBorrow.amount -= _amountToAdjust;
// Effects: write to state
totalAsset.amount -= _amountToAdjust;
}
}
emit Liquidate(
_borrower,
_collateralForLiquidator,
_sharesToLiquidate,
_amountLiquidatorToRepay,
_feesAmount,
_sharesToAdjust,
_amountToAdjust
);
}
// Effects & Interactions
// NOTE: reverts if _shares > userBorrowShares
_repayAsset(
_totalBorrow,
_amountLiquidatorToRepay,
_sharesToLiquidate + _sharesToAdjust,
msg.sender,
_borrower
); // liquidator repays shares on behalf of borrower
// NOTE: reverts if _collateralForLiquidator > userCollateralBalance
// Collateral is removed on behalf of borrower and sent to liquidator
// NOTE: reverts if _collateralForLiquidator > userCollateralBalance
_removeCollateral(_collateralForLiquidator, msg.sender, _borrower);
// Adjust bookkeeping only (decreases collateral held by borrower)
_removeCollateral(_feesAmount, address(this), _borrower);
// Adjusts bookkeeping only (increases collateral held by protocol)
_addCollateral(address(this), _feesAmount, address(this));
}
// ============================================================================================
// Functions: Leverage
// ============================================================================================
/// @notice The ```LeveragedPosition``` event is emitted when a borrower takes out a new leveraged position
/// @param _borrower The account for which the debt is debited
/// @param _swapperAddress The address of the swapper which conforms the FraxSwap interface
/// @param _borrowAmount The amount of Asset Token to be borrowed to be borrowed
/// @param _borrowShares The number of Borrow Shares the borrower is credited
/// @param _initialCollateralAmount The amount of initial Collateral Tokens supplied by the borrower
/// @param _amountCollateralOut The amount of Collateral Token which was received for the Asset Tokens
event LeveragedPosition(
address indexed _borrower,
address _swapperAddress,
uint256 _borrowAmount,
uint256 _borrowShares,
uint256 _initialCollateralAmount,
uint256 _amountCollateralOut
);
/// @notice The ```leveragedPosition``` function allows a user to enter a leveraged borrow position with minimal upfront Collateral
/// @dev Caller must invoke ```ERC20.approve()``` on the Collateral Token contract prior to calling function
/// @param _swapperAddress The address of the whitelisted swapper to use to swap borrowed Asset Tokens for Collateral Tokens
/// @param _borrowAmount The amount of Asset Tokens borrowed
/// @param _initialCollateralAmount The initial amount of Collateral Tokens supplied by the borrower
/// @param _amountCollateralOutMin The minimum amount of Collateral Tokens to be received in exchange for the borrowed Asset Tokens
/// @param _path An array containing the addresses of ERC20 tokens to swap. Adheres to UniV2 style path params.
/// @return _totalCollateralBalance The total amount of Collateral Tokens added to a users account (initial + swap)
function leveragedPosition(
address _swapperAddress,
uint256 _borrowAmount,
uint256 _initialCollateralAmount,
uint256 _amountCollateralOutMin,
address[] memory _path
) external nonReentrant isSolvent(msg.sender) returns (uint256 _totalCollateralBalance) {
// Accrue interest if necessary
_addInterest();
// Update exchange rate and check if borrow is allowed, revert if not
{
(bool _isBorrowAllowed, , ) = _updateExchangeRate();
if (!_isBorrowAllowed) revert ExceedsMaxOracleDeviation();
}
IERC20 _assetContract = assetContract;
IERC20 _collateralContract = collateralContract;
if (!swappers[_swapperAddress]) {
revert BadSwapper();
}
if (_path[0] != address(_assetContract)) {
revert InvalidPath(address(_assetContract), _path[0]);
}
if (_path[_path.length - 1] != address(_collateralContract)) {
revert InvalidPath(address(_collateralContract), _path[_path.length - 1]);
}
// Add initial collateral
if (_initialCollateralAmount > 0) {
_addCollateral(msg.sender, _initialCollateralAmount, msg.sender);
}
// Debit borrowers account
// setting recipient to address(this) means no transfer will happen
uint256 _borrowShares = _borrowAsset(_borrowAmount.toUint128(), address(this));
// Interactions
_assetContract.approve(_swapperAddress, _borrowAmount);
// Even though swappers are trusted, we verify the balance before and after swap
uint256 _initialCollateralBalance = _collateralContract.balanceOf(address(this));
ISwapper(_swapperAddress).swapExactTokensForTokens(
_borrowAmount,
_amountCollateralOutMin,
_path,
address(this),
block.timestamp
);
uint256 _finalCollateralBalance = _collateralContract.balanceOf(address(this));
// Note: VIOLATES CHECKS-EFFECTS-INTERACTION pattern, make sure function is NONREENTRANT
// Effects: bookkeeping & write to state
uint256 _amountCollateralOut = _finalCollateralBalance - _initialCollateralBalance;
if (_amountCollateralOut < _amountCollateralOutMin) {
revert SlippageTooHigh(_amountCollateralOutMin, _amountCollateralOut);
}
// address(this) as _sender means no transfer occurs as the pair has already received the collateral during swap
_addCollateral(address(this), _amountCollateralOut, msg.sender);
_totalCollateralBalance = _initialCollateralAmount + _amountCollateralOut;
emit LeveragedPosition(
msg.sender,
_swapperAddress,
_borrowAmount,
_borrowShares,
_initialCollateralAmount,
_amountCollateralOut
);
}
/// @notice The ```RepayAssetWithCollateral``` event is emitted whenever ```repayAssetWithCollateral()``` is invoked
/// @param _borrower The borrower account for which the repayment is taking place
/// @param _swapperAddress The address of the whitelisted swapper to use for token swaps
/// @param _collateralToSwap The amount of Collateral Token to swap and use for repayment
/// @param _amountAssetOut The amount of Asset Token which was repaid
/// @param _sharesRepaid The number of Borrow Shares which were repaid
event RepayAssetWithCollateral(
address indexed _borrower,
address _swapperAddress,
uint256 _collateralToSwap,
uint256 _amountAssetOut,
uint256 _sharesRepaid
);
/// @notice The ```repayAssetWithCollateral``` function allows a borrower to repay their debt using existing collateral in contract
/// @param _swapperAddress The address of the whitelisted swapper to use for token swaps
/// @param _collateralToSwap The amount of Collateral Tokens to swap for Asset Tokens
/// @param _amountAssetOutMin The minimum amount of Asset Tokens to receive during the swap
/// @param _path An array containing the addresses of ERC20 tokens to swap. Adheres to UniV2 style path params.
/// @return _amountAssetOut The amount of Asset Tokens received for the Collateral Tokens, the amount the borrowers account was credited
function repayAssetWithCollateral(
address _swapperAddress,
uint256 _collateralToSwap,
uint256 _amountAssetOutMin,
address[] calldata _path
) external nonReentrant isSolvent(msg.sender) returns (uint256 _amountAssetOut) {
// Accrue interest if necessary
_addInterest();
// Update exchange rate and check if borrow is allowed, revert if not
(bool _isBorrowAllowed, , ) = _updateExchangeRate();
if (!_isBorrowAllowed) revert ExceedsMaxOracleDeviation();
IERC20 _assetContract = assetContract;
IERC20 _collateralContract = collateralContract;
if (!swappers[_swapperAddress]) {
revert BadSwapper();
}
if (_path[0] != address(_collateralContract)) {
revert InvalidPath(address(_collateralContract), _path[0]);
}
if (_path[_path.length - 1] != address(_assetContract)) {
revert InvalidPath(address(_assetContract), _path[_path.length - 1]);
}
// Effects: bookkeeping & write to state
// Debit users collateral balance in preparation for swap, setting _recipient to address(this) means no transfer occurs
_removeCollateral(_collateralToSwap, address(this), msg.sender);
// Interactions
_collateralContract.approve(_swapperAddress, _collateralToSwap);
// Even though swappers are trusted, we verify the balance before and after swap
uint256 _initialAssetBalance = _assetContract.balanceOf(address(this));
ISwapper(_swapperAddress).swapExactTokensForTokens(
_collateralToSwap,
_amountAssetOutMin,
_path,
address(this),
block.timestamp
);
uint256 _finalAssetBalance = _assetContract.balanceOf(address(this));
// Note: VIOLATES CHECKS-EFFECTS-INTERACTION pattern, make sure function is NONREENTRANT
// Effects: bookkeeping
_amountAssetOut = _finalAssetBalance - _initialAssetBalance;
if (_amountAssetOut < _amountAssetOutMin) {
revert SlippageTooHigh(_amountAssetOutMin, _amountAssetOut);
}
VaultAccount memory _totalBorrow = totalBorrow;
uint256 _sharesToRepay = _totalBorrow.toShares(_amountAssetOut, false);
// Effects: write to state
// Note: setting _payer to address(this) means no actual transfer will occur. Contract already has funds
_repayAsset(_totalBorrow, _amountAssetOut.toUint128(), _sharesToRepay.toUint128(), address(this), msg.sender);
emit RepayAssetWithCollateral(msg.sender, _swapperAddress, _collateralToSwap, _amountAssetOut, _sharesToRepay);
}
}// SPDX-License-Identifier: ISC
pragma solidity ^0.8.19;
// ====================================================================
// | ______ _______ |
// | / _____________ __ __ / ____(_____ ____ _____ ________ |
// | / /_ / ___/ __ `| |/_/ / /_ / / __ \/ __ `/ __ \/ ___/ _ \ |
// | / __/ / / / /_/ _> < / __/ / / / / / /_/ / / / / /__/ __/ |
// | /_/ /_/ \__,_/_/|_| /_/ /_/_/ /_/\__,_/_/ /_/\___/\___/ |
// | |
// ====================================================================
// ========================== Timelock2Step ===========================
// ====================================================================
// Frax Finance: https://github.com/FraxFinance
// Primary Author
// Drake Evans: https://github.com/DrakeEvans
// Reviewers
// Dennis: https://github.com/denett
// ====================================================================
/// @title Timelock2Step
/// @author Drake Evans (Frax Finance) https://github.com/drakeevans
/// @dev Inspired by the OpenZeppelin's Ownable2Step contract
/// @notice An abstract contract which contains 2-step transfer and renounce logic for a timelock address
abstract contract Timelock2Step {
/// @notice The pending timelock address
address public pendingTimelockAddress;
/// @notice The current timelock address
address public timelockAddress;
constructor() {
timelockAddress = msg.sender;
}
/// @notice Emitted when timelock is transferred
error OnlyTimelock();
/// @notice Emitted when pending timelock is transferred
error OnlyPendingTimelock();
/// @notice The ```TimelockTransferStarted``` event is emitted when the timelock transfer is initiated
/// @param previousTimelock The address of the previous timelock
/// @param newTimelock The address of the new timelock
event TimelockTransferStarted(address indexed previousTimelock, address indexed newTimelock);
/// @notice The ```TimelockTransferred``` event is emitted when the timelock transfer is completed
/// @param previousTimelock The address of the previous timelock
/// @param newTimelock The address of the new timelock
event TimelockTransferred(address indexed previousTimelock, address indexed newTimelock);
/// @notice The ```_isSenderTimelock``` function checks if msg.sender is current timelock address
/// @return Whether or not msg.sender is current timelock address
function _isSenderTimelock() internal view returns (bool) {
return msg.sender == timelockAddress;
}
/// @notice The ```_requireTimelock``` function reverts if msg.sender is not current timelock address
function _requireTimelock() internal view {
if (msg.sender != timelockAddress) revert OnlyTimelock();
}
/// @notice The ```_isSenderPendingTimelock``` function checks if msg.sender is pending timelock address
/// @return Whether or not msg.sender is pending timelock address
function _isSenderPendingTimelock() internal view returns (bool) {
return msg.sender == pendingTimelockAddress;
}
/// @notice The ```_requirePendingTimelock``` function reverts if msg.sender is not pending timelock address
function _requirePendingTimelock() internal view {
if (msg.sender != pendingTimelockAddress) revert OnlyPendingTimelock();
}
/// @notice The ```_transferTimelock``` function initiates the timelock transfer
/// @dev This function is to be implemented by a public function
/// @param _newTimelock The address of the nominated (pending) timelock
function _transferTimelock(address _newTimelock) internal {
pendingTimelockAddress = _newTimelock;
emit TimelockTransferStarted(timelockAddress, _newTimelock);
}
/// @notice The ```_acceptTransferTimelock``` function completes the timelock transfer
/// @dev This function is to be implemented by a public function
function _acceptTransferTimelock() internal {
pendingTimelockAddress = address(0);
_setTimelock(msg.sender);
}
/// @notice The ```_setTimelock``` function sets the timelock address
/// @dev This function is to be implemented by a public function
/// @param _newTimelock The address of the new timelock
function _setTimelock(address _newTimelock) internal {
emit TimelockTransferred(timelockAddress, _newTimelock);
timelockAddress = _newTimelock;
}
/// @notice The ```transferTimelock``` function initiates the timelock transfer
/// @dev Must be called by the current timelock
/// @param _newTimelock The address of the nominated (pending) timelock
function transferTimelock(address _newTimelock) external virtual {
_requireTimelock();
_transferTimelock(_newTimelock);
}
/// @notice The ```acceptTransferTimelock``` function completes the timelock transfer
/// @dev Must be called by the pending timelock
function acceptTransferTimelock() external virtual {
_requirePendingTimelock();
_acceptTransferTimelock();
}
/// @notice The ```renounceTimelock``` function renounces the timelock after setting pending timelock to current timelock
/// @dev Pending timelock must be set to current timelock before renouncing, creating a 2-step renounce process
function renounceTimelock() external virtual {
_requireTimelock();
_requirePendingTimelock();
_transferTimelock(address(0));
_setTimelock(address(0));
}
}// SPDX-License-Identifier: ISC
pragma solidity ^0.8.19;
// NOTE: This is a generated file. Do not edit directly.
interface IConvexStakingWrapperFraxlend {
event Approval(address indexed owner, address indexed spender, uint256 value);
event Deposited(address indexed _user, address indexed _account, uint256 _amount, bool _wrapped);
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
event Transfer(address indexed from, address indexed to, uint256 value);
event Withdrawn(address indexed _user, uint256 _amount, bool _unwrapped);
struct EarnedData {
address token;
uint256 amount;
}
function addRewards() external;
function addTokenReward(address _token) external;
function allowance(address owner, address spender) external view returns (uint256);
function approve(address spender, uint256 amount) external returns (bool);
function balanceOf(address account) external view returns (uint256);
function collateralVault() external view returns (address);
function convexBooster() external view returns (address);
function convexPool() external view returns (address);
function convexPoolId() external view returns (uint256);
function convexToken() external view returns (address);
function crv() external view returns (address);
function curveToken() external view returns (address);
function cvx() external view returns (address);
function decimals() external view returns (uint8);
function decreaseAllowance(address spender, uint256 subtractedValue) external returns (bool);
function deposit(uint256 _amount, address _to) external;
function earned(address _account) external returns (IConvexStakingWrapperFraxlend.EarnedData[] memory claimable);
function earnedView(
address _account
) external view returns (IConvexStakingWrapperFraxlend.EarnedData[] memory claimable);
function getReward(address _account, address _forwardTo) external;
function getReward(address _account) external;
function increaseAllowance(address spender, uint256 addedValue) external returns (bool);
function initialize(uint256 _poolId) external;
function isInit() external view returns (bool);
function isShutdown() external view returns (bool);
function name() external view returns (string memory);
function owner() external view returns (address);
function registeredRewards(address) external view returns (uint256);
function renounceOwnership() external;
function rewardHook() external view returns (address);
function rewardLength() external view returns (uint256);
function rewards(
uint256
)
external
view
returns (address reward_token, address reward_pool, uint128 reward_integral, uint128 reward_remaining);
function setApprovals() external;
function setHook(address _hook) external;
function setVault(address _vault) external;
function shutdown() external;
function stake(uint256 _amount, address _to) external;
function symbol() external view returns (string memory);
function totalBalanceOf(address _account) external view returns (uint256);
function totalSupply() external view returns (uint256);
function transfer(address recipient, uint256 amount) external returns (bool);
function transferFrom(address sender, address recipient, uint256 amount) external returns (bool);
function transferOwnership(address newOwner) external;
function user_checkpoint(address _account) external returns (bool);
function withdraw(uint256 _amount) external;
function withdrawAndUnwrap(uint256 _amount) external;
}// SPDX-License-Identifier: UNLICENSED
pragma solidity 0.8.19;
import { IERC165 } from "@openzeppelin/contracts/utils/introspection/IERC165.sol";
interface IDualOracle is IERC165 {
function ORACLE_PRECISION() external view returns (uint256);
function BASE_TOKEN_0() external view returns (address);
function BASE_TOKEN_0_DECIMALS() external view returns (uint256);
function BASE_TOKEN_1() external view returns (address);
function BASE_TOKEN_1_DECIMALS() external view returns (uint256);
function decimals() external view returns (uint8);
function getPricesNormalized() external view returns (bool _isBadData, uint256 _priceLow, uint256 _priceHigh);
function getPrices() external view returns (bool _isBadData, uint256 _priceLow, uint256 _priceHigh);
function name() external view returns (string memory);
function NORMALIZATION_0() external view returns (int256);
function NORMALIZATION_1() external view returns (int256);
function QUOTE_TOKEN_0() external view returns (address);
function QUOTE_TOKEN_0_DECIMALS() external view returns (uint256);
function QUOTE_TOKEN_1() external view returns (address);
function QUOTE_TOKEN_1_DECIMALS() external view returns (uint256);
}// SPDX-License-Identifier: ISC
pragma solidity ^0.8.19;
interface IRateCalculatorV2 {
function name() external view returns (string memory);
function version() external view returns (uint256, uint256, uint256);
function getNewRate(
uint256 _deltaTime,
uint256 _utilization,
uint64 _maxInterest
) external view returns (uint64 _newRatePerSec, uint64 _newMaxInterest);
}// SPDX-License-Identifier: MIT
pragma solidity >=0.8.19;
interface ISwapper {
function swapExactTokensForTokens(
uint256 amountIn,
uint256 amountOutMin,
address[] calldata path,
address to,
uint256 deadline
) external returns (uint256[] memory amounts);
}// SPDX-License-Identifier: ISC
pragma solidity ^0.8.19;
import { IERC20 } from "@openzeppelin/contracts/interfaces/IERC20.sol";
import { SafeERC20 as OZSafeERC20 } from "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol";
// solhint-disable avoid-low-level-calls
// solhint-disable max-line-length
/// @title SafeERC20 provides helper functions for safe transfers as well as safe metadata access
/// @author Library originally written by @Boring_Crypto github.com/boring_crypto, modified by Drake Evans (Frax Finance) github.com/drakeevans
/// @dev original: https://github.com/boringcrypto/BoringSolidity/blob/fed25c5d43cb7ce20764cd0b838e21a02ea162e9/contracts/libraries/BoringERC20.sol
library SafeERC20 {
bytes4 private constant SIG_SYMBOL = 0x95d89b41; // symbol()
bytes4 private constant SIG_NAME = 0x06fdde03; // name()
bytes4 private constant SIG_DECIMALS = 0x313ce567; // decimals()
function returnDataToString(bytes memory data) internal pure returns (string memory) {
if (data.length >= 64) {
return abi.decode(data, (string));
} else if (data.length == 32) {
uint8 i = 0;
while (i < 32 && data[i] != 0) {
i++;
}
bytes memory bytesArray = new bytes(i);
for (i = 0; i < 32 && data[i] != 0; i++) {
bytesArray[i] = data[i];
}
return string(bytesArray);
} else {
return "???";
}
}
/// @notice Provides a safe ERC20.symbol version which returns '???' as fallback string.
/// @param token The address of the ERC-20 token contract.
/// @return (string) Token symbol.
function safeSymbol(IERC20 token) internal view returns (string memory) {
(bool success, bytes memory data) = address(token).staticcall(abi.encodeWithSelector(SIG_SYMBOL));
return success ? returnDataToString(data) : "???";
}
/// @notice Provides a safe ERC20.name version which returns '???' as fallback string.
/// @param token The address of the ERC-20 token contract.
/// @return (string) Token name.
function safeName(IERC20 token) internal view returns (string memory) {
(bool success, bytes memory data) = address(token).staticcall(abi.encodeWithSelector(SIG_NAME));
return success ? returnDataToString(data) : "???";
}
/// @notice Provides a safe ERC20.decimals version which returns '18' as fallback value.
/// @param token The address of the ERC-20 token contract.
/// @return (uint8) Token decimals.
function safeDecimals(IERC20 token) internal view returns (uint8) {
(bool success, bytes memory data) = address(token).staticcall(abi.encodeWithSelector(SIG_DECIMALS));
return success && data.length == 32 ? abi.decode(data, (uint8)) : 18;
}
function safeTransfer(IERC20 token, address to, uint256 value) internal {
OZSafeERC20.safeTransfer(token, to, value);
}
function safeTransferFrom(IERC20 token, address from, address to, uint256 value) internal {
OZSafeERC20.safeTransferFrom(token, from, to, value);
}
}// SPDX-License-Identifier: ISC
pragma solidity ^0.8.19;
struct VaultAccount {
uint128 amount; // Total amount, analogous to market cap
uint128 shares; // Total shares, analogous to shares outstanding
}
/// @title VaultAccount Library
/// @author Drake Evans (Frax Finance) github.com/drakeevans, modified from work by @Boring_Crypto github.com/boring_crypto
/// @notice Provides a library for use with the VaultAccount struct, provides convenient math implementations
/// @dev Uses uint128 to save on storage
library VaultAccountingLibrary {
/// @notice Calculates the shares value in relationship to `amount` and `total`
/// @dev Given an amount, return the appropriate number of shares
function toShares(VaultAccount memory total, uint256 amount, bool roundUp) internal pure returns (uint256 shares) {
if (total.amount == 0) {
shares = amount;
} else {
shares = (amount * total.shares) / total.amount;
if (roundUp && (shares * total.amount) / total.shares < amount) {
shares = shares + 1;
}
}
}
/// @notice Calculates the amount value in relationship to `shares` and `total`
/// @dev Given a number of shares, returns the appropriate amount
function toAmount(VaultAccount memory total, uint256 shares, bool roundUp) internal pure returns (uint256 amount) {
if (total.shares == 0) {
amount = shares;
} else {
amount = (shares * total.amount) / total.shares;
if (roundUp && (amount * total.shares) / total.amount < shares) {
amount = amount + 1;
}
}
}
}{
"remappings": [
"@chainlink/=node_modules/@chainlink/",
"@ensdomains/=node_modules/@ensdomains/",
"@eth-optimism/=node_modules/@eth-optimism/",
"@mean-finance/=node_modules/@mean-finance/",
"@openzeppelin/=node_modules/@openzeppelin/",
"@rari-capital/=node_modules/@rari-capital/",
"@uniswap/=node_modules/@uniswap/",
"base64-sol/=node_modules/base64-sol/",
"deploy/=deploy/",
"ds-test/=lib/frax-standard-solidity/lib/forge-std/lib/ds-test/src/",
"eth-gas-reporter/=node_modules/eth-gas-reporter/",
"forge-std/=lib/frax-standard-solidity/lib/forge-std/src/",
"frax-standard-solidity/=lib/frax-standard-solidity/src/",
"frax-std/=lib/frax-standard-solidity/src/",
"hardhat/=node_modules/hardhat/",
"interfaces/=src/interfaces/",
"script/=script/",
"solidity-bytes-utils/=node_modules/solidity-bytes-utils/",
"src/=src/"
],
"optimizer": {
"enabled": true,
"runs": 830
},
"metadata": {
"bytecodeHash": "none",
"appendCBOR": true
},
"outputSelection": {
"*": {
"*": [
"evm.bytecode",
"evm.deployedBytecode",
"devdoc",
"userdoc",
"metadata",
"abi"
]
}
},
"evmVersion": "london",
"viaIR": true,
"libraries": {}
}Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
Contract ABI
API[{"inputs":[{"internalType":"bytes","name":"_configData","type":"bytes"},{"internalType":"bytes","name":"_immutables","type":"bytes"},{"internalType":"bytes","name":"_customConfigData","type":"bytes"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"AccessControlRevoked","type":"error"},{"inputs":[],"name":"BadProtocolFee","type":"error"},{"inputs":[],"name":"BadSwapper","type":"error"},{"inputs":[],"name":"BorrowerSolvent","type":"error"},{"inputs":[],"name":"ExceedsBorrowLimit","type":"error"},{"inputs":[],"name":"ExceedsDepositLimit","type":"error"},{"inputs":[],"name":"ExceedsMaxOracleDeviation","type":"error"},{"inputs":[{"internalType":"uint256","name":"_borrow","type":"uint256"},{"internalType":"uint256","name":"_collateral","type":"uint256"},{"internalType":"uint256","name":"_exchangeRate","type":"uint256"}],"name":"Insolvent","type":"error"},{"inputs":[{"internalType":"uint256","name":"_assets","type":"uint256"},{"internalType":"uint256","name":"_request","type":"uint256"}],"name":"InsufficientAssetsInContract","type":"error"},{"inputs":[],"name":"InterestPaused","type":"error"},{"inputs":[{"internalType":"address","name":"_expected","type":"address"},{"internalType":"address","name":"_actual","type":"address"}],"name":"InvalidPath","type":"error"},{"inputs":[],"name":"InvalidReceiver","type":"error"},{"inputs":[],"name":"LiquidatePaused","type":"error"},{"inputs":[],"name":"OnlyPendingTimelock","type":"error"},{"inputs":[],"name":"OnlyProtocolOrOwner","type":"error"},{"inputs":[],"name":"OnlyTimelock","type":"error"},{"inputs":[],"name":"OnlyTimelockOrOwner","type":"error"},{"inputs":[{"internalType":"uint256","name":"_blockTimestamp","type":"uint256"},{"internalType":"uint256","name":"_deadline","type":"uint256"}],"name":"PastDeadline","type":"error"},{"inputs":[],"name":"RepayPaused","type":"error"},{"inputs":[],"name":"SetterRevoked","type":"error"},{"inputs":[{"internalType":"uint256","name":"_minOut","type":"uint256"},{"internalType":"uint256","name":"_actual","type":"uint256"}],"name":"SlippageTooHigh","type":"error"},{"inputs":[],"name":"WithdrawPaused","type":"error"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"sender","type":"address"},{"indexed":true,"internalType":"address","name":"borrower","type":"address"},{"indexed":false,"internalType":"uint256","name":"collateralAmount","type":"uint256"}],"name":"AddCollateral","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"interestEarned","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"rate","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"feesAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"feesShare","type":"uint256"}],"name":"AddInterest","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"spender","type":"address"},{"indexed":false,"internalType":"uint256","name":"value","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"_borrower","type":"address"},{"indexed":true,"internalType":"address","name":"_receiver","type":"address"},{"indexed":false,"internalType":"uint256","name":"_borrowAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_sharesAdded","type":"uint256"}],"name":"BorrowAsset","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint32","name":"newFee","type":"uint32"}],"name":"ChangeFee","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"caller","type":"address"},{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":false,"internalType":"uint256","name":"assets","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"shares","type":"uint256"}],"name":"Deposit","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"_borrower","type":"address"},{"indexed":false,"internalType":"address","name":"_swapperAddress","type":"address"},{"indexed":false,"internalType":"uint256","name":"_borrowAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_borrowShares","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_initialCollateralAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_amountCollateralOut","type":"uint256"}],"name":"LeveragedPosition","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"_borrower","type":"address"},{"indexed":false,"internalType":"uint256","name":"_collateralForLiquidator","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_sharesToLiquidate","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_amountLiquidatorToRepay","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_feesAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_sharesToAdjust","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_amountToAdjust","type":"uint256"}],"name":"Liquidate","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferStarted","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":false,"internalType":"bool","name":"isPaused","type":"bool"}],"name":"PauseInterest","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"bool","name":"isPaused","type":"bool"}],"name":"PauseLiquidate","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"bool","name":"isPaused","type":"bool"}],"name":"PauseRepay","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"bool","name":"isPaused","type":"bool"}],"name":"PauseWithdraw","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"_sender","type":"address"},{"indexed":false,"internalType":"uint256","name":"_collateralAmount","type":"uint256"},{"indexed":true,"internalType":"address","name":"_receiver","type":"address"},{"indexed":true,"internalType":"address","name":"_borrower","type":"address"}],"name":"RemoveCollateral","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"payer","type":"address"},{"indexed":true,"internalType":"address","name":"borrower","type":"address"},{"indexed":false,"internalType":"uint256","name":"amountToRepay","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"shares","type":"uint256"}],"name":"RepayAsset","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"_borrower","type":"address"},{"indexed":false,"internalType":"address","name":"_swapperAddress","type":"address"},{"indexed":false,"internalType":"uint256","name":"_collateralToSwap","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_amountAssetOut","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_sharesRepaid","type":"uint256"}],"name":"RepayAssetWithCollateral","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"borrowLimit","type":"uint256"}],"name":"RevokeBorrowAccessControl","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"depositLimit","type":"uint256"}],"name":"RevokeDepositAccessControl","type":"event"},{"anonymous":false,"inputs":[],"name":"RevokeInterestAccessControl","type":"event"},{"anonymous":false,"inputs":[],"name":"RevokeLiquidateAccessControl","type":"event"},{"anonymous":false,"inputs":[],"name":"RevokeLiquidationFeeSetter","type":"event"},{"anonymous":false,"inputs":[],"name":"RevokeMaxLTVSetter","type":"event"},{"anonymous":false,"inputs":[],"name":"RevokeOracleInfoSetter","type":"event"},{"anonymous":false,"inputs":[],"name":"RevokeRateContractSetter","type":"event"},{"anonymous":false,"inputs":[],"name":"RevokeRepayAccessControl","type":"event"},{"anonymous":false,"inputs":[],"name":"RevokeWithdrawAccessControl","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"limit","type":"uint256"}],"name":"SetBorrowLimit","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"oldCircuitBreaker","type":"address"},{"indexed":false,"internalType":"address","name":"newCircuitBreaker","type":"address"}],"name":"SetCircuitBreaker","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"limit","type":"uint256"}],"name":"SetDepositLimit","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"oldCleanLiquidationFee","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"oldDirtyLiquidationFee","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"oldProtocolLiquidationFee","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"newCleanLiquidationFee","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"newDirtyLiquidationFee","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"newProtocolLiquidationFee","type":"uint256"}],"name":"SetLiquidationFees","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"oldMaxLTV","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"newMaxLTV","type":"uint256"}],"name":"SetMaxLTV","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"oldOracle","type":"address"},{"indexed":false,"internalType":"uint32","name":"oldMaxOracleDeviation","type":"uint32"},{"indexed":false,"internalType":"address","name":"newOracle","type":"address"},{"indexed":false,"internalType":"uint32","name":"newMaxOracleDeviation","type":"uint32"}],"name":"SetOracleInfo","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"oldRateContract","type":"address"},{"indexed":false,"internalType":"address","name":"newRateContract","type":"address"}],"name":"SetRateContract","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"swapper","type":"address"},{"indexed":false,"internalType":"bool","name":"approval","type":"bool"}],"name":"SetSwapper","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousTimelock","type":"address"},{"indexed":true,"internalType":"address","name":"newTimelock","type":"address"}],"name":"TimelockTransferStarted","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousTimelock","type":"address"},{"indexed":true,"internalType":"address","name":"newTimelock","type":"address"}],"name":"TimelockTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":false,"internalType":"uint256","name":"value","type":"uint256"}],"name":"Transfer","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"lowExchangeRate","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"highExchangeRate","type":"uint256"}],"name":"UpdateExchangeRate","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"oldRatePerSec","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"oldFullUtilizationRate","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"newRatePerSec","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"newFullUtilizationRate","type":"uint256"}],"name":"UpdateRate","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"oracle","type":"address"}],"name":"WarnOracleData","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"caller","type":"address"},{"indexed":true,"internalType":"address","name":"receiver","type":"address"},{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":false,"internalType":"uint256","name":"assets","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"shares","type":"uint256"}],"name":"Withdraw","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint128","name":"shares","type":"uint128"},{"indexed":false,"internalType":"address","name":"recipient","type":"address"},{"indexed":false,"internalType":"uint256","name":"amountToTransfer","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"collateralAmount","type":"uint256"}],"name":"WithdrawFees","type":"event"},{"inputs":[],"name":"DEPLOYER_ADDRESS","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"DEVIATION_PRECISION","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"EXCHANGE_PRECISION","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"FEE_PRECISION","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"LIQ_PRECISION","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"LTV_PRECISION","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MAX_PROTOCOL_FEE","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"RATE_PRECISION","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"UTIL_PREC","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"acceptOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"acceptTransferTimelock","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_collateralAmount","type":"uint256"},{"internalType":"address","name":"_borrower","type":"address"}],"name":"addCollateral","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"_returnAccounting","type":"bool"}],"name":"addInterest","outputs":[{"internalType":"uint256","name":"_interestEarned","type":"uint256"},{"internalType":"uint256","name":"_feesAmount","type":"uint256"},{"internalType":"uint256","name":"_feesShare","type":"uint256"},{"components":[{"internalType":"uint32","name":"lastBlock","type":"uint32"},{"internalType":"uint32","name":"feeToProtocolRate","type":"uint32"},{"internalType":"uint64","name":"lastTimestamp","type":"uint64"},{"internalType":"uint64","name":"ratePerSec","type":"uint64"},{"internalType":"uint64","name":"fullUtilizationRate","type":"uint64"}],"internalType":"struct FraxlendPairCore.CurrentRateInfo","name":"_currentRateInfo","type":"tuple"},{"components":[{"internalType":"uint128","name":"amount","type":"uint128"},{"internalType":"uint128","name":"shares","type":"uint128"}],"internalType":"struct VaultAccount","name":"_totalAsset","type":"tuple"},{"components":[{"internalType":"uint128","name":"amount","type":"uint128"},{"internalType":"uint128","name":"shares","type":"uint128"}],"internalType":"struct VaultAccount","name":"_totalBorrow","type":"tuple"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"spender","type":"address"}],"name":"allowance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"approve","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"asset","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_borrowAmount","type":"uint256"},{"internalType":"uint256","name":"_collateralAmount","type":"uint256"},{"internalType":"address","name":"_receiver","type":"address"}],"name":"borrowAsset","outputs":[{"internalType":"uint256","name":"_shares","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"borrowLimit","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint32","name":"_newFee","type":"uint32"}],"name":"changeFee","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"checkPointContract","outputs":[{"internalType":"contract IConvexStakingWrapperFraxlend","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"circuitBreakerAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"cleanLiquidationFee","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"collateralContract","outputs":[{"internalType":"contract IERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_shares","type":"uint256"}],"name":"convertToAssets","outputs":[{"internalType":"uint256","name":"_assets","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_assets","type":"uint256"}],"name":"convertToShares","outputs":[{"internalType":"uint256","name":"_shares","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"currentRateInfo","outputs":[{"internalType":"uint32","name":"lastBlock","type":"uint32"},{"internalType":"uint32","name":"feeToProtocolRate","type":"uint32"},{"internalType":"uint64","name":"lastTimestamp","type":"uint64"},{"internalType":"uint64","name":"ratePerSec","type":"uint64"},{"internalType":"uint64","name":"fullUtilizationRate","type":"uint64"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"decimals","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"subtractedValue","type":"uint256"}],"name":"decreaseAllowance","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_amount","type":"uint256"},{"internalType":"address","name":"_receiver","type":"address"}],"name":"deposit","outputs":[{"internalType":"uint256","name":"_sharesReceived","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"depositLimit","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"dirtyLiquidationFee","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"exchangeRateInfo","outputs":[{"internalType":"address","name":"oracle","type":"address"},{"internalType":"uint32","name":"maxOracleDeviation","type":"uint32"},{"internalType":"uint184","name":"lastTimestamp","type":"uint184"},{"internalType":"uint256","name":"lowExchangeRate","type":"uint256"},{"internalType":"uint256","name":"highExchangeRate","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getConstants","outputs":[{"internalType":"uint256","name":"_LTV_PRECISION","type":"uint256"},{"internalType":"uint256","name":"_LIQ_PRECISION","type":"uint256"},{"internalType":"uint256","name":"_UTIL_PREC","type":"uint256"},{"internalType":"uint256","name":"_FEE_PRECISION","type":"uint256"},{"internalType":"uint256","name":"_EXCHANGE_PRECISION","type":"uint256"},{"internalType":"uint256","name":"_DEVIATION_PRECISION","type":"uint256"},{"internalType":"uint256","name":"_RATE_PRECISION","type":"uint256"},{"internalType":"uint256","name":"_MAX_PROTOCOL_FEE","type":"uint256"}],"stateMutability":"pure","type":"function"},{"inputs":[],"name":"getPairAccounting","outputs":[{"internalType":"uint128","name":"_totalAssetAmount","type":"uint128"},{"internalType":"uint128","name":"_totalAssetShares","type":"uint128"},{"internalType":"uint128","name":"_totalBorrowAmount","type":"uint128"},{"internalType":"uint128","name":"_totalBorrowShares","type":"uint128"},{"internalType":"uint256","name":"_totalCollateral","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_address","type":"address"}],"name":"getUserSnapshot","outputs":[{"internalType":"uint256","name":"_userAssetShares","type":"uint256"},{"internalType":"uint256","name":"_userBorrowShares","type":"uint256"},{"internalType":"uint256","name":"_userCollateralBalance","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"addedValue","type":"uint256"}],"name":"increaseAllowance","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"isBorrowAccessControlRevoked","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"isDepositAccessControlRevoked","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"isInterestAccessControlRevoked","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"isInterestPaused","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"isLiquidateAccessControlRevoked","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"isLiquidatePaused","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"isLiquidationFeeSetterRevoked","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"isMaxLTVSetterRevoked","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"isOracleSetterRevoked","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"isRateContractSetterRevoked","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"isRepayAccessControlRevoked","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"isRepayPaused","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"isWithdrawAccessControlRevoked","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"isWithdrawPaused","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_swapperAddress","type":"address"},{"internalType":"uint256","name":"_borrowAmount","type":"uint256"},{"internalType":"uint256","name":"_initialCollateralAmount","type":"uint256"},{"internalType":"uint256","name":"_amountCollateralOutMin","type":"uint256"},{"internalType":"address[]","name":"_path","type":"address[]"}],"name":"leveragedPosition","outputs":[{"internalType":"uint256","name":"_totalCollateralBalance","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint128","name":"_sharesToLiquidate","type":"uint128"},{"internalType":"uint256","name":"_deadline","type":"uint256"},{"internalType":"address","name":"_borrower","type":"address"}],"name":"liquidate","outputs":[{"internalType":"uint256","name":"_collateralForLiquidator","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_receiver","type":"address"}],"name":"maxDeposit","outputs":[{"internalType":"uint256","name":"_maxAssets","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxLTV","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_receiver","type":"address"}],"name":"maxMint","outputs":[{"internalType":"uint256","name":"_maxShares","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_owner","type":"address"}],"name":"maxRedeem","outputs":[{"internalType":"uint256","name":"_maxShares","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_owner","type":"address"}],"name":"maxWithdraw","outputs":[{"internalType":"uint256","name":"_maxAssets","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_shares","type":"uint256"},{"internalType":"address","name":"_receiver","type":"address"}],"name":"mint","outputs":[{"internalType":"uint256","name":"_amount","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"pause","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"pauseBorrow","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"pauseDeposit","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"_isPaused","type":"bool"}],"name":"pauseInterest","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"_isPaused","type":"bool"}],"name":"pauseLiquidate","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"_isPaused","type":"bool"}],"name":"pauseRepay","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"_isPaused","type":"bool"}],"name":"pauseWithdraw","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"pendingOwner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"pendingTimelockAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"previewAddInterest","outputs":[{"internalType":"uint256","name":"_interestEarned","type":"uint256"},{"internalType":"uint256","name":"_feesAmount","type":"uint256"},{"internalType":"uint256","name":"_feesShare","type":"uint256"},{"components":[{"internalType":"uint32","name":"lastBlock","type":"uint32"},{"internalType":"uint32","name":"feeToProtocolRate","type":"uint32"},{"internalType":"uint64","name":"lastTimestamp","type":"uint64"},{"internalType":"uint64","name":"ratePerSec","type":"uint64"},{"internalType":"uint64","name":"fullUtilizationRate","type":"uint64"}],"internalType":"struct FraxlendPairCore.CurrentRateInfo","name":"_newCurrentRateInfo","type":"tuple"},{"components":[{"internalType":"uint128","name":"amount","type":"uint128"},{"internalType":"uint128","name":"shares","type":"uint128"}],"internalType":"struct VaultAccount","name":"_totalAsset","type":"tuple"},{"components":[{"internalType":"uint128","name":"amount","type":"uint128"},{"internalType":"uint128","name":"shares","type":"uint128"}],"internalType":"struct VaultAccount","name":"_totalBorrow","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_assets","type":"uint256"}],"name":"previewDeposit","outputs":[{"internalType":"uint256","name":"_sharesReceived","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_shares","type":"uint256"}],"name":"previewMint","outputs":[{"internalType":"uint256","name":"_amount","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_shares","type":"uint256"}],"name":"previewRedeem","outputs":[{"internalType":"uint256","name":"_assets","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"previewWithdraw","outputs":[{"internalType":"uint256","name":"_sharesToBurn","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"pricePerShare","outputs":[{"internalType":"uint256","name":"_amount","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"protocolLiquidationFee","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"rateContract","outputs":[{"internalType":"contract IRateCalculatorV2","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_shares","type":"uint256"},{"internalType":"address","name":"_receiver","type":"address"},{"internalType":"address","name":"_owner","type":"address"}],"name":"redeem","outputs":[{"internalType":"uint256","name":"_amountToReturn","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_collateralAmount","type":"uint256"},{"internalType":"address","name":"_receiver","type":"address"}],"name":"removeCollateral","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"renounceTimelock","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_shares","type":"uint256"},{"internalType":"address","name":"_borrower","type":"address"}],"name":"repayAsset","outputs":[{"internalType":"uint256","name":"_amountToRepay","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_swapperAddress","type":"address"},{"internalType":"uint256","name":"_collateralToSwap","type":"uint256"},{"internalType":"uint256","name":"_amountAssetOutMin","type":"uint256"},{"internalType":"address[]","name":"_path","type":"address[]"}],"name":"repayAssetWithCollateral","outputs":[{"internalType":"uint256","name":"_amountAssetOut","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_borrowLimit","type":"uint256"}],"name":"revokeBorrowLimitAccessControl","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_depositLimit","type":"uint256"}],"name":"revokeDepositLimitAccessControl","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"revokeInterestAccessControl","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"revokeLiquidateAccessControl","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"revokeLiquidationFeeSetter","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"revokeMaxLTVSetter","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"revokeOracleInfoSetter","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"revokeRateContractSetter","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"revokeRepayAccessControl","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"revokeWithdrawAccessControl","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_limit","type":"uint256"}],"name":"setBorrowLimit","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_newCircuitBreaker","type":"address"}],"name":"setCircuitBreaker","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_limit","type":"uint256"}],"name":"setDepositLimit","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_newCleanLiquidationFee","type":"uint256"},{"internalType":"uint256","name":"_newDirtyLiquidationFee","type":"uint256"},{"internalType":"uint256","name":"_newProtocolLiquidationFee","type":"uint256"}],"name":"setLiquidationFees","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_newMaxLTV","type":"uint256"}],"name":"setMaxLTV","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_newOracle","type":"address"},{"internalType":"uint32","name":"_newMaxOracleDeviation","type":"uint32"}],"name":"setOracle","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_newRateContract","type":"address"}],"name":"setRateContract","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_swapper","type":"address"},{"internalType":"bool","name":"_approval","type":"bool"}],"name":"setSwapper","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"swappers","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"timelockAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_shares","type":"uint256"},{"internalType":"bool","name":"_roundUp","type":"bool"},{"internalType":"bool","name":"_previewInterest","type":"bool"}],"name":"toAssetAmount","outputs":[{"internalType":"uint256","name":"_amount","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_amount","type":"uint256"},{"internalType":"bool","name":"_roundUp","type":"bool"},{"internalType":"bool","name":"_previewInterest","type":"bool"}],"name":"toAssetShares","outputs":[{"internalType":"uint256","name":"_shares","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_shares","type":"uint256"},{"internalType":"bool","name":"_roundUp","type":"bool"},{"internalType":"bool","name":"_previewInterest","type":"bool"}],"name":"toBorrowAmount","outputs":[{"internalType":"uint256","name":"_amount","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_amount","type":"uint256"},{"internalType":"bool","name":"_roundUp","type":"bool"},{"internalType":"bool","name":"_previewInterest","type":"bool"}],"name":"toBorrowShares","outputs":[{"internalType":"uint256","name":"_shares","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalAsset","outputs":[{"internalType":"uint128","name":"amount","type":"uint128"},{"internalType":"uint128","name":"shares","type":"uint128"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalAssets","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalBorrow","outputs":[{"internalType":"uint128","name":"amount","type":"uint128"},{"internalType":"uint128","name":"shares","type":"uint128"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalCollateral","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"transfer","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"transferFrom","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_newTimelock","type":"address"}],"name":"transferTimelock","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"unpause","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"updateExchangeRate","outputs":[{"internalType":"bool","name":"_isBorrowAllowed","type":"bool"},{"internalType":"uint256","name":"_lowExchangeRate","type":"uint256"},{"internalType":"uint256","name":"_highExchangeRate","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"userBorrowShares","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"userCollateralBalance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"version","outputs":[{"internalType":"uint256","name":"_major","type":"uint256"},{"internalType":"uint256","name":"_minor","type":"uint256"},{"internalType":"uint256","name":"_patch","type":"uint256"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"uint256","name":"_amount","type":"uint256"},{"internalType":"address","name":"_receiver","type":"address"},{"internalType":"address","name":"_owner","type":"address"}],"name":"withdraw","outputs":[{"internalType":"uint256","name":"_sharesToBurn","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint128","name":"_shares","type":"uint128"},{"internalType":"address","name":"_recipient","type":"address"}],"name":"withdrawFees","outputs":[{"internalType":"uint256","name":"_amountToTransfer","type":"uint256"}],"stateMutability":"nonpayable","type":"function"}]Contract Creation Code
610120806040523462000958576200743a8038038091620000218285620013b4565b83398101606082820312620009585781516001600160401b0381116200095857816200004f918401620013d8565b60208301519092906001600160401b03811162000958578262000074918301620013d8565b60408201519092906001600160401b0381116200095857620000979201620013d8565b60405192620000a68462001398565b6000845260405192620000b98462001398565b60008452600180546001600160a01b03191633908117909155620000dd90620014ca565b60001960055560001960075560608180518101031262000958576200018362000109602083016200144f565b9162000118604082016200144f565b906001600160a01b039062000130906060016200144f565b600154911690816001600160a01b0382167f31b6c5a04b069b6ec1b3cef44c4e7c1eadd721349cda9823d0b1877b3551cdc6600080a36001600160a01b031916176001556001600160a01b0316620014ca565b33608052600480546001600160a01b0319166001600160a01b039290921691909117905583516001600160401b03811162000f3457600c54600181811c9116801562001355575b60208210146200103557601f8111620012eb575b506020601f82116001146200127b5781929394956000926200126f575b50508160011b916000199060031b1c191617600c555b82516001600160401b03811162000f3457600d54600181811c9116801562001264575b60208210146200103557601f8111620011fe575b506020601f82116001146200118f578192939460009262001183575b50508160011b916000199060031b1c191617600d555b6001600e556101408180518101031262000958576200029c602082016200144f565b90620002ab604082016200144f565b620002b9606083016200144f565b60808301519063ffffffff821682036200095857620002db60a085016200144f565b91620002ea60c0860162001464565b60e0860151966000602061010089015197620003106101406101208c01519b016200144f565b6001600160a01b0394851660a05290841660c081905260405163095ea7b360e01b8152948216600486015283196024860152909392839160449183915af18015620009665762001142575b506001600160a01b031660e05260175442600019810111620008a95743600019810111620008a9574260001990810160401b6fffffffffffffffff000000000000000016600160801b600160c01b0390921660c09390931b6001600160c01b031990811693909317919091174390910163ffffffff1617601755601880549091166001600160a01b039384161760a09290921b63ffffffff60a01b16919091179055601380546001600160a01b031916929091169190911790556010819055801562015f90808302839004141715620008a95762015f90620186a0910204601155601255600f55805181019060608183031262000958576020810151916001600160401b0383116200095857602092836200047d9201938491840101620013d8565b60408201519092906001600160401b03811162000958576060916020620004a792850101620013d8565b9101519060ff82168203620009585782516001600160401b03811162000f3457601554600181811c9116801562001137575b60208210146200103557601f8111620010d1575b506020601f821160011462001062578192939460009262001056575b50508160011b916000199060031b1c1916176015555b8051906001600160401b03821162000f345760165490600182811c921680156200104b575b6020831014620010355781601f84931162000fd4575b50602090601f831160011462000f565760009262000f4a575b50508160011b916000199060031b1c1916176016555b61010052600060806040516200059f8162001360565b8281528260208201528260408201528260608201520152604051620005c48162001360565b60175463ffffffff8116825263ffffffff8160201c16602083015260018060401b038160401c16604083015260018060401b038160801c16606083015260c01c60808201526040518061010081011060018060401b036101008301111762000f345761010081016040526000815260006020820152600060408201526000606082015260006080820152600060a082015260405162000663816200137c565b600081526000602082015260c082015260405162000681816200137c565b6000808252602082015260e082015260408201516001600160401b03164214158062000f23575b62000bbb575b80516200097a575b5050604051620006c68162001360565b6018546001600160a01b038116825260a081901c63ffffffff1660208301526019546001600160b81b03811660408401819052601a546060850152601b5460808501524214620009725760405163bd9a548b60e01b8152916060836004816001600160a01b0385165afa918215620009665760008094600094620008f8575b509160409391600080516020620073fa8339815191529593620008bf575b60018060b81b03421691828689015284606089015283608089015260018060a01b0388511663ffffffff60a01b60208a015160a01b169160018060c01b031916171760185560018060b81b0319161760195581601a5580601b5582519182526020820152a15b620007de608082015160608301519062001487565b80620186a00290620186a0820403620008a95760806200080192015190620014a9565b50604051615eb39081620015478239608051818181610da7015261445e015260a051818181610fe70152818161107801528181611145015281816113570152818161190a015281816135310152614fe2015260c051818181610f880152818161142b0152818161195f015281816119ff01528181611ac201528181611cc501526155d1015260e051818181613adf015281816154eb01526156c3015261010051816135950152f35b634e487b7160e01b600052601160045260246000fd5b7ffc131c36b7e444dacda44901fd43641dcdcfdc43fe9e2601b3c1dd87061db9e5602060018060a01b038951168751908152a162000763565b9450509091506060833d6060116200095d575b816200091a60609383620013b4565b8101031262000958578260409262000942600080516020620073fa8339815191529562001479565b6020830151949092015193949092919062000745565b600080fd5b3d91506200090b565b6040513d6000823e3d90fd5b5050620007c9565b6060810151917f2b5229f33f1d24d5baab718e1e25d0d86195a9b6d786c2c0868edfb21a460e256080808401519460a0850151957fc63977c8e2362a31182dc8e89a52252f9836922738e0abcfc0de6924972eafe58360018060401b0360608801511660018060401b03828901511660018060401b0360208b01511660018060401b0360408c01511691604051938452602084015260408301526060820152a160018060401b0360208701511660405192835260208301526040820152856060820152a16020828101516001600160401b03808216606085015260408086015180831660808088019190915242938416878401524363ffffffff168088529686015194811b600160801b600160c01b03169390921b6fffffffffffffffff00000000000000001693851b67ffffffff0000000016909517929092171760c093841b6001600160c01b0319161760175591830151805190820151831b6001600160801b03199081166001600160801b0392831617601c5560e090940151805192015190921b909216911617601d558062000b15575b80620006b6565b301562000b765762000b2a81600b546200151e565b600b55306000526009602052604060002081815401905560405190815260007fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60203093a33862000b0e565b60405162461bcd60e51b815260206004820152601f60248201527f45524332303a206d696e7420746f20746865207a65726f2061646472657373006044820152606490fd5b6001815260405162000bcd816200137c565b601c546001600160801b038116825260801c602082015260c082015260405162000bf7816200137c565b601d546001600160801b038116825260801c602082015260e0820152604082015162000c2d906001600160401b03164262001487565b60c0820151516001600160801b031662000edc576000905b60135460808501516040805163cd3181d560e01b81526004810185905260248101959095526001600160401b03909116604485015290839060649082906001600160a01b03165afa8015620009665760009260009162000e74575b506001600160401b039081166040850152909116602083015260e082015151670de0b6b3a76400009162000cfb9162000ce6916001600160801b03919091169062001495565b60208401516001600160401b03169062001495565b04806060830152801515908162000e4c575b508062000e21575b15620006ae57606081015160e0820151805190916001600160801b039162000d429183169083166200152c565b169052606081015160c0820151805190916001600160801b039162000d6c9183169083166200152c565b16905263ffffffff6020830151168062000d88575b50620006ae565b62000dc5620186a062000da462000deb93606086015162001495565b046080840181905260c0840151602001516001600160801b03169062001495565b60c083015151608084015162000de4916001600160801b031662001487565b90620014a9565b60a0820181905260c0820151602001805190916001600160801b039162000e179183169083166200152c565b1690523862000d81565b50606081015160c0820151516001600160801b039162000e44918316906200151e565b111562000d15565b60e0830151516001600160801b03925062000e6b91908316906200151e565b11153862000d0d565b9250506040823d60401162000ed3575b8162000e9360409383620013b4565b81010312620009585762000ce662000cfb918362000ec8602062000ec0670de0b6b3a76400009762001464565b920162001464565b925093509162000ca0565b3d915062000e84565b60e0820151516001600160801b0316620186a0808202048103620008a95760c08301515162000f1c916001600160801b0390911690620186a002620014a9565b9062000c45565b5060ff60085460381c1615620006a8565b634e487b7160e01b600052604160045260246000fd5b01519050388062000573565b6016600090815293506000805160206200741a83398151915291905b601f198416851062000fb8576001945083601f1981161062000f9e575b505050811b0160165562000589565b015160001960f88460031b161c1916905538808062000f8f565b8181015183556020948501946001909301929091019062000f72565b60166000529091506000805160206200741a833981519152601f840160051c8101602085106200102d575b90849392915b601f830160051c820181106200101d5750506200055a565b6000815585945060010162001005565b508062000fff565b634e487b7160e01b600052602260045260246000fd5b91607f169162000544565b01519050388062000509565b601560005260206000209060005b601f1984168110620010b85750600193949583601f198116106200109e575b505050811b016015556200051f565b015160001960f88460031b161c191690553880806200108f565b9091602060018192858a01518155019301910162001070565b60156000527f55f448fdea98c4d29eb340757ef0a66cd03dbb9538908a6a81d96026b71ec475601f830160051c8101602084106200112f575b601f830160051c8201811062001122575050620004ed565b600081556001016200110a565b50806200110a565b90607f1690620004d9565b6020813d6020116200117a575b816200115e60209383620013b4565b810103126200095857620011729062001479565b50386200035b565b3d91506200114f565b01519050388062000264565b600d60005260206000209060005b601f1984168110620011e55750600193949583601f19811610620011cb575b505050811b01600d556200027a565b015160001960f88460031b161c19169055388080620011bc565b9091602060018192858a0151815501930191016200119d565b600d6000527fd7b6990105719101dabeb77144f2a3385c8033acd3af97e9423a695e81ad1eb5601f830160051c8101602084106200125c575b601f830160051c820181106200124f57505062000248565b6000815560010162001237565b508062001237565b90607f169062000234565b015190503880620001fb565b600c60005260206000209060005b601f1984168110620012d2575060019394959683601f19811610620012b8575b505050811b01600c5562000211565b015160001960f88460031b161c19169055388080620012a9565b9091602060018192858b01518155019301910162001289565b600c6000527fdf6966c971051c3d54ec59162606531493a51404a002842f56009d7e5cf4a8c7601f830160051c810191602084106200134a575b601f0160051c01905b8181106200133d5750620001de565b600081556001016200132e565b909150819062001325565b90607f1690620001ca565b60a081019081106001600160401b0382111762000f3457604052565b604081019081106001600160401b0382111762000f3457604052565b602081019081106001600160401b0382111762000f3457604052565b601f909101601f19168101906001600160401b0382119082101762000f3457604052565b919080601f8401121562000958578251906001600160401b03821162000f34576040519160209162001414601f8301601f1916840185620013b4565b818452828287010111620009585760005b8181106200143b57508260009394955001015290565b858101830151848201840152820162001425565b51906001600160a01b03821682036200095857565b51906001600160401b03821682036200095857565b519081151582036200095857565b91908203918211620008a957565b81810292918115918404141715620008a957565b8115620014b4570490565b634e487b7160e01b600052601260045260246000fd5b600380546001600160a01b0319908116909155600280549182166001600160a01b0393841690811790915591167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0600080a3565b91908201809211620008a957565b6001600160801b039182169082160191908211620008a95756fe6080604052600436101561001257600080fd5b60003560e01c8062348d1814613bc757806301e1d11414613b9557806302ce728f14613b585780630475260e14613b03578063064f3fbd14613abf57806306fdde0314613a0157806307a2d13a146139d45780630880b2f01461398157806308a0c37514613921578063090f3f50146138fa578063095ea7b3146138d45780630a28a477146138a75780630a67918c146138845780630b4aecf0146138615780630c70661d1461383b578063115a334c1461381557806311a2e4bc146137f757806318160ddd146137d65780631bc23cf91461374c5780631c2591d3146137145780631c6c95971461367857806323b872dd146135b95780632b3ba681146114d5578063313ce5671461357b57806334680fe51461355557806338d52e0f1461351157806339030864146134c257806339509351146134705780633d417d2d146133eb5780633f2617cb146133665780633f4ba83a14613197578063402d267d1461313d57806345014095146130d357806345757e4a146130ad5780634732428c1461308f578063492924271461306857806349eb7af6146109005780634ac8eb5f1461304a5780634b4b418e14612fc45780634bc66f3214612f9d5780634c18a4fb14612f0f5780634c41799514612ebe5780634cdad506146117375780634f8b4ae714612e2d5780634fd422df14612df457806354fd4d5014612dca57806364e51d5d14612d6d57806367800b5f14612d4757806369026e8814612cf25780636b96668f14612c695780636cd3cc77146109005780636e553f6514612be157806370a0823114612ba7578063715018a614612b43578063721b0a471461282257806379ba50971461274d5780637d37bdd7146127155780637d63fbc2146126ef5780637ec4b571146126b25780637ecefa6e146126595780638142dd53146125995780638285ef401461256757806382beee89146124f05780638456cb5914612304578063858f1e6814612266578063886c033a1461220d5780638cad7fbe146121ce5780638da5cb5b146121a75780638f791f8b1461207b57806393f46f641461203e57806394bf804d14611fa357806395d14ca814611f4d57806395d89b4114611e4c57806399530b0614611dd25780639a295e7314611d7e5780639bc6ab8614611d585780639fe34bdc14610900578063a053db681461182d578063a457c2d71461176d578063a9059cbb1461173c578063b3d7f6b914611737578063b460af94146116cb578063b5af306214611691578063b68d0a091461162b578063b78294dd14610900578063b7db54f514611605578063b8ca3b83146115e8578063ba08765214611556578063bbb0962414611531578063bdc8144b146114da578063c0a7e892146114d5578063c58e4df6146114af578063c63d75b61461144f578063c6e1c7c91461140b578063c6e6f59214610720578063ca2298fe14610ecf578063cacf3b5814610ea0578063cadac47914610e4f578063cdd72d5214610dee578063ce96cb7714610dcb578063d2a156e014610d87578063d41ddc9614610cdc578063d905777e14610cb9578063daf33f2a14610bbb578063dd62ed3e14610b69578063e1e9277514610b1c578063e203641714610af9578063e30c397814610ad2578063e4b0007014610a81578063e551d11d14610a63578063e5f13b1614610905578063e63a391f14610900578063e7a33174146108a9578063e8596f7214610854578063eafecffa14610836578063ebd462cb1461079a578063ecf708581461077c578063eee2421914610755578063ef8b30f714610720578063f211c390146106fa578063f2fde38b14610690578063f384bd0514610672578063f6ccaad41461060c578063f9557ccb146105d65763fbbbf94c1461057a57600080fd5b346105d15760003660031901126105d15760a06018546001600160b81b0360195416601a54601b549163ffffffff604051946001600160a01b0381168652861c166020850152604084015260608301526080820152f35b600080fd5b346105d15760003660031901126105d157601c54604080516001600160801b038316815260809290921c602083015290f35b0390f35b346105d15760003660031901126105d157610625615afc565b6001600160a01b0319806000541660005560015490336001600160a01b0383167f31b6c5a04b069b6ec1b3cef44c4e7c1eadd721349cda9823d0b1877b3551cdc6600080a3163317600155005b346105d15760003660031901126105d1576020600f54604051908152f35b346105d15760203660031901126105d1576106a9613c69565b6106b1613eca565b6001600160a01b0380911690816001600160a01b03196003541617600355600254167f38d16b8cac22d99fc7c124b9cd0de2d3fa1faef420bfe791d8c362d765e22700600080a3005b346105d15760003660031901126105d157602060ff60085460381c166040519015158152f35b346105d15760203660031901126105d157602061074d61073e6145bd565b50935050505060043590615d69565b604051908152f35b346105d15760003660031901126105d15760206001600160a01b0360135416604051908152f35b346105d15760003660031901126105d1576020600754604051908152f35b346105d15760203660031901126105d1576107b3613cbf565b8015610829576107c1614410565b6008549060ff8260201c16610817577fc56dd3e14f5af3a74c61b7cdf855a3d8ab4401c78c0622a4d312de8a8f8736a29160209115159063ff0000008260181b169063ff000000191617600855604051908152a1005b60405163035b48f760e31b8152600490fd5b610831614498565b6107c1565b346105d15760003660031901126105d1576020601254604051908152f35b346105d15760003660031901126105d15761086d614410565b60ff600654166108175760006005557fbf1ce7fb3a8e648b70ea830f99b52f7ea31554186d29763280751f42e77f6386602060405160008152a1005b346105d15760203660031901126105d1576004356108c5614498565b60ff60065416610817576020817fbf1ce7fb3a8e648b70ea830f99b52f7ea31554186d29763280751f42e77f638692600555604051908152a1005b613dc5565b346105d15760603660031901126105d157600435602435610924613c95565b9061092d61451b565b6001600160a01b03821615610a51576109446149f7565b5050505050600554610961846001600160801b03601d5416613f22565b11610a3f5761096e614cbf565b505015610a2d5761098e928161098992610a1b575b50614391565b615391565b610996614344565b50601b54906109a582336152ed565b156109ba576020906001600e55604051908152f35b506109da6109c66141f6565b336000526020805260406000205490615e34565b33600052601f602052610a1760406000205492604051938493633b49de0f60e21b8552600485016040919493926060820195825260208201520152565b0390fd5b610a27903390336154db565b84610983565b60405163345513d960e01b8152600490fd5b6040516397ba4de360e01b8152600490fd5b604051631e4ec46b60e01b8152600490fd5b346105d15760003660031901126105d1576020600554604051908152f35b346105d15760003660031901126105d157610a9a615ad6565b6201000062ff00001960215416176021557f70328969870b42d0fc62ce5946530c50a466c5ad21af004a9f2f3a8eda9a4a0c600080a1005b346105d15760003660031901126105d15760206001600160a01b0360035416604051908152f35b346105d15760003660031901126105d157602060ff600854166040519015158152f35b346105d15760003660031901126105d157610b35615ad6565b600160ff1960215416176021557faa96740f913149dce2173396218295b4c082e86fabacc37ac8d45305239d26f3600080a1005b346105d15760403660031901126105d157610b82613c69565b610b8a613c7f565b906001600160a01b03809116600052600a602052604060002091166000526020526020604060002054604051908152f35b346105d15760403660031901126105d157610bd4613de3565b610bdc613c7f565b90610be5613eca565b806001600160a01b038316928315610a5157602093610c026141d0565b926001600160801b0380911615610ca1575b917faf48306b6b4f0ba30d00f05b21559d8d29934142980a553d8a014780c6c7e4529391610c6960809487169383610c4c8683615e34565b98610c588733306140ce565b3092610c638b614391565b9061502e565b30600052601f8752610c83604060002054923090846156b8565b604051928352868301528460408301526060820152a1604051908152f35b30600090815260098752604090205481169450610c14565b346105d15760203660031901126105d157602061074d610cd7613c69565b6142c5565b346105d15760403660031901126105d157610cf5613c7f565b610cfd61451b565b6001600160a01b03811615610a5157610d146149f7565b50505050503360005260208052604060002054610d67575b610d3a9033906004356156b8565b610d42614344565b50601b54610d5081336152ed565b15610d5c576001600e55005b6109da6109c66141f6565b610d6f614cbf565b5050610d2c5760405163345513d960e01b8152600490fd5b346105d15760003660031901126105d15760206040516001600160a01b037f0000000000000000000000000000000000000000000000000000000000000000168152f35b346105d15760203660031901126105d157602061074d610de9613c69565b614229565b346105d15760003660031901126105d15760a0610e096145bd565b9350935050506001600160801b03908160208185511694015116916020818351169201511690601e54926040519485526020850152604084015260608301526080820152f35b346105d15760403660031901126105d157610e68613c7f565b610e7061451b565b6001600160a01b03811615610a5157610e9990610e8b6149f7565b5050505050600435336154db565b6001600e55005b346105d15760003660031901126105d157610608610ebc6145bd565b9260409694969291925196879687613cfa565b346105d15760803660031901126105d157610ee8613c69565b60643567ffffffffffffffff8082116105d157366023830112156105d1578160040135116105d157366024826004013560051b830101116105d157610f2b61451b565b610f336149f7565b5050505050610f40614cbf565b505015610a2d576001600160a01b038216600052601460205260ff60406000205416156113f9578060040135156113e3576001600160a01b03610f8560248301615ac2565b817f00000000000000000000000000000000000000000000000000000000000000001691829116036113ae576004820135600019810190811161139857610fdc610fd782856004013560248701615ab2565b615ac2565b6001600160a01b03807f00000000000000000000000000000000000000000000000000000000000000001691160361132f575061101c33306024356156b8565b60405163095ea7b360e01b81526001600160a01b0384166004820152602480359082015290602090829060449082906000905af18015611267576112f6575b506040516370a0823160e01b8152306004820152906020826024817f00000000000000000000000000000000000000000000000000000000000000006001600160a01b03165afa918215611267576000926112c2575b5060405180916338ed173960e01b82526024356004830152604435602483015260a06044830152806004013560a483015260c4820190602481019060005b816004013581106112965750505090806000923060648301524260848301520381836001600160a01b0388165af1801561126757611273575b506040516370a0823160e01b8152306004820152906020826024817f00000000000000000000000000000000000000000000000000000000000000006001600160a01b03165afa801561126757600090611233575b611187925061421c565b906044358210611213576111996141f6565b6111c36111a68483615d69565b916111b085614391565b339130916111bd86614391565b91615861565b6001600160a01b0360405192168252602435602083015282604083015260608201527fe947f0f9b6255bdcf76d13d1257d34fbe380e0d5d4daa75e61c783a41e1607ba60803392a2610996614344565b604051633b5d56ed60e11b81526044803560048301526024820184905290fd5b506020823d60201161125f575b8161124d60209383613e5b565b810103126105d157611187915161117d565b3d9150611240565b6040513d6000823e3d90fd5b61128f903d806000833e6112878183613e5b565b810190615a38565b5082611128565b919350916020806001926001600160a01b036112b188613cab565b1681520194019101918493926110ef565b9091506020813d6020116112ee575b816112de60209383613e5b565b810103126105d1575190836110b1565b3d91506112d1565b6020813d602011611327575b8161130f60209383613e5b565b810103126105d15761132090614cb2565b508261105b565b3d9150611302565b611348610fd7610a179285602481600401359101615ab2565b60405163b0b3262d60e01b81527f00000000000000000000000000000000000000000000000000000000000000006001600160a01b03908116600483015290911660248201529081906044820190565b634e487b7160e01b600052601160045260246000fd5b6113ba60248301615ac2565b60405163b0b3262d60e01b81526001600160a01b03928316600482015291166024820152604490fd5b634e487b7160e01b600052603260045260246000fd5b604051631311dc6d60e01b8152600490fd5b346105d15760003660031901126105d15760206040516001600160a01b037f0000000000000000000000000000000000000000000000000000000000000000168152f35b346105d15760203660031901126105d157611468613c69565b50602061074d6114766145bd565b5093505050506001600160801b0381511660075490818110156000146114a0575050600090615d69565b6114a99161421c565b90615d69565b346105d15760003660031901126105d157602060ff60085460281c166040519015158152f35b613da2565b346105d15760203660031901126105d1576004356114f6614498565b60ff60085416610817576020817f854df3eb95564502c8bc871ebdd15310ee26270f955f6c6bd8cea68e75045bc092600755604051908152a1005b346105d15760003660031901126105d157602060085460ff60405191831c1615158152f35b346105d15761156436613e95565b919061156e61451b565b6001600160a01b03811615610a515760ff60085460181c166115d6576020926115c9916115996149f7565b50505050506115a66141d0565b6115b08582615e03565b946115c36115bd87614391565b91614391565b9161502e565b6001600e55604051908152f35b60405163e0a3980360e01b8152600490fd5b346105d15760003660031901126105d157602060405161c3508152f35b346105d15760003660031901126105d157602060ff60215460081c166040519015158152f35b346105d15760203660031901126105d1576001600160a01b0361164c613c69565b1660005260096020526040600020546020805260406000205490601f602052610608604060002054604051938493846040919493926060820195825260208201520152565b346105d15760203660031901126105d1576001600160a01b036116b2613c69565b16600052601f6020526020604060002054604051908152f35b346105d1576116d936613e95565b91906116e361451b565b6001600160a01b03811615610a515760ff60085460181c166115d6576020926115c99161170e6149f7565b505050505061171b6141d0565b61172e6117288683615cff565b95614391565b6115c386614391565b6139d4565b346105d15760403660031901126105d157611762611758613c69565b6024359033613f2f565b602060405160018152f35b346105d15760403660031901126105d157611786613c69565b6024359033600052600a60205260406000206001600160a01b038216600052602052604060002054918083106117c257611762920390336140ce565b60405162461bcd60e51b815260206004820152602560248201527f45524332303a2064656372656173656420616c6c6f77616e63652062656c6f7760448201527f207a65726f0000000000000000000000000000000000000000000000000000006064820152608490fd5b346105d15760a03660031901126105d157611846613c69565b6084359067ffffffffffffffff82116105d157366023830112156105d157816004013561187281613e7d565b926118806040519485613e5b565b818452602084016024819360051b830101913683116105d157602401905b828210611d40575050506118b061451b565b6118b86149f7565b50505050506118c5614cbf565b505015610a2d576001600160a01b038216600052601460205260ff60406000205416156113f9576001600160a01b036118fd84615a17565b5116926001600160a01b037f000000000000000000000000000000000000000000000000000000000000000016809403611d0157805160001990818101908111611398576119536001600160a01b039184615a24565b51166001600160a01b037f00000000000000000000000000000000000000000000000000000000000000001603611c8c5750604435611c7a575b6119a261199b602435614391565b3090615391565b60405163095ea7b360e01b81526001600160a01b038516600482015260248035908201529094602090829060449082906000905af1801561126757611c41575b506040516370a0823160e01b8152306004820152916020836024817f00000000000000000000000000000000000000000000000000000000000000006001600160a01b03165afa92831561126757600093611c0d575b506040516338ed173960e01b81526024803560048301526064359082015260a06044820152915160a48301819052829160c483019160005b818110611beb5750505090806000923060648301524260848301520381836001600160a01b0388165af1801561126757611bd0575b506040516370a0823160e01b8152306004820152906020826024817f00000000000000000000000000000000000000000000000000000000000000006001600160a01b03165afa801561126757600090611b9c575b611b04925061421c565b6064358110611b7c57611b183382306154db565b611b2481604435613f22565b926001600160a01b036040519316835260243560208401526040830152604435606083015260808201527fb19ca0df3f3a01af950d8e6ad62aeff167cf14c73e98af6c52afef1add5c97ed60a03392a2610996614344565b60449060405190633b5d56ed60e11b825260643560048301526024820152fd5b506020823d602011611bc8575b81611bb660209383613e5b565b810103126105d157611b049151611afa565b3d9150611ba9565b611be4903d806000833e6112878183613e5b565b5083611aa5565b82516001600160a01b0316845285945060209384019390920191600101611a70565b9092506020813d602011611c39575b81611c2960209383613e5b565b810103126105d157519185611a38565b3d9150611c1c565b6020813d602011611c72575b81611c5a60209383613e5b565b810103126105d157611c6b90614cb2565b50846119e2565b3d9150611c4d565b611c8733604435336154db565b61198d565b815190810190811161139857611cad6001600160a01b0391610a1793615a24565b5160405163b0b3262d60e01b81526001600160a01b037f0000000000000000000000000000000000000000000000000000000000000000811660048301529290911690911660248201529081906044820190565b6001600160a01b03611d138592615a17565b5160405163b0b3262d60e01b81526001600160a01b03938416600482015291169091166024820152604490fd5b60208091611d4d84613cab565b81520191019061189e565b346105d15760003660031901126105d157602060ff60215460181c166040519015158152f35b346105d15760003660031901126105d157610100604051620186a0808252806020830152806040830152806060830152670de0b6b3a76400009081608084015260a083015260c082015261c35060e0820152f35b346105d15760003660031901126105d157611deb6145bd565b509350505050602081016001600160801b03918282511615600014611e20575050506020670de0b6b3a7640000604051908152f35b51670de0b6b3a76400009083168181029182040361139857602092611e4792511690614680565b61074d565b346105d15760003660031901126105d157604051600090601654600181811c90808316928315611f43575b6020938484108114611f2d57838652908115611f0d5750600114611eb2575b61060884611ea681880382613e5b565b60405191829182613c20565b601660009081529294507fd833147d7dc355ba459fc788f669e58cfaf9dc25ddcd0702e87d69c7b51242895b828410611efa575050508161060893611ea69282010193611e96565b8054858501870152928501928101611ede565b60ff1916858501525050151560051b8201019150611ea681610608611e96565b634e487b7160e01b600052602260045260246000fd5b91607f1691611e77565b346105d15760003660031901126105d15760a06017546040519063ffffffff80821683528160201c16602083015267ffffffffffffffff808260401c1660408401528160801c16606083015260c01c6080820152f35b346105d15760403660031901126105d157600435611fbf613c7f565b611fc761451b565b6001600160a01b03811615610a5157611fde6149f7565b5050505050611feb6141d0565b91611ff68184615e03565b9160075461200e846001600160801b03875116613f22565b1161202c576020936115c9926120266115bd86614391565b91614f07565b604051630aad288560e21b8152600490fd5b346105d157602061204e36613cce565b1561206a5761074d9161205f6145bd565b945050505050615d96565b611e47916120766141f6565b615d96565b346105d15760403660031901126105d157612094613c69565b63ffffffff60243581811692918382036105d1576120b0615ad6565b60ff60215416612195576080937f78ba1c32ac8ea4b3d51133dd0b6f5d8f98e23797aade6afc381ea317d5d4f28b85612135936120eb614344565b966001600160a01b0390818951169260208a015116604051938452602084015216938460408301526060820152a16001600160a01b03166001600160a01b03196018541617601855565b63ffffffff60a01b1963ffffffff60a01b6018549260a01b169116176018556001600160b81b0360408201511676ffffffffffffffffffffffffffffffffffffffffffffff1960195416176019556060810151601a550151601b55600080f35b604051631186953760e31b8152600490fd5b346105d15760003660031901126105d15760206001600160a01b0360025416604051908152f35b346105d15760203660031901126105d1576001600160a01b036121ef613c69565b166000526014602052602060ff604060002054166040519015158152f35b346105d15760203660031901126105d1577f3ff713beec3d10b4dfe28953471682eab1f857ba2fdb6367366252381888a750602060043561224c615ad6565b600160ff19600854161760085580600755604051908152a1005b346105d15760203660031901126105d15761227f613cbf565b80156122f75761228d614410565b60ff60085460401c166108175760207fdea8bb46eee4300a7d2de86939c245f568dc5994576194cbfb69969e010dcb67916122c66149f7565b5050505050151560085467ff000000000000008260381b169067ff00000000000000191617600855604051908152a1005b6122ff614498565b61228d565b346105d15760003660031901126105d15761231d614410565b60ff8060065416156124bb575b600890815481811615612486575b818160101c1615612449575b508154818160201c1615612408575b508154818160301c16156123c3575b50815460401c161561237057005b6123786149f7565b505050505067010000000000000067ff00000000000000198254161790557fdea8bb46eee4300a7d2de86939c245f568dc5994576194cbfb69969e010dcb67602060405160018152a1005b650100000000009065ff000000000019161782557f28bc4f9e24da61e7ba3aa697dfaefd0167093d2425c00b6190a7d3152ee6dfaa602060405160018152a182612362565b63010000009063ff00000019161782557fc56dd3e14f5af3a74c61b7cdf855a3d8ab4401c78c0622a4d312de8a8f8736a2602060405160018152a182612353565b6101009061ff0019161782557f34a71a12fa81891b738d910d4d44ffabeeb12f8bc026844db237ea8bf8ebe8be602060405160018152a182612344565b60006007557f854df3eb95564502c8bc871ebdd15310ee26270f955f6c6bd8cea68e75045bc0602060405160008152a1612338565b60006005557fbf1ce7fb3a8e648b70ea830f99b52f7ea31554186d29763280751f42e77f6386602060405160008152a161232a565b346105d15760203660031901126105d1577f4cb8c9e37efb94c6cdbd2a80fe36cee1957b5584d1a1986fa2bae115180af59a61252a613c69565b612532615ad6565b600480546001600160a01b039283166001600160a01b03198216811790925560408051939091168352602083019190915290a1005b346105d15760003660031901126105d157601d54604080516001600160801b038316815260809290921c602083015290f35b346105d15760203660031901126105d15760043563ffffffff8116908181036105d1576125c4615ad6565b60ff60085460381c166126475761c3508211612635577f58a58c712558f3d6e20bed57421eb8f73048d881dea9e5bb80efb37c49680d1c916020916126076149f7565b505050505067ffffffff0000000060175491841b169067ffffffff00000000191617601755604051908152a1005b60405163da0afa5760e01b8152600490fd5b60405163a02a2bcd60e01b8152600490fd5b346105d15760003660031901126105d157612672615ad6565b660100000000000066ff0000000000001960085416176008557f60c2acdf5b421891c8cc7302420292f2680f0e835fc76dd15f35a7bb0dd5cbc8600080a1005b346105d15760206126c236613cce565b156126de5761074d916126d36145bd565b945050505050615e6f565b611e47916126ea6141f6565b615e6f565b346105d15760003660031901126105d157602060ff60215460101c166040519015158152f35b346105d157602061272536613cce565b156127415761074d916127366145bd565b509350505050615e6f565b611e47916126ea6141d0565b346105d15760003660031901126105d1576003546001600160a01b0333818316036127b7576001600160a01b03198092166003556002549133908316176002553391167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0600080a3005b60405162461bcd60e51b815260206004820152602960248201527f4f776e61626c6532537465703a2063616c6c6572206973206e6f74207468652060448201527f6e6577206f776e657200000000000000000000000000000000000000000000006064820152608490fd5b346105d15760603660031901126105d15761283b613de3565b60243590612847613c95565b61284f61451b565b6001600160a01b038116928315610a515760ff60085460281c16612b3157804211612b13575061287d6149f7565b505050505061288a614cbf565b50905061289781836152ed565b612b01576128a36141f6565b9184600052601f60205260406000205492602080526128c6604060002054614391565b926000936001600160801b0396670de0b6b3a76400006128f2898316946128ed8688615e03565b61466d565b049860105497620186a0988901808a11611398576129118a918d61466d565b0461292461291e83615982565b91615982565b90600082820392128183128116918313901516176113985760001280159190612ae15760209b505b809960125480612a9f575b50505086959293612a00979486938b6115c99c9d6129e6958d6129856129806129ee9d8c615e34565b614391565b98600097600093612a08575b509160c093917f821de4e13fff1938b3806eb2859b6a5d55111f00dcf286f8a793584228ff36f895936040519485526020850152828b166040850152606084015281881660808401521660a0820152a26146b5565b903392615861565b6129f98133876156b8565b30836156b8565b3090306154db565b7f821de4e13fff1938b3806eb2859b6a5d55111f00dcf286f8a793584228ff36f8959391985091612a3c8860c096946144d7565b98838d818c1680612a56575b505050919395509193612991565b82955090612a6a612980612a749383615e03565b95869151166144d7565b168d52601c5484612a87858284166144d7565b16906001600160801b03191617601c55838d38612a48565b6115c99b50829a50936129e693612ace8b9a9793612ac6612a009d9a966129ee9c9961466d565b04809d61421c565b9c50935093968296508195989950612957565b506011548901808a1161139857612afb8a9160209d61466d565b0461294c565b604051633af2cafd60e11b8152600490fd5b60449060405190635ba2a8d560e01b82524260048301526024820152fd5b604051631b4b0d7760e21b8152600490fd5b346105d15760003660031901126105d157612b5c613eca565b60006001600160a01b036001600160a01b03198060035416600355600254908116600255167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e08280a3005b346105d15760203660031901126105d1576001600160a01b03612bc8613c69565b1660005260096020526020604060002054604051908152f35b346105d15760403660031901126105d157600435612bfd613c7f565b90612c0661451b565b6001600160a01b03821615610a5157612c1d6149f7565b5050505050612c2a6141d0565b600754612c41836001600160801b03845116613f22565b1161202c5760209281612c60612c5a856115c995615d69565b94614391565b61202685614391565b346105d15760203660031901126105d157612c82613c69565b612c8a615ad6565b60ff60215460101c1661219557601354604080516001600160a01b038084168252848116602083015292936001600160a01b0319939290917faeae842c8b3cd009fbb602e1ed072dc1aec69750e431ceae97f7543b466cd04c9190a116911617601355600080f35b346105d15760003660031901126105d157612d0b614410565b60ff600854166108175760006007557f854df3eb95564502c8bc871ebdd15310ee26270f955f6c6bd8cea68e75045bc0602060405160008152a1005b346105d15760003660031901126105d157602060ff60085460181c166040519015158152f35b346105d15760003660031901126105d157612d86615ad6565b6801000000000000000068ff00000000000000001960085416176008557f16c0a933c76f28f1abdcef88bcea1650397c5f4bb4bf491a0d451a65cae016b6600080a1005b346105d15760003660031901126105d1576060604051600381526000602082015260006040820152f35b346105d15760203660031901126105d1576001600160a01b03612e15613c69565b16600052602080526020604060002054604051908152f35b346105d15760003660031901126105d157612e46615ad6565b612e4e615afc565b6001600160a01b0319806000541660005560015460006001600160a01b03821681817f162998b90abc2507f3953aa797827b03a14c42dbd9a35f09feaf02e0d592773a8280a37f31b6c5a04b069b6ec1b3cef44c4e7c1eadd721349cda9823d0b1877b3551cdc68280a316600155005b346105d15760003660031901126105d157612ed7615ad6565b6201000062ff00001960085416176008557f269ac55859865c2ff127a862e95c81ce7e3b9b13582036d3df419df5c07ec8b4600080a1005b346105d15760203660031901126105d157612f28613cbf565b8015612f9057612f36614410565b6008549060ff8260301c16610817577f28bc4f9e24da61e7ba3aa697dfaefd0167093d2425c00b6190a7d3152ee6dfaa9160209115159065ff00000000008260281b169065ff0000000000191617600855604051908152a1005b612f98614498565b612f36565b346105d15760003660031901126105d15760206001600160a01b0360015416604051908152f35b346105d15760203660031901126105d157612fdd613cbf565b801561303d57612feb614410565b6008549060ff8260101c16610817577f34a71a12fa81891b738d910d4d44ffabeeb12f8bc026844db237ea8bf8ebe8be9160209115159061ff008260081b169061ff00191617600855604051908152a1005b613045614498565b612feb565b346105d15760003660031901126105d1576020601e54604051908152f35b346105d15760003660031901126105d15760206001600160a01b0360045416604051908152f35b346105d15760003660031901126105d1576020601154604051908152f35b346105d15760003660031901126105d157602060ff60085460401c166040519015158152f35b346105d15760203660031901126105d1576130ec613c69565b6130f4615ad6565b6001600160a01b0380911690816001600160a01b03196000541617600055600154167f162998b90abc2507f3953aa797827b03a14c42dbd9a35f09feaf02e0d592773a600080a3005b346105d15760203660031901126105d157613156613c69565b506001600160801b036131676145bd565b50935050505051166007548082101560001461318b57505060206000604051908152f35b602091611e479161421c565b346105d15760003660031901126105d1576131b0614498565b60ff806006541615613330575b6008908154818116156132fa575b818160101c16156132c2575b508154818160201c1615613288575b508154818160301c161561324c575b50815460401c161561320357005b61320b6149f7565b505050505067ff000000000000001981541690557fdea8bb46eee4300a7d2de86939c245f568dc5994576194cbfb69969e010dcb67602060405160008152a1005b65ff0000000000191682557f28bc4f9e24da61e7ba3aa697dfaefd0167093d2425c00b6190a7d3152ee6dfaa602060405160008152a1826131f5565b63ff000000191682557fc56dd3e14f5af3a74c61b7cdf855a3d8ab4401c78c0622a4d312de8a8f8736a2602060405160008152a1826131e6565b61ff00191682557f34a71a12fa81891b738d910d4d44ffabeeb12f8bc026844db237ea8bf8ebe8be602060405160008152a1826131d7565b7f854df3eb95564502c8bc871ebdd15310ee26270f955f6c6bd8cea68e75045bc0602060001980600755604051908152a16131cb565b7fbf1ce7fb3a8e648b70ea830f99b52f7ea31554186d29763280751f42e77f6386602060001980600555604051908152a16131bd565b346105d15760403660031901126105d15761337f613c69565b602435908115158092036105d1577fea1eefb4fd58778d7b274fe54045a9feeec8f2847899c2e71126d3a74d486da5916001600160a01b036040926133c2613eca565b16908160005260146020528260002060ff1981541660ff831617905582519182526020820152a1005b346105d15760403660031901126105d157600435613407613c7f565b9061341061451b565b6001600160a01b03821615610a515760ff60085460081c1661345e576115c960209261343a6149f7565b50505050506134476141f6565b6134518482615e34565b936129e66115bd86614391565b604051631e61c1e960e11b8152600490fd5b346105d15760403660031901126105d15761176261348c613c69565b33600052600a60205260406000206001600160a01b0382166000526020526134bb602435604060002054613f22565b90336140ce565b346105d15760003660031901126105d1576134db615ad6565b61010061ff001960215416176021557f0af6d9d6ea0e3f0cdb71562ce1fce30aa597445ea04f5b25a939cfe0a252171c600080a1005b346105d15760003660031901126105d15760206040516001600160a01b037f0000000000000000000000000000000000000000000000000000000000000000168152f35b346105d15760003660031901126105d157602060ff60085460301c166040519015158152f35b346105d15760003660031901126105d157602060405160ff7f0000000000000000000000000000000000000000000000000000000000000000168152f35b346105d15760603660031901126105d1576135d2613c69565b6135da613c7f565b604435906001600160a01b038316600052600a602052604060002033600052602052604060002054926000198403613617575b6117629350613f2f565b8284106136335761362e83611762950333836140ce565b61360d565b60405162461bcd60e51b815260206004820152601d60248201527f45524332303a20696e73756666696369656e7420616c6c6f77616e63650000006044820152606490fd5b346105d15760203660031901126105d157613691613cbf565b6136996144f0565b506040516136a681613df9565b6000815260006020820152610608604051916136c183613df9565b60008352600060208401526136d461451b565b916136dd6149f7565b9296935090966136f9575b6001600e5560405196879687613cfa565b925090506137056141d0565b9061370e6141f6565b926136e8565b346105d157602061372436613cce565b156137405761074d916137356145bd565b509350505050615d96565b611e47916120766141d0565b346105d15760603660031901126105d15760443560243560043561376e615ad6565b60ff60215460181c16612195577fc9aa62b60be8f25ac9f285edbb80bde64199b3c53e1da1027058551d32695fca60c060105460115460125490604051928352602083015260408201528360608201528460808201528560a0820152a1601055601155601255005b346105d15760003660031901126105d1576020601c5460801c604051908152f35b346105d15760003660031901126105d1576020601054604051908152f35b346105d15760003660031901126105d157602060ff60085460081c166040519015158152f35b346105d15760003660031901126105d157602060ff60085460101c166040519015158152f35b346105d15760003660031901126105d157602060ff602154166040519015158152f35b346105d15760003660031901126105d157602060ff600654166040519015158152f35b346105d15760203660031901126105d157602061074d6138c56145bd565b50935050505060043590615cff565b346105d15760403660031901126105d1576117626138f0613c69565b60243590336140ce565b346105d15760003660031901126105d15760206001600160a01b0360005416604051908152f35b346105d15760203660031901126105d15760043561393d615ad6565b60ff60215460081c16612195577fe796e9ae748449310fcd1cc6718aab236c9b8d2e0e04dacb232ba564d5b338cc6040600f548151908152836020820152a1600f55005b346105d15760003660031901126105d15761399a615ad6565b630100000063ff0000001960215416176021557f1cd8398e5a04411acbddcb6451a57b51c242322c538947cea5e4a1a506700b87600080a1005b346105d15760203660031901126105d157602061074d6139f26145bd565b50935050505060043590615e03565b346105d15760003660031901126105d157604051600090601554600181811c90808316928315613ab5575b6020938484108114611f2d57838652908115611f0d5750600114613a5a5761060884611ea681880382613e5b565b601560009081529294507f55f448fdea98c4d29eb340757ef0a66cd03dbb9538908a6a81d96026b71ec4755b828410613aa2575050508161060893611ea69282010193611e96565b8054858501870152928501928101613a86565b91607f1691613a2c565b346105d15760003660031901126105d15760206040516001600160a01b037f0000000000000000000000000000000000000000000000000000000000000000168152f35b346105d15760003660031901126105d157613b1c615ad6565b64010000000064ff000000001960085416176008557fb949af551d0c88280e648f9205b986bb5f1d899c425498238655ee37617c0c39600080a1005b346105d15760003660031901126105d157613b7161451b565b6060613b7b614cbf565b906001600e55604051921515835260208301526040820152f35b346105d15760003660031901126105d15760206001600160801b03613bb86145bd565b50516040519516855250505050f35b346105d15760203660031901126105d1577fee4b3f9e70b2c6499288c7b5fbef140756009cf8839be64c473b1c7cb6d616c46020600435613c06615ad6565b600160ff19600654161760065580600555604051908152a1005b6020808252825181830181905290939260005b828110613c5557505060409293506000838284010152601f8019910116010190565b818101860151848201604001528501613c33565b600435906001600160a01b03821682036105d157565b602435906001600160a01b03821682036105d157565b604435906001600160a01b03821682036105d157565b35906001600160a01b03821682036105d157565b6004359081151582036105d157565b60609060031901126105d1576004359060243580151581036105d1579060443580151581036105d15790565b9194613d839197969461014094613da09761018086019a86526020860152604085015263ffffffff8082511660608601526020820151166080850152608067ffffffffffffffff918260408201511660a08701528260608201511660c087015201511660e0840152610100830190602090816001600160801b0391828151168552015116910152565b0190602090816001600160801b0391828151168552015116910152565b565b346105d15760003660031901126105d1576020604051670de0b6b3a76400008152f35b346105d15760003660031901126105d1576020604051620186a08152f35b600435906001600160801b03821682036105d157565b6040810190811067ffffffffffffffff821117613e1557604052565b634e487b7160e01b600052604160045260246000fd5b60a0810190811067ffffffffffffffff821117613e1557604052565b67ffffffffffffffff8111613e1557604052565b90601f8019910116810190811067ffffffffffffffff821117613e1557604052565b67ffffffffffffffff8111613e155760051b60200190565b60609060031901126105d157600435906001600160a01b039060243582811681036105d1579160443590811681036105d15790565b6001600160a01b03600254163303613ede57565b606460405162461bcd60e51b815260206004820152602060248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e65726044820152fd5b9190820180921161139857565b6001600160a01b0380911691821561406357169182156140125760008281526009602052604081205491808310613fa757604082827fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef95876020965260098652038282205586815220818154019055604051908152a3565b60405162461bcd60e51b815260206004820152602660248201527f45524332303a207472616e7366657220616d6f756e742065786365656473206260448201527f616c616e636500000000000000000000000000000000000000000000000000006064820152608490fd5b60405162461bcd60e51b815260206004820152602360248201527f45524332303a207472616e7366657220746f20746865207a65726f206164647260448201526265737360e81b6064820152608490fd5b60405162461bcd60e51b815260206004820152602560248201527f45524332303a207472616e736665722066726f6d20746865207a65726f20616460448201527f64726573730000000000000000000000000000000000000000000000000000006064820152608490fd5b6001600160a01b0380911691821561417f571691821561412f5760207f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b9259183600052600a8252604060002085600052825280604060002055604051908152a3565b60405162461bcd60e51b815260206004820152602260248201527f45524332303a20617070726f766520746f20746865207a65726f206164647265604482015261737360f01b6064820152608490fd5b60405162461bcd60e51b8152602060048201526024808201527f45524332303a20617070726f76652066726f6d20746865207a65726f206164646044820152637265737360e01b6064820152608490fd5b604051906141dd82613df9565b601c546001600160801b038116835260801c6020830152565b6040519061420382613df9565b601d546001600160801b038116835260801c6020830152565b9190820391821161139857565b60ff60085460181c166142bf5761429a6001600160a01b036142496145bd565b9593509691935050169060003083146000146142ab5750614277916000526009602052604060002054613f22565b905b6142936001600160801b03918280875116915116906144d7565b1692615e03565b808210156142a6575090565b905090565b905060409181526009602052205490614279565b50600090565b60ff60085460181c166142bf5761430d6001600160a01b036142e56145bd565b929694509250506143066001600160801b03918280855116915116906144d7565b1690615d69565b9216906000308303614331575061429a916000526009602052604060002054613f22565b905060409181526009602052205461429a565b6040519061435182613e2b565b6018546001600160a01b038116835260a01c63ffffffff1660208301526019546001600160b81b03166040830152601a546060830152601b546080830152565b6001600160801b03908181116143a5571690565b60405162461bcd60e51b815260206004820152602760248201527f53616665436173743a2076616c756520646f65736e27742066697420696e203160448201527f32382062697473000000000000000000000000000000000000000000000000006064820152608490fd5b6001600160a01b0380600454163314159081614488575b8161445b575b8161444c575b5061443a57565b604051631d1e647b60e01b8152600490fd5b90506001541633141538614433565b337f000000000000000000000000000000000000000000000000000000000000000082161415915061442d565b8091506002541633141590614427565b6001600160a01b03806002541633141590816144c8575b506144b657565b604051636f54526960e01b8152600490fd5b905060015416331415386144af565b6001600160801b03918216908216039190821161139857565b604051906144fd82613e2b565b60006080838281528260208201528260408201528260608201520152565b6002600e541461452c576002600e55565b60405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152606490fd5b6040519061457e82613e2b565b81608060175463ffffffff80821684528160201c16602084015267ffffffffffffffff808260401c16604085015281831c16606084015260c01c910152565b600090819081806145cc6144f0565b508060206040516145dc81613df9565b828152015260206040516145ef81613df9565b82815201526145fc614571565b93614606856146d0565b8051909490156146525750505060608201519260808301519260a08101519267ffffffffffffffff806020840151166060850152604083015116608084015260e060c083015192015190565b929093506146619491946141d0565b9061466a6141f6565b90565b8181029291811591840414171561139857565b811561468a570490565b634e487b7160e01b600052601260045260246000fd5b519067ffffffffffffffff821682036105d157565b9190916001600160801b038080941691160191821161139857565b60408051929167ffffffffffffffff9190610100850183811186821017613e1557825260009182865260208087019284845282880190858252606089019386855260808a019587875260a08b0198888a5260c08c0194835161473181613df9565b8a81528a88820152865260e08d0193805161474b81613df9565b8b81528b8982015285528d8187019084825116421415806149e7575b61477c575b5050505050505050505050505050565b6147a191600186925261478d6141d0565b8a526147976141f6565b885251164261421c565b916001600160801b039b8c89515116156000146149aa57805b836001600160a01b03601354169160648860808d015116918351948593849263cd3181d560e01b84528b6004850152602484015260448301525afa91821561499f578094819361493a575b505050928480936148309361483997670de0b6b3a764000099971690521684528c875151169061466d565b9151169061466d565b04808652878115159182614920575b505080614907575b614860575b80808080808061476c565b63ffffffff9184918861487c81895116925192828451166146b5565b1690528786511688614893865192828451166146b5565b16905201511692836148a6575b80614855565b6148f8946148e26148d4620186a06148c38a986148e8965161466d565b04808452878787510151169061466d565b91868551511690519061421c565b90614680565b80965251019316828451166146b5565b169052388080808080806148a0565b50866149198651828651511690613f22565b1115614850565b816149319293508451511690613f22565b11158738614848565b919450915083813d8111614998575b6149538183613e5b565b810103126149955750670de0b6b3a76400009492848361483093826149868e61497f6148399b996146a0565b94016146a0565b94979950509381959750614805565b80fd5b503d614949565b8451903d90823e3d90fd5b8c87515116620186a090808202918204036149d3576149ce908e8b51511690614680565b6147ba565b634e487b7160e01b82526011600452602482fd5b5060ff60085460381c1615614767565b600090600090600090600090614a0b6144f0565b50614a14614571565b90614a1e826146d0565b8051151580614a2b575050565b935095509250925092606081015192608092838301519360a084015193606084019067ffffffffffffffff80835116928487018281511692602095868601828151169189604097888a019486865116918a519485528c8501528984015260608301527fc63977c8e2362a31182dc8e89a52252f9836922738e0abcfc0de6924972eafe591a18d838251168751918252898201528d87820152898d60608301527f2b5229f33f1d24d5baab718e1e25d0d86195a9b6d786c2c0868edfb21a460e2591a151938285169052519181831690524216838901524363ffffffff1680895285890151861b67ffffffff00000000169160c01b7fffffffffffffffff000000000000000000000000000000000000000000000000169342901b6fffffffffffffffff00000000000000001691171790851b77ffffffffffffffff0000000000000000000000000000000016171760175560c081015191806001600160801b039384815116614bb0906001600160801b03166001600160801b0319601c541617601c55565b015191601c54846001600160801b03198095881b16911617601c5560e0015183815116614bf3906001600160801b03166001600160801b0319601d541617601d55565b0151601d54931b16911617601d5582614c0857565b613da083305b6001600160a01b0316908115614c6d577fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef602082614c50600094600b54613f22565b600b558484526009825260408420818154019055604051908152a3565b60405162461bcd60e51b815260206004820152601f60248201527f45524332303a206d696e7420746f20746865207a65726f2061646472657373006044820152606490fd5b519081151582036105d157565b600090600090614ccd614344565b90604091828101936001600160b81b039384865116421415600014614ef25760046001600160a01b039660608886511684519384809263bd9a548b60e01b82525afa908115614ee557849885938693614e6e575b5090614d8483927fc1f41e029acf5127d111625602160c4cee3e1a4d38e691e50544d1f7c68b77be9695949a859c614e3c575b42168093528460608a01528360808a01528851166001600160a01b03166001600160a01b03196018541617601855565b602087015163ffffffff60a01b1963ffffffff60a01b6018549260a01b1691161760185576ffffffffffffffffffffffffffffffffffffffffffffff19601954161760195581601a5580601b5582519182526020820152a15b60808201614df1815160608501519061421c565b91620186a09280840293840403614e285750614e1663ffffffff926020925190614680565b920151161015614e2257565b60019350565b634e487b7160e01b81526011600452602490fd5b7ffc131c36b7e444dacda44901fd43641dcdcfdc43fe9e2601b3c1dd87061db9e56020838c51168951908152a1614d54565b9950915091506060883d8211614edd575b81614e8c60609383613e5b565b81010312614ed957907fc1f41e029acf5127d111625602160c4cee3e1a4d38e691e50544d1f7c68b77be9291614ec189614cb2565b60208a01519984015190999394509190614d84614d21565b8380fd5b3d9150614e7f565b50505051903d90823e3d90fd5b50606082015160808301519095509350614ddd565b7fdcbc1c05240f31ff3ad067ef1ee35ce4997762752e3a095284754544f4c709d79192936150066001600160a01b03926001600160801b039081614f4e88828451166146b5565b168152614f9582602083019281614f688c828751166146b5565b168452614f77828c1688614c0e565b51166001600160801b03166001600160801b0319601c541617601c55565b5181601c54916001600160801b03199060801b16911617601c55604051906323b872dd60e01b60208301523360248301523060448301528616606482015260648152614fe081613e2b565b7f0000000000000000000000000000000000000000000000000000000000000000615b22565b604080516001600160801b03958616815295909416602086015216923392819081015b0390a3565b9091926001600160a01b03918286169586330361529b575b5061506b6150526141f6565b6001600160801b039182918280865116915116906144d7565b16908086169182811061527257508061508787828651166144d7565b1683526020926150c682858301926150a28b838651166144d7565b93828516905251166001600160801b03166001600160801b0319601c541617601c55565b81601c54916001600160801b03199060801b16911617601c55861691871561522457876000526009815260409283600020548181106151d55792614fe08360008c7fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef7ffbde797d201c681b91056529119e0b02407c7bb96a4a2c75c01fc9667232c8db9b9a9997876151ac986151d09b85875260098452038a86205580600b5403600b558951908152a3845163a9059cbb60e01b918101919091526001600160a01b0387166024820152604481019290925281606481015b03601f198101835282613e5b565b516001600160801b0395861681529590941660208601521692339281906040820190565b0390a4565b845162461bcd60e51b815260048101849052602260248201527f45524332303a206275726e20616d6f756e7420657863656564732062616c616e604482015261636560f01b6064820152608490fd5b6084906040519062461bcd60e51b82526004820152602160248201527f45524332303a206275726e2066726f6d20746865207a65726f206164647265736044820152607360f81b6064820152fd5b6040516362ddb6d760e11b815260048101919091526001600160801b0387166024820152604490fd5b86600052600a6020526040600020336000526020526040600020549060001982036152c7575b50615046565b6152de6152e6926001600160801b0389169061421c565b9033906140ce565b38806152c1565b90600f54918215615389576001600160a01b036153086141f6565b91169161532360009284845260208052604084205490615e34565b92831561537f578252601f602052604082205492831561537757670de0b6b3a7640000916153509161466d565b0490620186a091828102928184041490151715614e2857509061537291614680565b111590565b505091505090565b5050505050600190565b505050600190565b919061539b6141f6565b6001600160801b0390816153bd816153b16141d0565b511682845116906144d7565b1691808616928381106152725750806153e36153d98585615cff565b97828551166146b5565b1682526154228160208401936153fe828a16838751166146b5565b94828616905251166001600160801b03166001600160801b0319601d541617601d55565b601d54916001600160801b03199060801b16911617601d5533600052602080526040600020615452858254613f22565b90556001600160a01b038216918130840361549e575b50506040519081528360208201527f01348584ec81ac7acd52b7d66d9ade986dd909f3d513881c190fc31c90527efe60403392a3565b60405163a9059cbb60e01b60208201526001600160a01b03909216602483015260448201526154d490614fe0816064810161519e565b3881615468565b909291926001600160a01b0391827f00000000000000000000000000000000000000000000000000000000000000001691604084815197634b82009360e01b8952169485600489015260209360009885816024818d8b5af180156156ae57615676575b50868952601f8552828920615554858254613f22565b905561556284601e54613f22565b601e55169330850361559d575b507fa32435755c235de2976ed44a75a2f85cb01faf0c894f639fe0c32bb9455fea8f939495965051908152a3565b87906155f583516323b872dd60e01b87820152876024820152306044820152856064820152606481526155cf81613e2b565b7f0000000000000000000000000000000000000000000000000000000000000000615b22565b803b156156725781906044845180948193636e553f6560e01b83528860048401523060248401525af18015615668577fa32435755c235de2976ed44a75a2f85cb01faf0c894f639fe0c32bb9455fea8f9596979850615659575b879695945061556f565b61566290613e47565b3861564f565b82513d8a823e3d90fd5b5080fd5b8581813d83116156a7575b61568b8183613e5b565b810103126156a35761569c90614cb2565b503861553e565b8980fd5b503d615681565b84513d8c823e3d90fd5b6001600160a01b03807f0000000000000000000000000000000000000000000000000000000000000000169260409082825196634b82009360e01b885216958660048201526020936000918581602481868c5af180156158575761581f575b50878252601f855283822061572d87825461421c565b905561573b86601e5461421c565b601e55821695308703615778575b505050907fbc290bb45104f73cf92115c9603987c3f8fd30c182a13603d8cffa49b5f5995291519283523392a4565b803b15615672578180916024865180958193630e5a77ed60e21b83528b60048401525af190811561499f57506155cf7fbc290bb45104f73cf92115c9603987c3f8fd30c182a13603d8cffa49b5f59952959493615807938893615810575b50845163a9059cbb60e01b878201526001600160a01b0390911660248201526044810192909252816064810161519e565b90913880615749565b61581990613e47565b386157d6565b8581813d8311615850575b6158348183613e5b565b8101031261584c5761584590614cb2565b5038615717565b8280fd5b503d61582a565b85513d85823e3d90fd5b93907f9dc1449a0ff0c152e18e8289d865b47acc6e1b76b1ecb239c13d6ee22a9206a792916001600160801b03948561589d84828a51166144d7565b16875260208701866158b286828451166144d7565b1681526158ff876001600160a01b03809516998a6000526020805260406000206158df838a16825461421c565b905551166001600160801b03166001600160801b0319601d541617601d55565b5186601d54916001600160801b03199060801b16911617601d551693308503615947575b50604080516001600160801b03928316815292909116602083015281908101615029565b61597c90604051906323b872dd60e01b60208301528660248301523060448301528316606482015260648152614fe081613e2b565b38615923565b7f7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff81116159ac5790565b60405162461bcd60e51b815260206004820152602860248201527f53616665436173743a2076616c756520646f65736e27742066697420696e206160448201527f6e20696e743235360000000000000000000000000000000000000000000000006064820152608490fd5b8051156113e35760200190565b80518210156113e35760209160051b010190565b60209081818403126105d15780519067ffffffffffffffff82116105d157019180601f840112156105d1578251615a6e81613e7d565b93615a7c6040519586613e5b565b818552838086019260051b8201019283116105d1578301905b828210615aa3575050505090565b81518152908301908301615a95565b91908110156113e35760051b0190565b356001600160a01b03811681036105d15790565b6001600160a01b03600154163303615aea57565b604051630e05f48560e11b8152600490fd5b6001600160a01b03600054163303615b1057565b604051633d71279960e21b8152600490fd5b6001600160a01b031690604051615b3881613df9565b6020928382527f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564848301526000808486829651910182855af13d15615c64573d9167ffffffffffffffff8311615c505790615bb393929160405192615ba688601f19601f8401160185613e5b565b83523d868885013e615c6e565b90815180615bc2575b50505050565b82849181010312614995575081615bd99101614cb2565b15615be657808080615bbc565b6084906040519062461bcd60e51b82526004820152602a60248201527f5361666545524332303a204552433230206f7065726174696f6e20646964206e60448201527f6f742073756363656564000000000000000000000000000000000000000000006064820152fd5b634e487b7160e01b85526041600452602485fd5b90615bb392916060915b91929015615cd05750815115615c82575090565b3b15615c8b5790565b60405162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e74726163740000006044820152606490fd5b825190915015615ce35750805190602001fd5b60405162461bcd60e51b8152908190610a179060048301613c20565b91906001600160801b038084511615600014615d1a57509150565b615d53906020850190615d4a81615d41615d37828651168861466d565b828a511690614680565b9751168761466d565b91511690614680565b10615d5a57565b90600181018091116113985790565b6001600160801b038082511615600014615d8257505090565b615d4a61466a93826020850151169061466d565b90916001600160801b038083511615600014615db25750505090565b602083959492930190615dd5615dcb828451168561466d565b8288511690614680565b9584615de6575b50505050615d5a57565b615df993945081615d4a9151168761466d565b1038808080615ddc565b60208101906001600160801b03908183511615600014615e235750505090565b61466a9382615d4a9251169061466d565b919060208301926001600160801b038085511615600014615e56575090925050565b9081615d4a81615d41615d37615d53968651168861466d565b909160208201916001600160801b038084511615600014615e91575050505090565b615dd5615dcb8284989795969851168561466d56fea164736f6c6343000813000ac1f41e029acf5127d111625602160c4cee3e1a4d38e691e50544d1f7c68b77bed833147d7dc355ba459fc788f669e58cfaf9dc25ddcd0702e87d69c7b5124289000000000000000000000000000000000000000000000000000000000000006000000000000000000000000000000000000000000000000000000000000001c000000000000000000000000000000000000000000000000000000000000002400000000000000000000000000000000000000000000000000000000000000140000000000000000000000000853d955acef822db058eb8505911ed77f175b99e000000000000000000000000f43211935c781d5ca1a41d2041f397b8a7366c7a000000000000000000000000013723e5631c591af50e89c2892b464530103481000000000000000000000000000000000000000000000000000000000000138800000000000000000000000018500cb1f2fe7a40ebda393383a0b8548a31f2610000000000000000000000000000000000000000000000000000000235ef7f6800000000000000000000000000000000000000000000000000000000000124f8000000000000000000000000000000000000000000000000000000000000271000000000000000000000000000000000000000000000000000000000000007d000000000000000000000000045127c7fa0c3ce6a18fec9604d96edfbbc8a97000000000000000000000000000000000000000000000000000000000000000060000000000000000000000000fd3065c629ee890fd74f43b802c2fea4b7279b8c000000000000000000000000168200cf227d4543302686124ac28ae0eaf2ca0b0000000000000000000000008412ebf45bac1b340bbe8f318b928c466c4e39ca0000000000000000000000000000000000000000000000000000000000000100000000000000000000000000000000000000000000000000000000000000006000000000000000000000000000000000000000000000000000000000000000c000000000000000000000000000000000000000000000000000000000000000120000000000000000000000000000000000000000000000000000000000000039467261786c656e6420496e7465726573742042656172696e672046524158202843757276652e6669204554482f66727845544829202d203231000000000000000000000000000000000000000000000000000000000000000000000000000013664652415828667278455448435256292d323100000000000000000000000000
Deployed Bytecode
0x6080604052600436101561001257600080fd5b60003560e01c8062348d1814613bc757806301e1d11414613b9557806302ce728f14613b585780630475260e14613b03578063064f3fbd14613abf57806306fdde0314613a0157806307a2d13a146139d45780630880b2f01461398157806308a0c37514613921578063090f3f50146138fa578063095ea7b3146138d45780630a28a477146138a75780630a67918c146138845780630b4aecf0146138615780630c70661d1461383b578063115a334c1461381557806311a2e4bc146137f757806318160ddd146137d65780631bc23cf91461374c5780631c2591d3146137145780631c6c95971461367857806323b872dd146135b95780632b3ba681146114d5578063313ce5671461357b57806334680fe51461355557806338d52e0f1461351157806339030864146134c257806339509351146134705780633d417d2d146133eb5780633f2617cb146133665780633f4ba83a14613197578063402d267d1461313d57806345014095146130d357806345757e4a146130ad5780634732428c1461308f578063492924271461306857806349eb7af6146109005780634ac8eb5f1461304a5780634b4b418e14612fc45780634bc66f3214612f9d5780634c18a4fb14612f0f5780634c41799514612ebe5780634cdad506146117375780634f8b4ae714612e2d5780634fd422df14612df457806354fd4d5014612dca57806364e51d5d14612d6d57806367800b5f14612d4757806369026e8814612cf25780636b96668f14612c695780636cd3cc77146109005780636e553f6514612be157806370a0823114612ba7578063715018a614612b43578063721b0a471461282257806379ba50971461274d5780637d37bdd7146127155780637d63fbc2146126ef5780637ec4b571146126b25780637ecefa6e146126595780638142dd53146125995780638285ef401461256757806382beee89146124f05780638456cb5914612304578063858f1e6814612266578063886c033a1461220d5780638cad7fbe146121ce5780638da5cb5b146121a75780638f791f8b1461207b57806393f46f641461203e57806394bf804d14611fa357806395d14ca814611f4d57806395d89b4114611e4c57806399530b0614611dd25780639a295e7314611d7e5780639bc6ab8614611d585780639fe34bdc14610900578063a053db681461182d578063a457c2d71461176d578063a9059cbb1461173c578063b3d7f6b914611737578063b460af94146116cb578063b5af306214611691578063b68d0a091461162b578063b78294dd14610900578063b7db54f514611605578063b8ca3b83146115e8578063ba08765214611556578063bbb0962414611531578063bdc8144b146114da578063c0a7e892146114d5578063c58e4df6146114af578063c63d75b61461144f578063c6e1c7c91461140b578063c6e6f59214610720578063ca2298fe14610ecf578063cacf3b5814610ea0578063cadac47914610e4f578063cdd72d5214610dee578063ce96cb7714610dcb578063d2a156e014610d87578063d41ddc9614610cdc578063d905777e14610cb9578063daf33f2a14610bbb578063dd62ed3e14610b69578063e1e9277514610b1c578063e203641714610af9578063e30c397814610ad2578063e4b0007014610a81578063e551d11d14610a63578063e5f13b1614610905578063e63a391f14610900578063e7a33174146108a9578063e8596f7214610854578063eafecffa14610836578063ebd462cb1461079a578063ecf708581461077c578063eee2421914610755578063ef8b30f714610720578063f211c390146106fa578063f2fde38b14610690578063f384bd0514610672578063f6ccaad41461060c578063f9557ccb146105d65763fbbbf94c1461057a57600080fd5b346105d15760003660031901126105d15760a06018546001600160b81b0360195416601a54601b549163ffffffff604051946001600160a01b0381168652861c166020850152604084015260608301526080820152f35b600080fd5b346105d15760003660031901126105d157601c54604080516001600160801b038316815260809290921c602083015290f35b0390f35b346105d15760003660031901126105d157610625615afc565b6001600160a01b0319806000541660005560015490336001600160a01b0383167f31b6c5a04b069b6ec1b3cef44c4e7c1eadd721349cda9823d0b1877b3551cdc6600080a3163317600155005b346105d15760003660031901126105d1576020600f54604051908152f35b346105d15760203660031901126105d1576106a9613c69565b6106b1613eca565b6001600160a01b0380911690816001600160a01b03196003541617600355600254167f38d16b8cac22d99fc7c124b9cd0de2d3fa1faef420bfe791d8c362d765e22700600080a3005b346105d15760003660031901126105d157602060ff60085460381c166040519015158152f35b346105d15760203660031901126105d157602061074d61073e6145bd565b50935050505060043590615d69565b604051908152f35b346105d15760003660031901126105d15760206001600160a01b0360135416604051908152f35b346105d15760003660031901126105d1576020600754604051908152f35b346105d15760203660031901126105d1576107b3613cbf565b8015610829576107c1614410565b6008549060ff8260201c16610817577fc56dd3e14f5af3a74c61b7cdf855a3d8ab4401c78c0622a4d312de8a8f8736a29160209115159063ff0000008260181b169063ff000000191617600855604051908152a1005b60405163035b48f760e31b8152600490fd5b610831614498565b6107c1565b346105d15760003660031901126105d1576020601254604051908152f35b346105d15760003660031901126105d15761086d614410565b60ff600654166108175760006005557fbf1ce7fb3a8e648b70ea830f99b52f7ea31554186d29763280751f42e77f6386602060405160008152a1005b346105d15760203660031901126105d1576004356108c5614498565b60ff60065416610817576020817fbf1ce7fb3a8e648b70ea830f99b52f7ea31554186d29763280751f42e77f638692600555604051908152a1005b613dc5565b346105d15760603660031901126105d157600435602435610924613c95565b9061092d61451b565b6001600160a01b03821615610a51576109446149f7565b5050505050600554610961846001600160801b03601d5416613f22565b11610a3f5761096e614cbf565b505015610a2d5761098e928161098992610a1b575b50614391565b615391565b610996614344565b50601b54906109a582336152ed565b156109ba576020906001600e55604051908152f35b506109da6109c66141f6565b336000526020805260406000205490615e34565b33600052601f602052610a1760406000205492604051938493633b49de0f60e21b8552600485016040919493926060820195825260208201520152565b0390fd5b610a27903390336154db565b84610983565b60405163345513d960e01b8152600490fd5b6040516397ba4de360e01b8152600490fd5b604051631e4ec46b60e01b8152600490fd5b346105d15760003660031901126105d1576020600554604051908152f35b346105d15760003660031901126105d157610a9a615ad6565b6201000062ff00001960215416176021557f70328969870b42d0fc62ce5946530c50a466c5ad21af004a9f2f3a8eda9a4a0c600080a1005b346105d15760003660031901126105d15760206001600160a01b0360035416604051908152f35b346105d15760003660031901126105d157602060ff600854166040519015158152f35b346105d15760003660031901126105d157610b35615ad6565b600160ff1960215416176021557faa96740f913149dce2173396218295b4c082e86fabacc37ac8d45305239d26f3600080a1005b346105d15760403660031901126105d157610b82613c69565b610b8a613c7f565b906001600160a01b03809116600052600a602052604060002091166000526020526020604060002054604051908152f35b346105d15760403660031901126105d157610bd4613de3565b610bdc613c7f565b90610be5613eca565b806001600160a01b038316928315610a5157602093610c026141d0565b926001600160801b0380911615610ca1575b917faf48306b6b4f0ba30d00f05b21559d8d29934142980a553d8a014780c6c7e4529391610c6960809487169383610c4c8683615e34565b98610c588733306140ce565b3092610c638b614391565b9061502e565b30600052601f8752610c83604060002054923090846156b8565b604051928352868301528460408301526060820152a1604051908152f35b30600090815260098752604090205481169450610c14565b346105d15760203660031901126105d157602061074d610cd7613c69565b6142c5565b346105d15760403660031901126105d157610cf5613c7f565b610cfd61451b565b6001600160a01b03811615610a5157610d146149f7565b50505050503360005260208052604060002054610d67575b610d3a9033906004356156b8565b610d42614344565b50601b54610d5081336152ed565b15610d5c576001600e55005b6109da6109c66141f6565b610d6f614cbf565b5050610d2c5760405163345513d960e01b8152600490fd5b346105d15760003660031901126105d15760206040516001600160a01b037f000000000000000000000000eb8816baeb70690660ce6c0eda2b07a21493e664168152f35b346105d15760203660031901126105d157602061074d610de9613c69565b614229565b346105d15760003660031901126105d15760a0610e096145bd565b9350935050506001600160801b03908160208185511694015116916020818351169201511690601e54926040519485526020850152604084015260608301526080820152f35b346105d15760403660031901126105d157610e68613c7f565b610e7061451b565b6001600160a01b03811615610a5157610e9990610e8b6149f7565b5050505050600435336154db565b6001600e55005b346105d15760003660031901126105d157610608610ebc6145bd565b9260409694969291925196879687613cfa565b346105d15760803660031901126105d157610ee8613c69565b60643567ffffffffffffffff8082116105d157366023830112156105d1578160040135116105d157366024826004013560051b830101116105d157610f2b61451b565b610f336149f7565b5050505050610f40614cbf565b505015610a2d576001600160a01b038216600052601460205260ff60406000205416156113f9578060040135156113e3576001600160a01b03610f8560248301615ac2565b817f000000000000000000000000f43211935c781d5ca1a41d2041f397b8a7366c7a1691829116036113ae576004820135600019810190811161139857610fdc610fd782856004013560248701615ab2565b615ac2565b6001600160a01b03807f000000000000000000000000853d955acef822db058eb8505911ed77f175b99e1691160361132f575061101c33306024356156b8565b60405163095ea7b360e01b81526001600160a01b0384166004820152602480359082015290602090829060449082906000905af18015611267576112f6575b506040516370a0823160e01b8152306004820152906020826024817f000000000000000000000000853d955acef822db058eb8505911ed77f175b99e6001600160a01b03165afa918215611267576000926112c2575b5060405180916338ed173960e01b82526024356004830152604435602483015260a06044830152806004013560a483015260c4820190602481019060005b816004013581106112965750505090806000923060648301524260848301520381836001600160a01b0388165af1801561126757611273575b506040516370a0823160e01b8152306004820152906020826024817f000000000000000000000000853d955acef822db058eb8505911ed77f175b99e6001600160a01b03165afa801561126757600090611233575b611187925061421c565b906044358210611213576111996141f6565b6111c36111a68483615d69565b916111b085614391565b339130916111bd86614391565b91615861565b6001600160a01b0360405192168252602435602083015282604083015260608201527fe947f0f9b6255bdcf76d13d1257d34fbe380e0d5d4daa75e61c783a41e1607ba60803392a2610996614344565b604051633b5d56ed60e11b81526044803560048301526024820184905290fd5b506020823d60201161125f575b8161124d60209383613e5b565b810103126105d157611187915161117d565b3d9150611240565b6040513d6000823e3d90fd5b61128f903d806000833e6112878183613e5b565b810190615a38565b5082611128565b919350916020806001926001600160a01b036112b188613cab565b1681520194019101918493926110ef565b9091506020813d6020116112ee575b816112de60209383613e5b565b810103126105d1575190836110b1565b3d91506112d1565b6020813d602011611327575b8161130f60209383613e5b565b810103126105d15761132090614cb2565b508261105b565b3d9150611302565b611348610fd7610a179285602481600401359101615ab2565b60405163b0b3262d60e01b81527f000000000000000000000000853d955acef822db058eb8505911ed77f175b99e6001600160a01b03908116600483015290911660248201529081906044820190565b634e487b7160e01b600052601160045260246000fd5b6113ba60248301615ac2565b60405163b0b3262d60e01b81526001600160a01b03928316600482015291166024820152604490fd5b634e487b7160e01b600052603260045260246000fd5b604051631311dc6d60e01b8152600490fd5b346105d15760003660031901126105d15760206040516001600160a01b037f000000000000000000000000f43211935c781d5ca1a41d2041f397b8a7366c7a168152f35b346105d15760203660031901126105d157611468613c69565b50602061074d6114766145bd565b5093505050506001600160801b0381511660075490818110156000146114a0575050600090615d69565b6114a99161421c565b90615d69565b346105d15760003660031901126105d157602060ff60085460281c166040519015158152f35b613da2565b346105d15760203660031901126105d1576004356114f6614498565b60ff60085416610817576020817f854df3eb95564502c8bc871ebdd15310ee26270f955f6c6bd8cea68e75045bc092600755604051908152a1005b346105d15760003660031901126105d157602060085460ff60405191831c1615158152f35b346105d15761156436613e95565b919061156e61451b565b6001600160a01b03811615610a515760ff60085460181c166115d6576020926115c9916115996149f7565b50505050506115a66141d0565b6115b08582615e03565b946115c36115bd87614391565b91614391565b9161502e565b6001600e55604051908152f35b60405163e0a3980360e01b8152600490fd5b346105d15760003660031901126105d157602060405161c3508152f35b346105d15760003660031901126105d157602060ff60215460081c166040519015158152f35b346105d15760203660031901126105d1576001600160a01b0361164c613c69565b1660005260096020526040600020546020805260406000205490601f602052610608604060002054604051938493846040919493926060820195825260208201520152565b346105d15760203660031901126105d1576001600160a01b036116b2613c69565b16600052601f6020526020604060002054604051908152f35b346105d1576116d936613e95565b91906116e361451b565b6001600160a01b03811615610a515760ff60085460181c166115d6576020926115c99161170e6149f7565b505050505061171b6141d0565b61172e6117288683615cff565b95614391565b6115c386614391565b6139d4565b346105d15760403660031901126105d157611762611758613c69565b6024359033613f2f565b602060405160018152f35b346105d15760403660031901126105d157611786613c69565b6024359033600052600a60205260406000206001600160a01b038216600052602052604060002054918083106117c257611762920390336140ce565b60405162461bcd60e51b815260206004820152602560248201527f45524332303a2064656372656173656420616c6c6f77616e63652062656c6f7760448201527f207a65726f0000000000000000000000000000000000000000000000000000006064820152608490fd5b346105d15760a03660031901126105d157611846613c69565b6084359067ffffffffffffffff82116105d157366023830112156105d157816004013561187281613e7d565b926118806040519485613e5b565b818452602084016024819360051b830101913683116105d157602401905b828210611d40575050506118b061451b565b6118b86149f7565b50505050506118c5614cbf565b505015610a2d576001600160a01b038216600052601460205260ff60406000205416156113f9576001600160a01b036118fd84615a17565b5116926001600160a01b037f000000000000000000000000853d955acef822db058eb8505911ed77f175b99e16809403611d0157805160001990818101908111611398576119536001600160a01b039184615a24565b51166001600160a01b037f000000000000000000000000f43211935c781d5ca1a41d2041f397b8a7366c7a1603611c8c5750604435611c7a575b6119a261199b602435614391565b3090615391565b60405163095ea7b360e01b81526001600160a01b038516600482015260248035908201529094602090829060449082906000905af1801561126757611c41575b506040516370a0823160e01b8152306004820152916020836024817f000000000000000000000000f43211935c781d5ca1a41d2041f397b8a7366c7a6001600160a01b03165afa92831561126757600093611c0d575b506040516338ed173960e01b81526024803560048301526064359082015260a06044820152915160a48301819052829160c483019160005b818110611beb5750505090806000923060648301524260848301520381836001600160a01b0388165af1801561126757611bd0575b506040516370a0823160e01b8152306004820152906020826024817f000000000000000000000000f43211935c781d5ca1a41d2041f397b8a7366c7a6001600160a01b03165afa801561126757600090611b9c575b611b04925061421c565b6064358110611b7c57611b183382306154db565b611b2481604435613f22565b926001600160a01b036040519316835260243560208401526040830152604435606083015260808201527fb19ca0df3f3a01af950d8e6ad62aeff167cf14c73e98af6c52afef1add5c97ed60a03392a2610996614344565b60449060405190633b5d56ed60e11b825260643560048301526024820152fd5b506020823d602011611bc8575b81611bb660209383613e5b565b810103126105d157611b049151611afa565b3d9150611ba9565b611be4903d806000833e6112878183613e5b565b5083611aa5565b82516001600160a01b0316845285945060209384019390920191600101611a70565b9092506020813d602011611c39575b81611c2960209383613e5b565b810103126105d157519185611a38565b3d9150611c1c565b6020813d602011611c72575b81611c5a60209383613e5b565b810103126105d157611c6b90614cb2565b50846119e2565b3d9150611c4d565b611c8733604435336154db565b61198d565b815190810190811161139857611cad6001600160a01b0391610a1793615a24565b5160405163b0b3262d60e01b81526001600160a01b037f000000000000000000000000f43211935c781d5ca1a41d2041f397b8a7366c7a811660048301529290911690911660248201529081906044820190565b6001600160a01b03611d138592615a17565b5160405163b0b3262d60e01b81526001600160a01b03938416600482015291169091166024820152604490fd5b60208091611d4d84613cab565b81520191019061189e565b346105d15760003660031901126105d157602060ff60215460181c166040519015158152f35b346105d15760003660031901126105d157610100604051620186a0808252806020830152806040830152806060830152670de0b6b3a76400009081608084015260a083015260c082015261c35060e0820152f35b346105d15760003660031901126105d157611deb6145bd565b509350505050602081016001600160801b03918282511615600014611e20575050506020670de0b6b3a7640000604051908152f35b51670de0b6b3a76400009083168181029182040361139857602092611e4792511690614680565b61074d565b346105d15760003660031901126105d157604051600090601654600181811c90808316928315611f43575b6020938484108114611f2d57838652908115611f0d5750600114611eb2575b61060884611ea681880382613e5b565b60405191829182613c20565b601660009081529294507fd833147d7dc355ba459fc788f669e58cfaf9dc25ddcd0702e87d69c7b51242895b828410611efa575050508161060893611ea69282010193611e96565b8054858501870152928501928101611ede565b60ff1916858501525050151560051b8201019150611ea681610608611e96565b634e487b7160e01b600052602260045260246000fd5b91607f1691611e77565b346105d15760003660031901126105d15760a06017546040519063ffffffff80821683528160201c16602083015267ffffffffffffffff808260401c1660408401528160801c16606083015260c01c6080820152f35b346105d15760403660031901126105d157600435611fbf613c7f565b611fc761451b565b6001600160a01b03811615610a5157611fde6149f7565b5050505050611feb6141d0565b91611ff68184615e03565b9160075461200e846001600160801b03875116613f22565b1161202c576020936115c9926120266115bd86614391565b91614f07565b604051630aad288560e21b8152600490fd5b346105d157602061204e36613cce565b1561206a5761074d9161205f6145bd565b945050505050615d96565b611e47916120766141f6565b615d96565b346105d15760403660031901126105d157612094613c69565b63ffffffff60243581811692918382036105d1576120b0615ad6565b60ff60215416612195576080937f78ba1c32ac8ea4b3d51133dd0b6f5d8f98e23797aade6afc381ea317d5d4f28b85612135936120eb614344565b966001600160a01b0390818951169260208a015116604051938452602084015216938460408301526060820152a16001600160a01b03166001600160a01b03196018541617601855565b63ffffffff60a01b1963ffffffff60a01b6018549260a01b169116176018556001600160b81b0360408201511676ffffffffffffffffffffffffffffffffffffffffffffff1960195416176019556060810151601a550151601b55600080f35b604051631186953760e31b8152600490fd5b346105d15760003660031901126105d15760206001600160a01b0360025416604051908152f35b346105d15760203660031901126105d1576001600160a01b036121ef613c69565b166000526014602052602060ff604060002054166040519015158152f35b346105d15760203660031901126105d1577f3ff713beec3d10b4dfe28953471682eab1f857ba2fdb6367366252381888a750602060043561224c615ad6565b600160ff19600854161760085580600755604051908152a1005b346105d15760203660031901126105d15761227f613cbf565b80156122f75761228d614410565b60ff60085460401c166108175760207fdea8bb46eee4300a7d2de86939c245f568dc5994576194cbfb69969e010dcb67916122c66149f7565b5050505050151560085467ff000000000000008260381b169067ff00000000000000191617600855604051908152a1005b6122ff614498565b61228d565b346105d15760003660031901126105d15761231d614410565b60ff8060065416156124bb575b600890815481811615612486575b818160101c1615612449575b508154818160201c1615612408575b508154818160301c16156123c3575b50815460401c161561237057005b6123786149f7565b505050505067010000000000000067ff00000000000000198254161790557fdea8bb46eee4300a7d2de86939c245f568dc5994576194cbfb69969e010dcb67602060405160018152a1005b650100000000009065ff000000000019161782557f28bc4f9e24da61e7ba3aa697dfaefd0167093d2425c00b6190a7d3152ee6dfaa602060405160018152a182612362565b63010000009063ff00000019161782557fc56dd3e14f5af3a74c61b7cdf855a3d8ab4401c78c0622a4d312de8a8f8736a2602060405160018152a182612353565b6101009061ff0019161782557f34a71a12fa81891b738d910d4d44ffabeeb12f8bc026844db237ea8bf8ebe8be602060405160018152a182612344565b60006007557f854df3eb95564502c8bc871ebdd15310ee26270f955f6c6bd8cea68e75045bc0602060405160008152a1612338565b60006005557fbf1ce7fb3a8e648b70ea830f99b52f7ea31554186d29763280751f42e77f6386602060405160008152a161232a565b346105d15760203660031901126105d1577f4cb8c9e37efb94c6cdbd2a80fe36cee1957b5584d1a1986fa2bae115180af59a61252a613c69565b612532615ad6565b600480546001600160a01b039283166001600160a01b03198216811790925560408051939091168352602083019190915290a1005b346105d15760003660031901126105d157601d54604080516001600160801b038316815260809290921c602083015290f35b346105d15760203660031901126105d15760043563ffffffff8116908181036105d1576125c4615ad6565b60ff60085460381c166126475761c3508211612635577f58a58c712558f3d6e20bed57421eb8f73048d881dea9e5bb80efb37c49680d1c916020916126076149f7565b505050505067ffffffff0000000060175491841b169067ffffffff00000000191617601755604051908152a1005b60405163da0afa5760e01b8152600490fd5b60405163a02a2bcd60e01b8152600490fd5b346105d15760003660031901126105d157612672615ad6565b660100000000000066ff0000000000001960085416176008557f60c2acdf5b421891c8cc7302420292f2680f0e835fc76dd15f35a7bb0dd5cbc8600080a1005b346105d15760206126c236613cce565b156126de5761074d916126d36145bd565b945050505050615e6f565b611e47916126ea6141f6565b615e6f565b346105d15760003660031901126105d157602060ff60215460101c166040519015158152f35b346105d157602061272536613cce565b156127415761074d916127366145bd565b509350505050615e6f565b611e47916126ea6141d0565b346105d15760003660031901126105d1576003546001600160a01b0333818316036127b7576001600160a01b03198092166003556002549133908316176002553391167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0600080a3005b60405162461bcd60e51b815260206004820152602960248201527f4f776e61626c6532537465703a2063616c6c6572206973206e6f74207468652060448201527f6e6577206f776e657200000000000000000000000000000000000000000000006064820152608490fd5b346105d15760603660031901126105d15761283b613de3565b60243590612847613c95565b61284f61451b565b6001600160a01b038116928315610a515760ff60085460281c16612b3157804211612b13575061287d6149f7565b505050505061288a614cbf565b50905061289781836152ed565b612b01576128a36141f6565b9184600052601f60205260406000205492602080526128c6604060002054614391565b926000936001600160801b0396670de0b6b3a76400006128f2898316946128ed8688615e03565b61466d565b049860105497620186a0988901808a11611398576129118a918d61466d565b0461292461291e83615982565b91615982565b90600082820392128183128116918313901516176113985760001280159190612ae15760209b505b809960125480612a9f575b50505086959293612a00979486938b6115c99c9d6129e6958d6129856129806129ee9d8c615e34565b614391565b98600097600093612a08575b509160c093917f821de4e13fff1938b3806eb2859b6a5d55111f00dcf286f8a793584228ff36f895936040519485526020850152828b166040850152606084015281881660808401521660a0820152a26146b5565b903392615861565b6129f98133876156b8565b30836156b8565b3090306154db565b7f821de4e13fff1938b3806eb2859b6a5d55111f00dcf286f8a793584228ff36f8959391985091612a3c8860c096946144d7565b98838d818c1680612a56575b505050919395509193612991565b82955090612a6a612980612a749383615e03565b95869151166144d7565b168d52601c5484612a87858284166144d7565b16906001600160801b03191617601c55838d38612a48565b6115c99b50829a50936129e693612ace8b9a9793612ac6612a009d9a966129ee9c9961466d565b04809d61421c565b9c50935093968296508195989950612957565b506011548901808a1161139857612afb8a9160209d61466d565b0461294c565b604051633af2cafd60e11b8152600490fd5b60449060405190635ba2a8d560e01b82524260048301526024820152fd5b604051631b4b0d7760e21b8152600490fd5b346105d15760003660031901126105d157612b5c613eca565b60006001600160a01b036001600160a01b03198060035416600355600254908116600255167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e08280a3005b346105d15760203660031901126105d1576001600160a01b03612bc8613c69565b1660005260096020526020604060002054604051908152f35b346105d15760403660031901126105d157600435612bfd613c7f565b90612c0661451b565b6001600160a01b03821615610a5157612c1d6149f7565b5050505050612c2a6141d0565b600754612c41836001600160801b03845116613f22565b1161202c5760209281612c60612c5a856115c995615d69565b94614391565b61202685614391565b346105d15760203660031901126105d157612c82613c69565b612c8a615ad6565b60ff60215460101c1661219557601354604080516001600160a01b038084168252848116602083015292936001600160a01b0319939290917faeae842c8b3cd009fbb602e1ed072dc1aec69750e431ceae97f7543b466cd04c9190a116911617601355600080f35b346105d15760003660031901126105d157612d0b614410565b60ff600854166108175760006007557f854df3eb95564502c8bc871ebdd15310ee26270f955f6c6bd8cea68e75045bc0602060405160008152a1005b346105d15760003660031901126105d157602060ff60085460181c166040519015158152f35b346105d15760003660031901126105d157612d86615ad6565b6801000000000000000068ff00000000000000001960085416176008557f16c0a933c76f28f1abdcef88bcea1650397c5f4bb4bf491a0d451a65cae016b6600080a1005b346105d15760003660031901126105d1576060604051600381526000602082015260006040820152f35b346105d15760203660031901126105d1576001600160a01b03612e15613c69565b16600052602080526020604060002054604051908152f35b346105d15760003660031901126105d157612e46615ad6565b612e4e615afc565b6001600160a01b0319806000541660005560015460006001600160a01b03821681817f162998b90abc2507f3953aa797827b03a14c42dbd9a35f09feaf02e0d592773a8280a37f31b6c5a04b069b6ec1b3cef44c4e7c1eadd721349cda9823d0b1877b3551cdc68280a316600155005b346105d15760003660031901126105d157612ed7615ad6565b6201000062ff00001960085416176008557f269ac55859865c2ff127a862e95c81ce7e3b9b13582036d3df419df5c07ec8b4600080a1005b346105d15760203660031901126105d157612f28613cbf565b8015612f9057612f36614410565b6008549060ff8260301c16610817577f28bc4f9e24da61e7ba3aa697dfaefd0167093d2425c00b6190a7d3152ee6dfaa9160209115159065ff00000000008260281b169065ff0000000000191617600855604051908152a1005b612f98614498565b612f36565b346105d15760003660031901126105d15760206001600160a01b0360015416604051908152f35b346105d15760203660031901126105d157612fdd613cbf565b801561303d57612feb614410565b6008549060ff8260101c16610817577f34a71a12fa81891b738d910d4d44ffabeeb12f8bc026844db237ea8bf8ebe8be9160209115159061ff008260081b169061ff00191617600855604051908152a1005b613045614498565b612feb565b346105d15760003660031901126105d1576020601e54604051908152f35b346105d15760003660031901126105d15760206001600160a01b0360045416604051908152f35b346105d15760003660031901126105d1576020601154604051908152f35b346105d15760003660031901126105d157602060ff60085460401c166040519015158152f35b346105d15760203660031901126105d1576130ec613c69565b6130f4615ad6565b6001600160a01b0380911690816001600160a01b03196000541617600055600154167f162998b90abc2507f3953aa797827b03a14c42dbd9a35f09feaf02e0d592773a600080a3005b346105d15760203660031901126105d157613156613c69565b506001600160801b036131676145bd565b50935050505051166007548082101560001461318b57505060206000604051908152f35b602091611e479161421c565b346105d15760003660031901126105d1576131b0614498565b60ff806006541615613330575b6008908154818116156132fa575b818160101c16156132c2575b508154818160201c1615613288575b508154818160301c161561324c575b50815460401c161561320357005b61320b6149f7565b505050505067ff000000000000001981541690557fdea8bb46eee4300a7d2de86939c245f568dc5994576194cbfb69969e010dcb67602060405160008152a1005b65ff0000000000191682557f28bc4f9e24da61e7ba3aa697dfaefd0167093d2425c00b6190a7d3152ee6dfaa602060405160008152a1826131f5565b63ff000000191682557fc56dd3e14f5af3a74c61b7cdf855a3d8ab4401c78c0622a4d312de8a8f8736a2602060405160008152a1826131e6565b61ff00191682557f34a71a12fa81891b738d910d4d44ffabeeb12f8bc026844db237ea8bf8ebe8be602060405160008152a1826131d7565b7f854df3eb95564502c8bc871ebdd15310ee26270f955f6c6bd8cea68e75045bc0602060001980600755604051908152a16131cb565b7fbf1ce7fb3a8e648b70ea830f99b52f7ea31554186d29763280751f42e77f6386602060001980600555604051908152a16131bd565b346105d15760403660031901126105d15761337f613c69565b602435908115158092036105d1577fea1eefb4fd58778d7b274fe54045a9feeec8f2847899c2e71126d3a74d486da5916001600160a01b036040926133c2613eca565b16908160005260146020528260002060ff1981541660ff831617905582519182526020820152a1005b346105d15760403660031901126105d157600435613407613c7f565b9061341061451b565b6001600160a01b03821615610a515760ff60085460081c1661345e576115c960209261343a6149f7565b50505050506134476141f6565b6134518482615e34565b936129e66115bd86614391565b604051631e61c1e960e11b8152600490fd5b346105d15760403660031901126105d15761176261348c613c69565b33600052600a60205260406000206001600160a01b0382166000526020526134bb602435604060002054613f22565b90336140ce565b346105d15760003660031901126105d1576134db615ad6565b61010061ff001960215416176021557f0af6d9d6ea0e3f0cdb71562ce1fce30aa597445ea04f5b25a939cfe0a252171c600080a1005b346105d15760003660031901126105d15760206040516001600160a01b037f000000000000000000000000853d955acef822db058eb8505911ed77f175b99e168152f35b346105d15760003660031901126105d157602060ff60085460301c166040519015158152f35b346105d15760003660031901126105d157602060405160ff7f0000000000000000000000000000000000000000000000000000000000000012168152f35b346105d15760603660031901126105d1576135d2613c69565b6135da613c7f565b604435906001600160a01b038316600052600a602052604060002033600052602052604060002054926000198403613617575b6117629350613f2f565b8284106136335761362e83611762950333836140ce565b61360d565b60405162461bcd60e51b815260206004820152601d60248201527f45524332303a20696e73756666696369656e7420616c6c6f77616e63650000006044820152606490fd5b346105d15760203660031901126105d157613691613cbf565b6136996144f0565b506040516136a681613df9565b6000815260006020820152610608604051916136c183613df9565b60008352600060208401526136d461451b565b916136dd6149f7565b9296935090966136f9575b6001600e5560405196879687613cfa565b925090506137056141d0565b9061370e6141f6565b926136e8565b346105d157602061372436613cce565b156137405761074d916137356145bd565b509350505050615d96565b611e47916120766141d0565b346105d15760603660031901126105d15760443560243560043561376e615ad6565b60ff60215460181c16612195577fc9aa62b60be8f25ac9f285edbb80bde64199b3c53e1da1027058551d32695fca60c060105460115460125490604051928352602083015260408201528360608201528460808201528560a0820152a1601055601155601255005b346105d15760003660031901126105d1576020601c5460801c604051908152f35b346105d15760003660031901126105d1576020601054604051908152f35b346105d15760003660031901126105d157602060ff60085460081c166040519015158152f35b346105d15760003660031901126105d157602060ff60085460101c166040519015158152f35b346105d15760003660031901126105d157602060ff602154166040519015158152f35b346105d15760003660031901126105d157602060ff600654166040519015158152f35b346105d15760203660031901126105d157602061074d6138c56145bd565b50935050505060043590615cff565b346105d15760403660031901126105d1576117626138f0613c69565b60243590336140ce565b346105d15760003660031901126105d15760206001600160a01b0360005416604051908152f35b346105d15760203660031901126105d15760043561393d615ad6565b60ff60215460081c16612195577fe796e9ae748449310fcd1cc6718aab236c9b8d2e0e04dacb232ba564d5b338cc6040600f548151908152836020820152a1600f55005b346105d15760003660031901126105d15761399a615ad6565b630100000063ff0000001960215416176021557f1cd8398e5a04411acbddcb6451a57b51c242322c538947cea5e4a1a506700b87600080a1005b346105d15760203660031901126105d157602061074d6139f26145bd565b50935050505060043590615e03565b346105d15760003660031901126105d157604051600090601554600181811c90808316928315613ab5575b6020938484108114611f2d57838652908115611f0d5750600114613a5a5761060884611ea681880382613e5b565b601560009081529294507f55f448fdea98c4d29eb340757ef0a66cd03dbb9538908a6a81d96026b71ec4755b828410613aa2575050508161060893611ea69282010193611e96565b8054858501870152928501928101613a86565b91607f1691613a2c565b346105d15760003660031901126105d15760206040516001600160a01b037f00000000000000000000000045127c7fa0c3ce6a18fec9604d96edfbbc8a9700168152f35b346105d15760003660031901126105d157613b1c615ad6565b64010000000064ff000000001960085416176008557fb949af551d0c88280e648f9205b986bb5f1d899c425498238655ee37617c0c39600080a1005b346105d15760003660031901126105d157613b7161451b565b6060613b7b614cbf565b906001600e55604051921515835260208301526040820152f35b346105d15760003660031901126105d15760206001600160801b03613bb86145bd565b50516040519516855250505050f35b346105d15760203660031901126105d1577fee4b3f9e70b2c6499288c7b5fbef140756009cf8839be64c473b1c7cb6d616c46020600435613c06615ad6565b600160ff19600654161760065580600555604051908152a1005b6020808252825181830181905290939260005b828110613c5557505060409293506000838284010152601f8019910116010190565b818101860151848201604001528501613c33565b600435906001600160a01b03821682036105d157565b602435906001600160a01b03821682036105d157565b604435906001600160a01b03821682036105d157565b35906001600160a01b03821682036105d157565b6004359081151582036105d157565b60609060031901126105d1576004359060243580151581036105d1579060443580151581036105d15790565b9194613d839197969461014094613da09761018086019a86526020860152604085015263ffffffff8082511660608601526020820151166080850152608067ffffffffffffffff918260408201511660a08701528260608201511660c087015201511660e0840152610100830190602090816001600160801b0391828151168552015116910152565b0190602090816001600160801b0391828151168552015116910152565b565b346105d15760003660031901126105d1576020604051670de0b6b3a76400008152f35b346105d15760003660031901126105d1576020604051620186a08152f35b600435906001600160801b03821682036105d157565b6040810190811067ffffffffffffffff821117613e1557604052565b634e487b7160e01b600052604160045260246000fd5b60a0810190811067ffffffffffffffff821117613e1557604052565b67ffffffffffffffff8111613e1557604052565b90601f8019910116810190811067ffffffffffffffff821117613e1557604052565b67ffffffffffffffff8111613e155760051b60200190565b60609060031901126105d157600435906001600160a01b039060243582811681036105d1579160443590811681036105d15790565b6001600160a01b03600254163303613ede57565b606460405162461bcd60e51b815260206004820152602060248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e65726044820152fd5b9190820180921161139857565b6001600160a01b0380911691821561406357169182156140125760008281526009602052604081205491808310613fa757604082827fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef95876020965260098652038282205586815220818154019055604051908152a3565b60405162461bcd60e51b815260206004820152602660248201527f45524332303a207472616e7366657220616d6f756e742065786365656473206260448201527f616c616e636500000000000000000000000000000000000000000000000000006064820152608490fd5b60405162461bcd60e51b815260206004820152602360248201527f45524332303a207472616e7366657220746f20746865207a65726f206164647260448201526265737360e81b6064820152608490fd5b60405162461bcd60e51b815260206004820152602560248201527f45524332303a207472616e736665722066726f6d20746865207a65726f20616460448201527f64726573730000000000000000000000000000000000000000000000000000006064820152608490fd5b6001600160a01b0380911691821561417f571691821561412f5760207f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b9259183600052600a8252604060002085600052825280604060002055604051908152a3565b60405162461bcd60e51b815260206004820152602260248201527f45524332303a20617070726f766520746f20746865207a65726f206164647265604482015261737360f01b6064820152608490fd5b60405162461bcd60e51b8152602060048201526024808201527f45524332303a20617070726f76652066726f6d20746865207a65726f206164646044820152637265737360e01b6064820152608490fd5b604051906141dd82613df9565b601c546001600160801b038116835260801c6020830152565b6040519061420382613df9565b601d546001600160801b038116835260801c6020830152565b9190820391821161139857565b60ff60085460181c166142bf5761429a6001600160a01b036142496145bd565b9593509691935050169060003083146000146142ab5750614277916000526009602052604060002054613f22565b905b6142936001600160801b03918280875116915116906144d7565b1692615e03565b808210156142a6575090565b905090565b905060409181526009602052205490614279565b50600090565b60ff60085460181c166142bf5761430d6001600160a01b036142e56145bd565b929694509250506143066001600160801b03918280855116915116906144d7565b1690615d69565b9216906000308303614331575061429a916000526009602052604060002054613f22565b905060409181526009602052205461429a565b6040519061435182613e2b565b6018546001600160a01b038116835260a01c63ffffffff1660208301526019546001600160b81b03166040830152601a546060830152601b546080830152565b6001600160801b03908181116143a5571690565b60405162461bcd60e51b815260206004820152602760248201527f53616665436173743a2076616c756520646f65736e27742066697420696e203160448201527f32382062697473000000000000000000000000000000000000000000000000006064820152608490fd5b6001600160a01b0380600454163314159081614488575b8161445b575b8161444c575b5061443a57565b604051631d1e647b60e01b8152600490fd5b90506001541633141538614433565b337f000000000000000000000000eb8816baeb70690660ce6c0eda2b07a21493e66482161415915061442d565b8091506002541633141590614427565b6001600160a01b03806002541633141590816144c8575b506144b657565b604051636f54526960e01b8152600490fd5b905060015416331415386144af565b6001600160801b03918216908216039190821161139857565b604051906144fd82613e2b565b60006080838281528260208201528260408201528260608201520152565b6002600e541461452c576002600e55565b60405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152606490fd5b6040519061457e82613e2b565b81608060175463ffffffff80821684528160201c16602084015267ffffffffffffffff808260401c16604085015281831c16606084015260c01c910152565b600090819081806145cc6144f0565b508060206040516145dc81613df9565b828152015260206040516145ef81613df9565b82815201526145fc614571565b93614606856146d0565b8051909490156146525750505060608201519260808301519260a08101519267ffffffffffffffff806020840151166060850152604083015116608084015260e060c083015192015190565b929093506146619491946141d0565b9061466a6141f6565b90565b8181029291811591840414171561139857565b811561468a570490565b634e487b7160e01b600052601260045260246000fd5b519067ffffffffffffffff821682036105d157565b9190916001600160801b038080941691160191821161139857565b60408051929167ffffffffffffffff9190610100850183811186821017613e1557825260009182865260208087019284845282880190858252606089019386855260808a019587875260a08b0198888a5260c08c0194835161473181613df9565b8a81528a88820152865260e08d0193805161474b81613df9565b8b81528b8982015285528d8187019084825116421415806149e7575b61477c575b5050505050505050505050505050565b6147a191600186925261478d6141d0565b8a526147976141f6565b885251164261421c565b916001600160801b039b8c89515116156000146149aa57805b836001600160a01b03601354169160648860808d015116918351948593849263cd3181d560e01b84528b6004850152602484015260448301525afa91821561499f578094819361493a575b505050928480936148309361483997670de0b6b3a764000099971690521684528c875151169061466d565b9151169061466d565b04808652878115159182614920575b505080614907575b614860575b80808080808061476c565b63ffffffff9184918861487c81895116925192828451166146b5565b1690528786511688614893865192828451166146b5565b16905201511692836148a6575b80614855565b6148f8946148e26148d4620186a06148c38a986148e8965161466d565b04808452878787510151169061466d565b91868551511690519061421c565b90614680565b80965251019316828451166146b5565b169052388080808080806148a0565b50866149198651828651511690613f22565b1115614850565b816149319293508451511690613f22565b11158738614848565b919450915083813d8111614998575b6149538183613e5b565b810103126149955750670de0b6b3a76400009492848361483093826149868e61497f6148399b996146a0565b94016146a0565b94979950509381959750614805565b80fd5b503d614949565b8451903d90823e3d90fd5b8c87515116620186a090808202918204036149d3576149ce908e8b51511690614680565b6147ba565b634e487b7160e01b82526011600452602482fd5b5060ff60085460381c1615614767565b600090600090600090600090614a0b6144f0565b50614a14614571565b90614a1e826146d0565b8051151580614a2b575050565b935095509250925092606081015192608092838301519360a084015193606084019067ffffffffffffffff80835116928487018281511692602095868601828151169189604097888a019486865116918a519485528c8501528984015260608301527fc63977c8e2362a31182dc8e89a52252f9836922738e0abcfc0de6924972eafe591a18d838251168751918252898201528d87820152898d60608301527f2b5229f33f1d24d5baab718e1e25d0d86195a9b6d786c2c0868edfb21a460e2591a151938285169052519181831690524216838901524363ffffffff1680895285890151861b67ffffffff00000000169160c01b7fffffffffffffffff000000000000000000000000000000000000000000000000169342901b6fffffffffffffffff00000000000000001691171790851b77ffffffffffffffff0000000000000000000000000000000016171760175560c081015191806001600160801b039384815116614bb0906001600160801b03166001600160801b0319601c541617601c55565b015191601c54846001600160801b03198095881b16911617601c5560e0015183815116614bf3906001600160801b03166001600160801b0319601d541617601d55565b0151601d54931b16911617601d5582614c0857565b613da083305b6001600160a01b0316908115614c6d577fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef602082614c50600094600b54613f22565b600b558484526009825260408420818154019055604051908152a3565b60405162461bcd60e51b815260206004820152601f60248201527f45524332303a206d696e7420746f20746865207a65726f2061646472657373006044820152606490fd5b519081151582036105d157565b600090600090614ccd614344565b90604091828101936001600160b81b039384865116421415600014614ef25760046001600160a01b039660608886511684519384809263bd9a548b60e01b82525afa908115614ee557849885938693614e6e575b5090614d8483927fc1f41e029acf5127d111625602160c4cee3e1a4d38e691e50544d1f7c68b77be9695949a859c614e3c575b42168093528460608a01528360808a01528851166001600160a01b03166001600160a01b03196018541617601855565b602087015163ffffffff60a01b1963ffffffff60a01b6018549260a01b1691161760185576ffffffffffffffffffffffffffffffffffffffffffffff19601954161760195581601a5580601b5582519182526020820152a15b60808201614df1815160608501519061421c565b91620186a09280840293840403614e285750614e1663ffffffff926020925190614680565b920151161015614e2257565b60019350565b634e487b7160e01b81526011600452602490fd5b7ffc131c36b7e444dacda44901fd43641dcdcfdc43fe9e2601b3c1dd87061db9e56020838c51168951908152a1614d54565b9950915091506060883d8211614edd575b81614e8c60609383613e5b565b81010312614ed957907fc1f41e029acf5127d111625602160c4cee3e1a4d38e691e50544d1f7c68b77be9291614ec189614cb2565b60208a01519984015190999394509190614d84614d21565b8380fd5b3d9150614e7f565b50505051903d90823e3d90fd5b50606082015160808301519095509350614ddd565b7fdcbc1c05240f31ff3ad067ef1ee35ce4997762752e3a095284754544f4c709d79192936150066001600160a01b03926001600160801b039081614f4e88828451166146b5565b168152614f9582602083019281614f688c828751166146b5565b168452614f77828c1688614c0e565b51166001600160801b03166001600160801b0319601c541617601c55565b5181601c54916001600160801b03199060801b16911617601c55604051906323b872dd60e01b60208301523360248301523060448301528616606482015260648152614fe081613e2b565b7f000000000000000000000000853d955acef822db058eb8505911ed77f175b99e615b22565b604080516001600160801b03958616815295909416602086015216923392819081015b0390a3565b9091926001600160a01b03918286169586330361529b575b5061506b6150526141f6565b6001600160801b039182918280865116915116906144d7565b16908086169182811061527257508061508787828651166144d7565b1683526020926150c682858301926150a28b838651166144d7565b93828516905251166001600160801b03166001600160801b0319601c541617601c55565b81601c54916001600160801b03199060801b16911617601c55861691871561522457876000526009815260409283600020548181106151d55792614fe08360008c7fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef7ffbde797d201c681b91056529119e0b02407c7bb96a4a2c75c01fc9667232c8db9b9a9997876151ac986151d09b85875260098452038a86205580600b5403600b558951908152a3845163a9059cbb60e01b918101919091526001600160a01b0387166024820152604481019290925281606481015b03601f198101835282613e5b565b516001600160801b0395861681529590941660208601521692339281906040820190565b0390a4565b845162461bcd60e51b815260048101849052602260248201527f45524332303a206275726e20616d6f756e7420657863656564732062616c616e604482015261636560f01b6064820152608490fd5b6084906040519062461bcd60e51b82526004820152602160248201527f45524332303a206275726e2066726f6d20746865207a65726f206164647265736044820152607360f81b6064820152fd5b6040516362ddb6d760e11b815260048101919091526001600160801b0387166024820152604490fd5b86600052600a6020526040600020336000526020526040600020549060001982036152c7575b50615046565b6152de6152e6926001600160801b0389169061421c565b9033906140ce565b38806152c1565b90600f54918215615389576001600160a01b036153086141f6565b91169161532360009284845260208052604084205490615e34565b92831561537f578252601f602052604082205492831561537757670de0b6b3a7640000916153509161466d565b0490620186a091828102928184041490151715614e2857509061537291614680565b111590565b505091505090565b5050505050600190565b505050600190565b919061539b6141f6565b6001600160801b0390816153bd816153b16141d0565b511682845116906144d7565b1691808616928381106152725750806153e36153d98585615cff565b97828551166146b5565b1682526154228160208401936153fe828a16838751166146b5565b94828616905251166001600160801b03166001600160801b0319601d541617601d55565b601d54916001600160801b03199060801b16911617601d5533600052602080526040600020615452858254613f22565b90556001600160a01b038216918130840361549e575b50506040519081528360208201527f01348584ec81ac7acd52b7d66d9ade986dd909f3d513881c190fc31c90527efe60403392a3565b60405163a9059cbb60e01b60208201526001600160a01b03909216602483015260448201526154d490614fe0816064810161519e565b3881615468565b909291926001600160a01b0391827f00000000000000000000000045127c7fa0c3ce6a18fec9604d96edfbbc8a97001691604084815197634b82009360e01b8952169485600489015260209360009885816024818d8b5af180156156ae57615676575b50868952601f8552828920615554858254613f22565b905561556284601e54613f22565b601e55169330850361559d575b507fa32435755c235de2976ed44a75a2f85cb01faf0c894f639fe0c32bb9455fea8f939495965051908152a3565b87906155f583516323b872dd60e01b87820152876024820152306044820152856064820152606481526155cf81613e2b565b7f000000000000000000000000f43211935c781d5ca1a41d2041f397b8a7366c7a615b22565b803b156156725781906044845180948193636e553f6560e01b83528860048401523060248401525af18015615668577fa32435755c235de2976ed44a75a2f85cb01faf0c894f639fe0c32bb9455fea8f9596979850615659575b879695945061556f565b61566290613e47565b3861564f565b82513d8a823e3d90fd5b5080fd5b8581813d83116156a7575b61568b8183613e5b565b810103126156a35761569c90614cb2565b503861553e565b8980fd5b503d615681565b84513d8c823e3d90fd5b6001600160a01b03807f00000000000000000000000045127c7fa0c3ce6a18fec9604d96edfbbc8a9700169260409082825196634b82009360e01b885216958660048201526020936000918581602481868c5af180156158575761581f575b50878252601f855283822061572d87825461421c565b905561573b86601e5461421c565b601e55821695308703615778575b505050907fbc290bb45104f73cf92115c9603987c3f8fd30c182a13603d8cffa49b5f5995291519283523392a4565b803b15615672578180916024865180958193630e5a77ed60e21b83528b60048401525af190811561499f57506155cf7fbc290bb45104f73cf92115c9603987c3f8fd30c182a13603d8cffa49b5f59952959493615807938893615810575b50845163a9059cbb60e01b878201526001600160a01b0390911660248201526044810192909252816064810161519e565b90913880615749565b61581990613e47565b386157d6565b8581813d8311615850575b6158348183613e5b565b8101031261584c5761584590614cb2565b5038615717565b8280fd5b503d61582a565b85513d85823e3d90fd5b93907f9dc1449a0ff0c152e18e8289d865b47acc6e1b76b1ecb239c13d6ee22a9206a792916001600160801b03948561589d84828a51166144d7565b16875260208701866158b286828451166144d7565b1681526158ff876001600160a01b03809516998a6000526020805260406000206158df838a16825461421c565b905551166001600160801b03166001600160801b0319601d541617601d55565b5186601d54916001600160801b03199060801b16911617601d551693308503615947575b50604080516001600160801b03928316815292909116602083015281908101615029565b61597c90604051906323b872dd60e01b60208301528660248301523060448301528316606482015260648152614fe081613e2b565b38615923565b7f7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff81116159ac5790565b60405162461bcd60e51b815260206004820152602860248201527f53616665436173743a2076616c756520646f65736e27742066697420696e206160448201527f6e20696e743235360000000000000000000000000000000000000000000000006064820152608490fd5b8051156113e35760200190565b80518210156113e35760209160051b010190565b60209081818403126105d15780519067ffffffffffffffff82116105d157019180601f840112156105d1578251615a6e81613e7d565b93615a7c6040519586613e5b565b818552838086019260051b8201019283116105d1578301905b828210615aa3575050505090565b81518152908301908301615a95565b91908110156113e35760051b0190565b356001600160a01b03811681036105d15790565b6001600160a01b03600154163303615aea57565b604051630e05f48560e11b8152600490fd5b6001600160a01b03600054163303615b1057565b604051633d71279960e21b8152600490fd5b6001600160a01b031690604051615b3881613df9565b6020928382527f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564848301526000808486829651910182855af13d15615c64573d9167ffffffffffffffff8311615c505790615bb393929160405192615ba688601f19601f8401160185613e5b565b83523d868885013e615c6e565b90815180615bc2575b50505050565b82849181010312614995575081615bd99101614cb2565b15615be657808080615bbc565b6084906040519062461bcd60e51b82526004820152602a60248201527f5361666545524332303a204552433230206f7065726174696f6e20646964206e60448201527f6f742073756363656564000000000000000000000000000000000000000000006064820152fd5b634e487b7160e01b85526041600452602485fd5b90615bb392916060915b91929015615cd05750815115615c82575090565b3b15615c8b5790565b60405162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e74726163740000006044820152606490fd5b825190915015615ce35750805190602001fd5b60405162461bcd60e51b8152908190610a179060048301613c20565b91906001600160801b038084511615600014615d1a57509150565b615d53906020850190615d4a81615d41615d37828651168861466d565b828a511690614680565b9751168761466d565b91511690614680565b10615d5a57565b90600181018091116113985790565b6001600160801b038082511615600014615d8257505090565b615d4a61466a93826020850151169061466d565b90916001600160801b038083511615600014615db25750505090565b602083959492930190615dd5615dcb828451168561466d565b8288511690614680565b9584615de6575b50505050615d5a57565b615df993945081615d4a9151168761466d565b1038808080615ddc565b60208101906001600160801b03908183511615600014615e235750505090565b61466a9382615d4a9251169061466d565b919060208301926001600160801b038085511615600014615e56575090925050565b9081615d4a81615d41615d37615d53968651168861466d565b909160208201916001600160801b038084511615600014615e91575050505090565b615dd5615dcb8284989795969851168561466d56fea164736f6c6343000813000a
Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)
000000000000000000000000000000000000000000000000000000000000006000000000000000000000000000000000000000000000000000000000000001c000000000000000000000000000000000000000000000000000000000000002400000000000000000000000000000000000000000000000000000000000000140000000000000000000000000853d955acef822db058eb8505911ed77f175b99e000000000000000000000000f43211935c781d5ca1a41d2041f397b8a7366c7a000000000000000000000000013723e5631c591af50e89c2892b464530103481000000000000000000000000000000000000000000000000000000000000138800000000000000000000000018500cb1f2fe7a40ebda393383a0b8548a31f2610000000000000000000000000000000000000000000000000000000235ef7f6800000000000000000000000000000000000000000000000000000000000124f8000000000000000000000000000000000000000000000000000000000000271000000000000000000000000000000000000000000000000000000000000007d000000000000000000000000045127c7fa0c3ce6a18fec9604d96edfbbc8a97000000000000000000000000000000000000000000000000000000000000000060000000000000000000000000fd3065c629ee890fd74f43b802c2fea4b7279b8c000000000000000000000000168200cf227d4543302686124ac28ae0eaf2ca0b0000000000000000000000008412ebf45bac1b340bbe8f318b928c466c4e39ca0000000000000000000000000000000000000000000000000000000000000100000000000000000000000000000000000000000000000000000000000000006000000000000000000000000000000000000000000000000000000000000000c000000000000000000000000000000000000000000000000000000000000000120000000000000000000000000000000000000000000000000000000000000039467261786c656e6420496e7465726573742042656172696e672046524158202843757276652e6669204554482f66727845544829202d203231000000000000000000000000000000000000000000000000000000000000000000000000000013664652415828667278455448435256292d323100000000000000000000000000
-----Decoded View---------------
Arg [0] : _configData (bytes): 0x000000000000000000000000853d955acef822db058eb8505911ed77f175b99e000000000000000000000000f43211935c781d5ca1a41d2041f397b8a7366c7a000000000000000000000000013723e5631c591af50e89c2892b464530103481000000000000000000000000000000000000000000000000000000000000138800000000000000000000000018500cb1f2fe7a40ebda393383a0b8548a31f2610000000000000000000000000000000000000000000000000000000235ef7f6800000000000000000000000000000000000000000000000000000000000124f8000000000000000000000000000000000000000000000000000000000000271000000000000000000000000000000000000000000000000000000000000007d000000000000000000000000045127c7fa0c3ce6a18fec9604d96edfbbc8a9700
Arg [1] : _immutables (bytes): 0x000000000000000000000000fd3065c629ee890fd74f43b802c2fea4b7279b8c000000000000000000000000168200cf227d4543302686124ac28ae0eaf2ca0b0000000000000000000000008412ebf45bac1b340bbe8f318b928c466c4e39ca
Arg [2] : _customConfigData (bytes): 0x000000000000000000000000000000000000000000000000000000000000006000000000000000000000000000000000000000000000000000000000000000c000000000000000000000000000000000000000000000000000000000000000120000000000000000000000000000000000000000000000000000000000000039467261786c656e6420496e7465726573742042656172696e672046524158202843757276652e6669204554482f66727845544829202d203231000000000000000000000000000000000000000000000000000000000000000000000000000013664652415828667278455448435256292d323100000000000000000000000000
-----Encoded View---------------
27 Constructor Arguments found :
Arg [0] : 0000000000000000000000000000000000000000000000000000000000000060
Arg [1] : 00000000000000000000000000000000000000000000000000000000000001c0
Arg [2] : 0000000000000000000000000000000000000000000000000000000000000240
Arg [3] : 0000000000000000000000000000000000000000000000000000000000000140
Arg [4] : 000000000000000000000000853d955acef822db058eb8505911ed77f175b99e
Arg [5] : 000000000000000000000000f43211935c781d5ca1a41d2041f397b8a7366c7a
Arg [6] : 000000000000000000000000013723e5631c591af50e89c2892b464530103481
Arg [7] : 0000000000000000000000000000000000000000000000000000000000001388
Arg [8] : 00000000000000000000000018500cb1f2fe7a40ebda393383a0b8548a31f261
Arg [9] : 0000000000000000000000000000000000000000000000000000000235ef7f68
Arg [10] : 00000000000000000000000000000000000000000000000000000000000124f8
Arg [11] : 0000000000000000000000000000000000000000000000000000000000002710
Arg [12] : 00000000000000000000000000000000000000000000000000000000000007d0
Arg [13] : 00000000000000000000000045127c7fa0c3ce6a18fec9604d96edfbbc8a9700
Arg [14] : 0000000000000000000000000000000000000000000000000000000000000060
Arg [15] : 000000000000000000000000fd3065c629ee890fd74f43b802c2fea4b7279b8c
Arg [16] : 000000000000000000000000168200cf227d4543302686124ac28ae0eaf2ca0b
Arg [17] : 0000000000000000000000008412ebf45bac1b340bbe8f318b928c466c4e39ca
Arg [18] : 0000000000000000000000000000000000000000000000000000000000000100
Arg [19] : 0000000000000000000000000000000000000000000000000000000000000060
Arg [20] : 00000000000000000000000000000000000000000000000000000000000000c0
Arg [21] : 0000000000000000000000000000000000000000000000000000000000000012
Arg [22] : 0000000000000000000000000000000000000000000000000000000000000039
Arg [23] : 467261786c656e6420496e7465726573742042656172696e6720465241582028
Arg [24] : 43757276652e6669204554482f66727845544829202d20323100000000000000
Arg [25] : 0000000000000000000000000000000000000000000000000000000000000013
Arg [26] : 664652415828667278455448435256292d323100000000000000000000000000
Loading...
Loading
Loading...
Loading
Net Worth in USD
$99.14
Net Worth in ETH
0.051005
Token Allocations
FRAX
100.00%
Multichain Portfolio | 34 Chains
| Chain | Token | Portfolio % | Price | Amount | Value |
|---|---|---|---|---|---|
| ETH | 100.00% | $0.991378 | 100.0014 | $99.14 |
Loading...
Loading
Loading...
Loading
Loading...
Loading
[ Download: CSV Export ]
[ Download: CSV Export ]
A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.