Overview
ETH Balance
0 ETH
Eth Value
$0.00More Info
Private Name Tags
ContractCreator
TokenTracker
Latest 7 from a total of 7 transactions
Transaction Hash |
Method
|
Block
|
From
|
To
|
|||||
---|---|---|---|---|---|---|---|---|---|
Redeem | 16524475 | 707 days ago | IN | 0 ETH | 0.00129523 | ||||
Deposit | 16524462 | 707 days ago | IN | 0 ETH | 0.00316563 | ||||
Redeem | 16524366 | 707 days ago | IN | 0 ETH | 0.00110876 | ||||
Remove Collatera... | 16524099 | 707 days ago | IN | 0 ETH | 0.00129417 | ||||
Repay Asset | 16524096 | 707 days ago | IN | 0 ETH | 0.00133534 | ||||
Borrow Asset | 16507635 | 709 days ago | IN | 0 ETH | 0.00471692 | ||||
Deposit | 16487353 | 712 days ago | IN | 0 ETH | 0.00430226 |
Latest 1 internal transaction
Advanced mode:
Parent Transaction Hash | Block |
From
|
To
|
|||
---|---|---|---|---|---|---|
16485554 | 712 days ago | Contract Creation | 0 ETH |
Loading...
Loading
Contract Name:
FraxlendPair
Compiler Version
v0.8.17+commit.8df45f5f
Optimization Enabled:
Yes with 3000 runs
Other Settings:
default evmVersion
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: ISC pragma solidity ^0.8.17; // ==================================================================== // | ______ _______ | // | / _____________ __ __ / ____(_____ ____ _____ ________ | // | / /_ / ___/ __ `| |/_/ / /_ / / __ \/ __ `/ __ \/ ___/ _ \ | // | / __/ / / / /_/ _> < / __/ / / / / / /_/ / / / / /__/ __/ | // | /_/ /_/ \__,_/_/|_| /_/ /_/_/ /_/\__,_/_/ /_/\___/\___/ | // | | // ==================================================================== // ========================== 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 "@openzeppelin/contracts/token/ERC20/IERC20.sol"; import "@openzeppelin/contracts/token/ERC20/extensions/IERC20Metadata.sol"; import "@openzeppelin/contracts/security/ReentrancyGuard.sol"; import "@chainlink/contracts/src/v0.8/interfaces/AggregatorV3Interface.sol"; import "./FraxlendPairConstants.sol"; import "./FraxlendPairCore.sol"; import "./libraries/VaultAccount.sol"; import "./libraries/SafeERC20.sol"; import "./interfaces/IERC4626.sol"; import "./interfaces/IFraxlendWhitelist.sol"; import "./interfaces/IRateCalculator.sol"; import "./interfaces/ISwapper.sol"; contract FraxlendPair is IERC20Metadata, FraxlendPairCore { using VaultAccountingLibrary for VaultAccount; using SafeERC20 for IERC20; using SafeCast for uint256; 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 ? depositLimit : depositLimit - _totalAsset.amount; } function maxMint(address _receiver) external view returns (uint256 _maxShares) { (, , , , VaultAccount memory _totalAsset, ) = previewAddInterest(); uint256 _maxDeposit = _totalAsset.amount < depositLimit ? depositLimit : 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 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 ```SetTimelock``` event fires when the timelockAddress is set /// @param _oldAddress The original address /// @param _newAddress The new address event SetTimelock(address _oldAddress, address _newAddress); /// @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) { // 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(true); if (!isWithdrawAccessControlRevoked) _pauseWithdraw(true); if (!isLiquidateAccessControlRevoked) _pauseLiquidate(true); if (!isInterestAccessControlRevoked) { _addInterest(); _pauseInterest(true); } } /// @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 { _requireTimelock(); 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 { _requireTimelock(); 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 pragma solidity ^0.8.0; interface AggregatorV3Interface { function decimals() external view returns (uint8); function description() external view returns (string memory); function version() external view returns (uint256); function getRoundData(uint80 _roundId) external view returns ( uint80 roundId, int256 answer, uint256 startedAt, uint256 updatedAt, uint80 answeredInRound ); function latestRoundData() external view returns ( uint80 roundId, int256 answer, uint256 startedAt, uint256 updatedAt, uint80 answeredInRound ); }
// 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 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 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 (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 (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 (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.17; // ==================================================================== // | ______ _______ | // | / _____________ __ __ / ____(_____ ____ _____ ________ | // | / /_ / ___/ __ `| |/_/ / /_ / / __ \/ __ `/ __ \/ ___/ _ \ | // | / __/ / / / /_/ _> < / __/ / / / / / /_/ / / / / /__/ __/ | // | /_/ /_/ \__,_/_/|_| /_/ /_/_/ /_/\__,_/_/ /_/\___/\___/ | // | | // ==================================================================== // ========================= 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 "@openzeppelin/contracts/access/Ownable2Step.sol"; import "./Timelock2Step.sol"; import "./FraxlendPairAccessControlErrors.sol"; /// @title FraxlendPairAccessControl /// @author Drake Evans (Frax Finance) https://github.com/drakeevans /// @notice An abstract contract which contains the access control logic abstract contract FraxlendPairAccessControl is Timelock2Step, Ownable2Step, FraxlendPairAccessControlErrors { // Deployer address public immutable DEPLOYER_ADDRESS; // Admin contracts address public immutable CIRCUIT_BREAKER_ADDRESS; // 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; constructor(bytes memory _immutables) Timelock2Step() Ownable() { // 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; CIRCUIT_BREAKER_ADDRESS = _circuitBreakerAddress; } // ============================================================================================ // Functions: Access Control // ============================================================================================ function _requireProtocolOrOwner() internal view { if ( msg.sender != CIRCUIT_BREAKER_ADDRESS && 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); } }
// SPDX-License-Identifier: ISC pragma solidity ^0.8.17; // ==================================================================== // | ______ _______ | // | / _____________ __ __ / ____(_____ ____ _____ ________ | // | / /_ / ___/ __ `| |/_/ / /_ / / __ \/ __ `/ __ \/ ___/ _ \ | // | / __/ / / / /_/ _> < / __/ / / / / / /_/ / / / / /__/ __/ | // | /_/ /_/ \__,_/_/|_| /_/ /_/_/ /_/\__,_/_/ /_/\___/\___/ | // | | // ==================================================================== // ===================== 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 // ==================================================================== 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.17; // ==================================================================== // | ______ _______ | // | / _____________ __ __ / ____(_____ ____ _____ ________ | // | / /_ / ___/ __ `| |/_/ / /_ / / __ \/ __ `/ __ \/ ___/ _ \ | // | / __/ / / / /_/ _> < / __/ / / / / / /_/ / / / / /__/ __/ | // | /_/ /_/ \__,_/_/|_| /_/ /_/_/ /_/\__,_/_/ /_/\___/\___/ | // | | // ==================================================================== // ===================== 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 // ==================================================================== 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(); }
// SPDX-License-Identifier: ISC pragma solidity ^0.8.17; // ==================================================================== // | ______ _______ | // | / _____________ __ __ / ____(_____ ____ _____ ________ | // | / /_ / ___/ __ `| |/_/ / /_ / / __ \/ __ `/ __ \/ ___/ _ \ | // | / __/ / / / /_/ _> < / __/ / / / / / /_/ / / / / /__/ __/ | // | /_/ /_/ \__,_/_/|_| /_/ /_/_/ /_/\__,_/_/ /_/\___/\___/ | // | | // ==================================================================== // ========================= 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 "@openzeppelin/contracts/token/ERC20/ERC20.sol"; import "@openzeppelin/contracts/token/ERC20/extensions/IERC20Metadata.sol"; import "@openzeppelin/contracts/token/ERC20/IERC20.sol"; import "@openzeppelin/contracts/security/ReentrancyGuard.sol"; import "@openzeppelin/contracts/access/Ownable.sol"; import "@openzeppelin/contracts/utils/math/SafeCast.sol"; import "./FraxlendPairAccessControl.sol"; import "./FraxlendPairConstants.sol"; import "./libraries/VaultAccount.sol"; import "./libraries/SafeERC20.sol"; import "./interfaces/IDualOracle.sol"; import "./interfaces/IRateCalculatorV2.sol"; import "./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() & initialize() // ============================================================================================ // Asset and collateral contracts IERC20 internal immutable assetContract; IERC20 public immutable collateralContract; // LTV Settings uint256 public maxLTV; // Liquidation Fee /// @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 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 smaller 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() // ============================================================================================ // Initialize // ============================================================================================ /// @notice The ```constructor``` function is called on deployment /// @param _configData abi.encode(address _asset, address _collateral, address _oracleMultiply, address _oracleDivide, uint256 _oracleNormalization, address _rateContract, uint64 _fullUtilizationRate) /// @param _customConfigData abi.encode(string memory _nameOfContract, string memory _symbolOfContract, uint8 _decimalsOfContract, uint256 _maxLTV, uint256 _liquidationFee) 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 ) = abi.decode( _configData, (address, address, address, uint32, address, uint64, uint256, uint256, uint256) ); // Pair Settings assetContract = IERC20(_asset); collateralContract = IERC20(_collateral); 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 * 90000) / 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 _deltaTime The time elapsed since last interest accrual /// @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 _deltaTime, uint256 _feesAmount, uint256 _feesShare ); /// @notice The ```UpdateRate``` event is emitted when the interest rate is updated /// @param _ratePerSec The old interest rate (per second) /// @param _deltaTime The time elapsed since last update /// @param _utilizationRate The utilization of assets in the Pair /// @param _newRatePerSec The new interest rate (per second) event UpdateRate(uint256 _ratePerSec, uint256 _deltaTime, uint256 _utilizationRate, uint256 _newRatePerSec); /// @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 is paused 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, using original interest rate not new rate (new rate is for the time period moving forward) _results.interestEarned = (_deltaTime * _results.totalBorrow.amount * _currentRateInfo.ratePerSec) / RATE_PRECISION; // Accrue interest (if any) and fees if 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; // 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) { // 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) { // 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) { // 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) { // 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) { // 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); } 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 { // Effects: write to state userCollateralBalance[_borrower] += _collateralAmount; totalCollateral += _collateralAmount; // Interactions if (_sender != address(this)) { collateralContract.safeTransferFrom(_sender, address(this), _collateralAmount); } 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 { _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 { // 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)) { 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) { _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) { // 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) { // 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; { uint128 _amountToAdjust; 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.17; interface IDualOracle { function getPrices() external view returns (bool _isBadData, uint256 _priceLow, uint256 _priceHigh); function decimals() external view returns (uint8); function oracleType() external view returns (uint256); function name() external view returns (string memory); }
// SPDX-License-Identifier: ISC pragma solidity >=0.8.17; import "@openzeppelin/contracts/token/ERC20/IERC20.sol"; import "@openzeppelin/contracts/token/ERC20/extensions/IERC20Metadata.sol"; interface IERC4626 is IERC20, IERC20Metadata { event Deposit(address indexed caller, address indexed owner, uint256 assets, uint256 shares); event Withdraw( address indexed caller, address indexed receiver, address indexed owner, uint256 assets, uint256 shares ); function asset() external view returns (address); function convertToAssets(uint256 shares) external view returns (uint256); function convertToShares(uint256 assets) external view returns (uint256); function maxDeposit(address) external view returns (uint256); function maxMint(address) external view returns (uint256); function maxRedeem(address owner) external view returns (uint256); function maxWithdraw(address owner) external view returns (uint256); function previewDeposit(uint256 assets) external view returns (uint256); function previewMint(uint256 shares) external view returns (uint256); function previewRedeem(uint256 shares) external view returns (uint256); function previewWithdraw(uint256 assets) external view returns (uint256); function totalAssets() external view returns (uint256); function mint(uint256 shares, address receiver) external returns (uint256 assets); function deposit(uint256 assets, address receiver) external returns (uint256 shares); function redeem( uint256 shares, address receiver, address owner ) external returns (uint256 assets); function withdraw( uint256 assets, address receiver, address owner ) external returns (uint256 shares); }
// SPDX-License-Identifier: ISC pragma solidity >=0.8.17; interface IFraxlendWhitelist { function fraxlendDeployerWhitelist(address) external view returns (bool); function oracleContractWhitelist(address) external view returns (bool); function owner() external view returns (address); function rateContractWhitelist(address) external view returns (bool); function renounceOwnership() external; function setFraxlendDeployerWhitelist(address[] calldata _addresses, bool _bool) external; function setOracleContractWhitelist(address[] calldata _addresses, bool _bool) external; function setRateContractWhitelist(address[] calldata _addresses, bool _bool) external; function transferOwnership(address newOwner) external; }
// SPDX-License-Identifier: ISC pragma solidity >=0.8.17; interface IRateCalculator { function name() external pure returns (string memory); function requireValidInitData(bytes calldata _initData) external pure; function getConstants() external pure returns (bytes memory _calldata); function getNewRate(bytes calldata _data, bytes calldata _initData) external pure returns (uint64 _newRatePerSec); }
// SPDX-License-Identifier: ISC pragma solidity ^0.8.17; 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.17; 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.17; import "@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.17; 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; } } } }
// SPDX-License-Identifier: ISC pragma solidity ^0.8.17; // ==================================================================== // | ______ _______ | // | / _____________ __ __ / ____(_____ ____ _____ ________ | // | / /_ / ___/ __ `| |/_/ / /_ / / __ \/ __ `/ __ \/ ___/ _ \ | // | / __/ / / / /_/ _> < / __/ / / / / / /_/ / / / / /__/ __/ | // | /_/ /_/ \__,_/_/|_| /_/ /_/_/ /_/\__,_/_/ /_/\___/\___/ | // | | // ==================================================================== // ========================== Timelock2Step =========================== // ==================================================================== // Frax Finance: https://github.com/FraxFinance // Primary Author // Drake Evans: https://github.com/DrakeEvans // ==================================================================== 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)); } }
{ "metadata": { "bytecodeHash": "none" }, "viaIR": true, "optimizer": { "enabled": true, "runs": 3000 }, "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "devdoc", "userdoc", "metadata", "abi" ] } }, "libraries": {} }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"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":"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":"_deltaTime","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":"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":false,"internalType":"address","name":"_oldAddress","type":"address"},{"indexed":false,"internalType":"address","name":"_newAddress","type":"address"}],"name":"SetTimelock","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":"_ratePerSec","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_deltaTime","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_utilizationRate","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_newRatePerSec","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":"CIRCUIT_BREAKER_ADDRESS","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"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":"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":"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
6101208060405234620008e457620073e680380380916200002182856200125e565b83398101606082820312620008e45781516001600160401b038111620008e457816200004f91840162001282565b60208301519092906001600160401b038111620008e457826200007491830162001282565b60408201519092906001600160401b038111620008e45762000097920162001282565b60405192620000a68462001242565b6000845260405192620000b98462001242565b60008452600180546001600160a01b03191633908117909155620000dd9062001366565b600019600455600019600655606081805181010312620008e457620001836200010960208301620012f9565b916200011860408201620012f9565b906001600160a01b03906200013090606001620012f9565b600154911690816001600160a01b0382167f31b6c5a04b069b6ec1b3cef44c4e7c1eadd721349cda9823d0b1877b3551cdc6600080a36001600160a01b031916176001556001600160a01b031662001366565b336080526001600160a01b031660a05283516001600160401b03811162000e0357600b54600181811c91168015620011ff575b602082101462000f0457601f811162001195575b50602094601f821160011462001128579481929394956000926200111c575b50508160011b916000199060031b1c191617600b555b82516001600160401b03811162000e0357600c54600181811c9116801562001111575b602082101462000f0457601f8111620010a7575b506020601f82116001146200101d578192939460009262001011575b50508160011b916000199060031b1c191617600c555b6001600d5561012081805181010312620008e4576200028a60208201620012f9565b6200029860408301620012f9565b91620002a760608201620012f9565b60808201519063ffffffff82168203620008e457620002c960a08401620012f9565b91620002d860c085016200130e565b60e085810151610100870151610120909701516001600160a01b0398891660c0529790981690526016544260001981011162000839574360001981011162000839574260001990810160401b6fffffffffffffffff000000000000000016600160801b600160c01b0390921660c09390931b6001600160c01b031990811693909317919091174390910163ffffffff1617601655601780549091166001600160a01b039384161760a09290921b63ffffffff60a01b16919091179055601280546001600160a01b03191692909116919091179055600f819055801562015f90808302839004141715620008395762015f90620186a0910204601055601155600e558051810190606081830312620008e4576020810151916001600160401b038311620008e4576020928362000414920193849184010162001282565b60408201519092906001600160401b038111620008e45760609160206200043e9285010162001282565b9101519060ff82168203620008e45782516001600160401b03811162000e0357601454600181811c9116801562001006575b602082101462000f0457601f811162000fa0575b506020601f821160011462000f31578192939460009262000f25575b50508160011b916000199060031b1c1916176014555b8051906001600160401b03821162000e035760155490600182811c9216801562000f1a575b602083101462000f045781601f84931162000ea3575b50602090601f831160011462000e255760009262000e19575b50508160011b916000199060031b1c1916176015555b610100526000608060405162000536816200120a565b82815282602082015282604082015282606082015201526040516200055b816200120a565b60165463ffffffff8116825263ffffffff8160201c16602083015260018060401b038160401c16604083015260018060401b038160801c16606083015260c01c60808201526040518061010081011060018060401b036101008301111762000e035761010081016040526000815260006020820152600060408201526000606082015260006080820152600060a0820152604051620005fa8162001226565b600081526000602082015260c0820152604051620006188162001226565b6000808252602082015260e082015260408201516001600160401b03164214158062000df2575b62000a8a575b805162000906575b50506040516200065d816200120a565b6017546001600160a01b038116825260a081901c63ffffffff1660208301526018546001600160b81b038116604084018190526019546060850152601a5460808501524214620008fe5760405163bd9a548b60e01b8152906060826004816001600160a01b0387165afa908115620008f2576000809360009362000888575b50936040939291600080516020620073c6833981519152956200084f575b60018060b81b03421691828689015284606089015283608089015260018060a01b0388511663ffffffff60a01b60208a015160a01b169160018060c01b031916171760175560018060b81b031916176018558160195580601a5582519182526020820152a15b62000775608082015160608301519062001323565b80620186a00290620186a082040362000839576080620007989201519062001345565b50604051615fc39081620013e38239608051818181610dfd01526144e9015260a0518181816118020152614470015260c05181818161102d015281816110df015281816111c5015281816113e7015281816119f30152818161354d01526151cb015260e051818181610fce015281816114d301528181611a4801528181611b0901528181611be501528181611df90152615827015261010051816135b10152f35b634e487b7160e01b600052601160045260246000fd5b7ffc131c36b7e444dacda44901fd43641dcdcfdc43fe9e2601b3c1dd87061db9e5602060018060a01b038951168751908152a1620006fa565b92505091506060813d606011620008e9575b81620008a9606093836200125e565b81010312620008e4578051908115158203620008e457602081015160409091015190929091600080516020620073c6833981519152620006dc565b600080fd5b3d91506200089a565b6040513d6000823e3d90fd5b505062000760565b60a08101516020808301516001600160401b0380821660608701526040808601518083166080808a019190915242938416838a01524363ffffffff16808a529886015194811b600160801b600160c01b03169390921b6fffffffffffffffff00000000000000001693851b67ffffffff0000000016909717929092171760c095861b6001600160c01b0319161760165593830151805190820151851b6001600160801b03199081166001600160801b0392831617601b5560e090940151805192015190941b9092169190921617601c5580620009e4575b806200064d565b301562000a4557620009f981600a54620013ba565b600a55306000526008602052604060002081815401905560405190815260007fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60203093a338620009dd565b60405162461bcd60e51b815260206004820152601f60248201527f45524332303a206d696e7420746f20746865207a65726f2061646472657373006044820152606490fd5b6001815260405162000a9c8162001226565b601b546001600160801b038116825260801c602082015260c082015260405162000ac68162001226565b601c546001600160801b038116825260801c602082015260e0820152604082015162000afc906001600160401b03164262001323565b60c0820151516001600160801b031662000dab576000905b60125460808501516040805163cd3181d560e01b81526004810185905260248101959095526001600160401b03909116604485015290839060649082906001600160a01b03165afa8015620008f25760009260009162000d43575b506001600160401b039081166040850152909116602083015260e082015151670de0b6b3a76400009162000bca9162000bb5916001600160801b03919091169062001331565b60608501516001600160401b03169062001331565b04806060830152801515908162000d1b575b508062000cf0575b156200064557606081015160e0820151805190916001600160801b039162000c11918316908316620013c8565b169052606081015160c0820151805190916001600160801b039162000c3b918316908316620013c8565b16905263ffffffff6020830151168062000c57575b5062000645565b62000c94620186a062000c7362000cba93606086015162001331565b046080840181905260c0840151602001516001600160801b03169062001331565b60c083015151608084015162000cb3916001600160801b031662001323565b9062001345565b60a0820181905260c0820151602001805190916001600160801b039162000ce6918316908316620013c8565b1690523862000c50565b50606081015160c0820151516001600160801b039162000d1391831690620013ba565b111562000be4565b60e0830151516001600160801b03925062000d3a9190831690620013ba565b11153862000bdc565b9250506040823d60401162000da2575b8162000d62604093836200125e565b81010312620008e45762000bb562000bca918362000d97602062000d8f670de0b6b3a7640000976200130e565b92016200130e565b925093509162000b6f565b3d915062000d53565b60e0820151516001600160801b0316620186a0808202048103620008395760c08301515162000deb916001600160801b0390911690620186a00262001345565b9062000b14565b5060ff60075460381c16156200063f565b634e487b7160e01b600052604160045260246000fd5b0151905038806200050a565b601560009081529350600080516020620073a683398151915291905b601f198416851062000e87576001945083601f1981161062000e6d575b505050811b0160155562000520565b015160001960f88460031b161c1916905538808062000e5e565b8181015183556020948501946001909301929091019062000e41565b6015600052909150600080516020620073a6833981519152601f840160051c81016020851062000efc575b90849392915b601f830160051c8201811062000eec575050620004f1565b6000815585945060010162000ed4565b508062000ece565b634e487b7160e01b600052602260045260246000fd5b91607f1691620004db565b015190503880620004a0565b601460005260206000209060005b601f198416811062000f875750600193949583601f1981161062000f6d575b505050811b01601455620004b6565b015160001960f88460031b161c1916905538808062000f5e565b9091602060018192858a01518155019301910162000f3f565b60146000527fce6d7b5282bd9a3661ae061feed1dbda4e52ab073b1f9285be6e155d9c38d4ec601f830160051c81016020841062000ffe575b601f830160051c8201811062000ff157505062000484565b6000815560010162000fd9565b508062000fd9565b90607f169062000470565b01519050388062000252565b600c60009081527fdf6966c971051c3d54ec59162606531493a51404a002842f56009d7e5cf4a8c79190601f198416905b8181106200108e5750958360019596971062001074575b505050811b01600c5562000268565b015160001960f88460031b161c1916905538808062001065565b9192602060018192868b0151815501940192016200104e565b600c6000527fdf6966c971051c3d54ec59162606531493a51404a002842f56009d7e5cf4a8c7601f830160051c8101916020841062001106575b601f0160051c01905b818110620010f9575062000236565b60008155600101620010ea565b9091508190620010e1565b90607f169062000222565b015190503880620001e9565b601f19821695600b60005260206000209160005b8881106200117c5750836001959697981062001162575b505050811b01600b55620001ff565b015160001960f88460031b161c1916905538808062001153565b919260206001819286850151815501940192016200113c565b600b6000527f0175b7a638427703f0dbe7bb9bbf987a2551717b34e79f33b5b1008d1fa01db9601f830160051c81019160208410620011f4575b601f0160051c01905b818110620011e75750620001ca565b60008155600101620011d8565b9091508190620011cf565b90607f1690620001b6565b60a081019081106001600160401b0382111762000e0357604052565b604081019081106001600160401b0382111762000e0357604052565b602081019081106001600160401b0382111762000e0357604052565b601f909101601f19168101906001600160401b0382119082101762000e0357604052565b919080601f84011215620008e4578251906001600160401b03821162000e035760405191602091620012be601f8301601f19168401856200125e565b818452828287010111620008e45760005b818110620012e557508260009394955001015290565b8581018301518482018401528201620012cf565b51906001600160a01b0382168203620008e457565b51906001600160401b0382168203620008e457565b919082039182116200083957565b818102929181159184041417156200083957565b811562001350570490565b634e487b7160e01b600052601260045260246000fd5b600380546001600160a01b0319908116909155600280549182166001600160a01b0393841690811790915591167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0600080a3565b919082018092116200083957565b6001600160801b039182169082160191908211620008395756fe60806040526004361015610013575b600080fd5b60003560e01c8062348d1814613bb757806301e1d11414613b8557806302ce728f14613b485780630475260e14613ad957806306fdde0314613a1b57806307a2d13a146139ee5780630880b2f01461399b57806308a0c3751461393b578063090f3f5014613914578063095ea7b3146138ee5780630a28a477146138c15780630a67918c1461389e5780630b4aecf01461387c5780630c70661d14613856578063115a334c1461383057806311a2e4bc1461381257806318160ddd146137f15780631bc23cf9146137675780631c2591d31461372f5780631c6c95971461369357806323b872dd146135d55780632b3ba68114611579578063313ce5671461359757806334680fe51461357157806338d52e0f1461352d57806339030864146134de578063395093511461348c5780633d417d2d146133fe5780633f2617cb146133795780633f4ba83a146132aa578063402d267d1461325857806345014095146131e157806345757e4a146131bb5780634732428c1461319d57806349eb7af6146109285780634ac8eb5f1461317f5780634b4b418e146130f95780634bc66f32146130d25780634c18a4fb146130445780634c41799514612ff35780634cdad506146117d95780634f8b4ae714612f555780634fd422df14612f1b57806354fd4d5014612ef157806364e51d5d14612e7e57806367800b5f14612e5857806369026e8814612e035780636b96668f14612d6d5780636cd3cc77146109285780636e553f6514612cf457806370a0823114612cba578063715018a614612c49578063721b0a47146128da57806379ba5097146127f95780637d37bdd7146127c15780637d63fbc21461279c5780637ec4b5711461275f5780637ecefa6e146126ee5780638142dd53146125e75780638285ef40146125b55780638456cb59146124a6578063858f1e6814612408578063886c033a146123af5780638cad7fbe146123705780638da5cb5b146123495780638f791f8b146121b757806393f46f641461217a57806394bf804d146120d657806395d14ca81461208057806395d89b4114611f7f57806399530b0614611f055780639a295e7314611eb15780639bc6ab8614611e8c5780639fe34bdc14610928578063a053db6814611916578063a457c2d714611857578063a9059cbb14611826578063ad0c3bb5146117e2578063b3d7f6b9146117d9578063b460af941461177c578063b5af306214611742578063b68d0a09146116db578063b78294dd14610928578063b7db54f5146116b6578063b8ca3b8314611699578063ba087652146115fe578063bbb09624146115d9578063bdc8144b14611582578063c0a7e89214611579578063c58e4df614611553578063c63d75b6146114f7578063c6e1c7c9146114b3578063c6e6f59214610730578063ca2298fe14610f15578063cacf3b5814610ee6578063cadac47914610ea5578063cdd72d5214610e44578063ce96cb7714610e21578063d2a156e014610ddd578063d41ddc9614610d28578063d905777e14610d05578063daf33f2a14610c10578063dd62ed3e14610bbe578063e1e9277514610b71578063e203641714610b4e578063e30c397814610b27578063e4b0007014610ad6578063e551d11d14610ab8578063e5f13b1614610931578063e63a391f14610928578063e7a33174146108d1578063e8596f721461087c578063eafecffa1461085e578063ebd462cb146107aa578063ecf708581461078c578063eee2421914610765578063ef8b30f714610730578063f211c3901461070a578063f2fde38b14610693578063f384bd0514610675578063f6ccaad414610602578063f9557ccb146105cc5763fbbbf94c1461056557600080fd5b3461000e57600060031936011261000e5760a060175476ffffffffffffffffffffffffffffffffffffffffffffff60185416601954601a549163ffffffff604051946001600160a01b0381168652861c166020850152604084015260608301526080820152f35b3461000e57600060031936011261000e57601b54604080516001600160801b038316815260809290921c602083015290f35b0390f35b3461000e57600060031936011261000e5761061b615bf5565b73ffffffffffffffffffffffffffffffffffffffff19806000541660005560015490336001600160a01b0383167f31b6c5a04b069b6ec1b3cef44c4e7c1eadd721349cda9823d0b1877b3551cdc6600080a3163317600155005b3461000e57600060031936011261000e576020600e54604051908152f35b3461000e57602060031936011261000e576106ac613c58565b6106b4613ec7565b6001600160a01b03809116908173ffffffffffffffffffffffffffffffffffffffff196003541617600355600254167f38d16b8cac22d99fc7c124b9cd0de2d3fa1faef420bfe791d8c362d765e22700600080a3005b3461000e57600060031936011261000e57602060ff60075460381c166040519015158152f35b3461000e57602060031936011261000e57602061075d61074e614767565b50935050505060043590615e79565b604051908152f35b3461000e57600060031936011261000e5760206001600160a01b0360125416604051908152f35b3461000e57600060031936011261000e576020600654604051908152f35b3461000e57602060031936011261000e576107c3613cae565b8015610851576107d1614465565b6007549060ff8260201c16610827577fc56dd3e14f5af3a74c61b7cdf855a3d8ab4401c78c0622a4d312de8a8f8736a29160209115159063ff0000001963ff0000008360181b16911617600755604051908152a1005b60046040517f1ada47b8000000000000000000000000000000000000000000000000000000008152fd5b610859614523565b6107d1565b3461000e57600060031936011261000e576020601154604051908152f35b3461000e57600060031936011261000e57610895614465565b60ff600554166108275760006004557fbf1ce7fb3a8e648b70ea830f99b52f7ea31554186d29763280751f42e77f6386602060405160008152a1005b3461000e57602060031936011261000e576004356108ed615bb7565b60ff60055416610827576020817fbf1ce7fb3a8e648b70ea830f99b52f7ea31554186d29763280751f42e77f638692600455604051908152a1005b5061000e613db5565b3461000e57606060031936011261000e57600435602435610950613c84565b906109596146c6565b610961614bbc565b505050505060045461097e846001600160801b03601c5416613f1f565b11610a8e5761098b614e17565b505015610a64576109ab92816109a692610a52575b506143e7565b6155f9565b6109b361438a565b50601a54906109c28233615554565b156109d7576020906001600d55604051908152f35b506109f86109e361423c565b33600052601f60205260406000205490615f44565b33600052601e602052610a4e604060002054926040519384937fed27783c000000000000000000000000000000000000000000000000000000008552600485016040919493926060820195825260208201520152565b0390fd5b610a5e9033903361576e565b846109a0565b60046040517f345513d9000000000000000000000000000000000000000000000000000000008152fd5b60046040517f97ba4de3000000000000000000000000000000000000000000000000000000008152fd5b3461000e57600060031936011261000e576020600454604051908152f35b3461000e57600060031936011261000e57610aef615bb7565b6201000062ff00001960205416176020557f70328969870b42d0fc62ce5946530c50a466c5ad21af004a9f2f3a8eda9a4a0c600080a1005b3461000e57600060031936011261000e5760206001600160a01b0360035416604051908152f35b3461000e57600060031936011261000e57602060ff600754166040519015158152f35b3461000e57600060031936011261000e57610b8a615bb7565b600160ff1960205416176020557faa96740f913149dce2173396218295b4c082e86fabacc37ac8d45305239d26f3600080a1005b3461000e57604060031936011261000e57610bd7613c58565b610bdf613c6e565b906001600160a01b038091166000526009602052604060002091166000526020526020604060002054604051908152f35b3461000e57604060031936011261000e576020610c2b613df3565b610c33613c6e565b90610c3c613ec7565b8091610c46614216565b916001600160801b0380911615610ced575b91608091610cac7faf48306b6b4f0ba30d00f05b21559d8d29934142980a553d8a014780c6c7e4529486169282610c8f8583615f44565b97610c9b8633306140e2565b3092610ca68a6143e7565b90615217565b30600052601e86526001600160a01b0360406000205491610cce308285615850565b60405193845216868301528460408301526060820152a1604051908152f35b30600090815260088652604090205481169350610c58565b3461000e57602060031936011261000e57602061075d610d23613c58565b61430b565b3461000e57604060031936011261000e57610d41613c6e565b610d496146c6565b610d51614bbc565b505050505033600052601f602052604060002054610da5575b610d78903390600435615850565b610d8061438a565b50601a54610d8e8133615554565b15610d9a576001600d55005b6109f86109e361423c565b610dad614e17565b5050610d6a5760046040517f345513d9000000000000000000000000000000000000000000000000000000008152fd5b3461000e57600060031936011261000e5760206040516001600160a01b037f0000000000000000000000000000000000000000000000000000000000000000168152f35b3461000e57602060031936011261000e57602061075d610e3f613c58565b61426f565b3461000e57600060031936011261000e5760a0610e5f614767565b9350935050506001600160801b03908160208185511694015116916020818351169201511690601d54926040519485526020850152604084015260608301526080820152f35b3461000e57604060031936011261000e57610edf610ec1613c6e565b610ec96146c6565b610ed1614bbc565b50505050506004353361576e565b6001600d55005b3461000e57600060031936011261000e576105fe610f02614767565b9260409694969291925196879687613ce9565b3461000e57608060031936011261000e57610f2e613c58565b60643567ffffffffffffffff80821161000e573660238301121561000e5781600401351161000e57366024826004013560051b8301011161000e57610f716146c6565b610f79614bbc565b5050505050610f86614e17565b505015610a64576001600160a01b038216600052601360205260ff604060002054161561148957806004013515611473576001600160a01b03610fcb60248301615ba3565b817f000000000000000000000000000000000000000000000000000000000000000016918291160361143e57600482013560001981019081116114285761102261101d82856004013560248701615b93565b615ba3565b6001600160a01b03807f0000000000000000000000000000000000000000000000000000000000000000169116036113bf57506110623330602435615850565b6040517f095ea7b30000000000000000000000000000000000000000000000000000000081526001600160a01b0384166004820152602480359082015290602090829060449082906000905af180156112f757611386575b50604051906370a0823160e01b82523060048301526020826024816001600160a01b037f0000000000000000000000000000000000000000000000000000000000000000165afa9182156112f757600092611352575b5060405180917f38ed17390000000000000000000000000000000000000000000000000000000082526024356004830152604435602483015260a06044830152806004013560a483015260c4820190602481019060005b816004013581106113265750505090806000923060648301524260848301520381836001600160a01b0388165af180156112f757611303575b50604051906370a0823160e01b82523060048301526020826024816001600160a01b037f0000000000000000000000000000000000000000000000000000000000000000165afa80156112f7576000906112c3575b6111ff9250614262565b90604435821061128b5761121161423c565b61123b61121e8483615e79565b91611228856143e7565b33913091611235866143e7565b91615918565b6001600160a01b0360405192168252602435602083015282604083015260608201527fe947f0f9b6255bdcf76d13d1257d34fbe380e0d5d4daa75e61c783a41e1607ba60803392a26109b361438a565b604482604051907f76baadda000000000000000000000000000000000000000000000000000000008252823560048301526024820152fd5b506020823d6020116112ef575b816112dd60209383613e57565b8101031261000e576111ff91516111f5565b3d91506112d0565b6040513d6000823e3d90fd5b61131f903d806000833e6113178183613e57565b810190615b19565b50826111a0565b919350916020806001926001600160a01b0361134188613c9a565b168152019401910191849392611167565b9091506020813d60201161137e575b8161136e60209383613e57565b8101031261000e57519083611110565b3d9150611361565b6020813d6020116113b7575b8161139f60209383613e57565b8101031261000e576113b090614e0a565b50826110ba565b3d9150611392565b6113d861101d610a4e9285602481600401359101615b93565b60405163b0b3262d60e01b81527f00000000000000000000000000000000000000000000000000000000000000006001600160a01b03908116600483015290911660248201529081906044820190565b634e487b7160e01b600052601160045260246000fd5b61144a60248301615ba3565b60405163b0b3262d60e01b81526001600160a01b03928316600482015291166024820152604490fd5b634e487b7160e01b600052603260045260246000fd5b60046040517f1311dc6d000000000000000000000000000000000000000000000000000000008152fd5b3461000e57600060031936011261000e5760206040516001600160a01b037f0000000000000000000000000000000000000000000000000000000000000000168152f35b3461000e57602060031936011261000e57611510613c58565b50602061075d61151e614767565b5093505050506001600160801b0381511660065490818110600014611544575090615e79565b61154d91614262565b90615e79565b3461000e57600060031936011261000e57602060ff60075460281c166040519015158152f35b5061000e613d91565b3461000e57602060031936011261000e5760043561159e615bb7565b60ff60075416610827576020817f854df3eb95564502c8bc871ebdd15310ee26270f955f6c6bd8cea68e75045bc092600655604051908152a1005b3461000e57600060031936011261000e57602060075460ff60405191831c1615158152f35b3461000e5761160c36613e92565b91906116166146c6565b60ff60075460181c1661166f5760209261166291611632614bbc565b505050505061163f614216565b6116498582615f13565b9461165c611656876143e7565b916143e7565b91615217565b6001600d55604051908152f35b60046040517fe0a39803000000000000000000000000000000000000000000000000000000008152fd5b3461000e57600060031936011261000e57602060405161c3508152f35b3461000e57600060031936011261000e57602060ff815460081c166040519015158152f35b3461000e57602060031936011261000e576001600160a01b036116fc613c58565b166000526008602052604060002054601f60205260406000205490601e6020526105fe604060002054604051938493846040919493926060820195825260208201520152565b3461000e57602060031936011261000e576001600160a01b03611763613c58565b16600052601e6020526020604060002054604051908152f35b3461000e5761178a36613e92565b91906117946146c6565b60ff60075460181c1661166f57602092611662916117b0614bbc565b50505050506117bd614216565b6117d06117ca8683615e0f565b956143e7565b61165c866143e7565b5061000e613dd4565b3461000e57600060031936011261000e5760206040516001600160a01b037f0000000000000000000000000000000000000000000000000000000000000000168152f35b3461000e57604060031936011261000e5761184c611842613c58565b6024359033613f2c565b602060405160018152f35b3461000e57604060031936011261000e57611870613c58565b6024359033600052600960205260406000206001600160a01b038216600052602052604060002054918083106118ac5761184c920390336140e2565b608460405162461bcd60e51b815260206004820152602560248201527f45524332303a2064656372656173656420616c6c6f77616e63652062656c6f7760448201527f207a65726f0000000000000000000000000000000000000000000000000000006064820152fd5b3461000e5760a060031936011261000e5761192f613c58565b6084359067ffffffffffffffff821161000e573660238301121561000e57816004013561195b81613e7a565b926119696040519485613e57565b818452602084016024819360051b8301019136831161000e57602401905b828210611e74575050506119996146c6565b6119a1614bbc565b50505050506119ae614e17565b505015610a64576001600160a01b038216600052601360205260ff6040600020541615611489576001600160a01b036119e684615af8565b5116926001600160a01b037f000000000000000000000000000000000000000000000000000000000000000016809403611e355780516000199081810190811161142857611a3c6001600160a01b039184615b05565b51166001600160a01b037f00000000000000000000000000000000000000000000000000000000000000001603611dc05750604435611dae575b611a8b611a846024356143e7565b30906155f9565b6040517f095ea7b30000000000000000000000000000000000000000000000000000000081526001600160a01b038516600482015260248035908201529094602090829060449082906000905af180156112f757611d75575b50604051916370a0823160e01b83523060048401526020836024816001600160a01b037f0000000000000000000000000000000000000000000000000000000000000000165afa9283156112f757600093611d41575b506040517f38ed17390000000000000000000000000000000000000000000000000000000081526024803560048301526064359082015260a06044820152915160a48301819052829160c483019160005b818110611d1f5750505090806000923060648301524260848301520381836001600160a01b0388165af180156112f757611d04575b50604051906370a0823160e01b82523060048301526020826024816001600160a01b037f0000000000000000000000000000000000000000000000000000000000000000165afa80156112f757600090611cd0575b611c1f9250614262565b6064358110611c9757611c3333823061576e565b611c3f81604435613f1f565b926001600160a01b036040519316835260243560208401526040830152604435606083015260808201527fb19ca0df3f3a01af950d8e6ad62aeff167cf14c73e98af6c52afef1add5c97ed60a03392a26109b361438a565b604490604051907f76baadda00000000000000000000000000000000000000000000000000000000825260643560048301526024820152fd5b506020823d602011611cfc575b81611cea60209383613e57565b8101031261000e57611c1f9151611c15565b3d9150611cdd565b611d18903d806000833e6113178183613e57565b5083611bc0565b82516001600160a01b0316845285945060209384019390920191600101611b8b565b9092506020813d602011611d6d575b81611d5d60209383613e57565b8101031261000e57519185611b3a565b3d9150611d50565b6020813d602011611da6575b81611d8e60209383613e57565b8101031261000e57611d9f90614e0a565b5084611ae4565b3d9150611d81565b611dbb336044353361576e565b611a76565b815190810190811161142857611de16001600160a01b0391610a4e93615b05565b5160405163b0b3262d60e01b81526001600160a01b037f0000000000000000000000000000000000000000000000000000000000000000811660048301529290911690911660248201529081906044820190565b6001600160a01b03611e478592615af8565b5160405163b0b3262d60e01b81526001600160a01b03938416600482015291169091166024820152604490fd5b60208091611e8184613c9a565b815201910190611987565b3461000e57600060031936011261000e57602060ff815460181c166040519015158152f35b3461000e57600060031936011261000e57610100604051620186a0808252806020830152806040830152806060830152670de0b6b3a76400009081608084015260a083015260c082015261c35060e0820152f35b3461000e57600060031936011261000e57611f1e614767565b509350505050602081016001600160801b03918282511615600014611f53575050506020670de0b6b3a7640000604051908152f35b51670de0b6b3a76400009083168181029182040361142857602092611f7a9251169061482a565b61075d565b3461000e57600060031936011261000e57604051600090601554600181811c90808316928315612076575b6020938484108114612060578386529081156120405750600114611fe5575b6105fe84611fd981880382613e57565b60405191829182613c10565b601560009081529294507f55f448fdea98c4d29eb340757ef0a66cd03dbb9538908a6a81d96026b71ec4755b82841061202d57505050816105fe93611fd99282010193611fc9565b8054858501870152928501928101612011565b60ff1916858501525050151560051b8201019150611fd9816105fe611fc9565b634e487b7160e01b600052602260045260246000fd5b91607f1691611faa565b3461000e57600060031936011261000e5760a06016546040519063ffffffff80821683528160201c16602083015267ffffffffffffffff808260401c1660408401528160801c16606083015260c01c6080820152f35b3461000e57604060031936011261000e576004356120f2613c6e565b6120fa6146c6565b612102614bbc565b505050505061210f614216565b9161211a8184615f13565b91600654612132846001600160801b03875116613f1f565b11612150576020936116629261214a611656866143e7565b916150c6565b60046040517f2ab4a214000000000000000000000000000000000000000000000000000000008152fd5b3461000e57602061218a36613cbd565b156121a65761075d9161219b614767565b945050505050615ea6565b611f7a916121b261423c565b615ea6565b3461000e57604060031936011261000e576121d0613c58565b63ffffffff602435818116929183820361000e576121ec615bb7565b60ff6020541661231f576080937f78ba1c32ac8ea4b3d51133dd0b6f5d8f98e23797aade6afc381ea317d5d4f28b8561227e9361222761438a565b966001600160a01b0390818951169260208a015116604051938452602084015216938460408301526060820152a16001600160a01b031673ffffffffffffffffffffffffffffffffffffffff196017541617601755565b7fffffffffffffffff00000000ffffffffffffffffffffffffffffffffffffffff77ffffffff00000000000000000000000000000000000000006017549260a01b1691161760175576ffffffffffffffffffffffffffffffffffffffffffffff6040820151167fffffffffffffffffff0000000000000000000000000000000000000000000000601854161760185560608101516019550151601a55600080f35b60046040517f8c34a9b8000000000000000000000000000000000000000000000000000000008152fd5b3461000e57600060031936011261000e5760206001600160a01b0360025416604051908152f35b3461000e57602060031936011261000e576001600160a01b03612391613c58565b166000526013602052602060ff604060002054166040519015158152f35b3461000e57602060031936011261000e577f3ff713beec3d10b4dfe28953471682eab1f857ba2fdb6367366252381888a75060206004356123ee615bb7565b600160ff19600754161760075580600655604051908152a1005b3461000e57602060031936011261000e57612421613cae565b80156124995761242f614465565b60ff60075460401c166108275760207fdea8bb46eee4300a7d2de86939c245f568dc5994576194cbfb69969e010dcb6791612468614bbc565b5050505050151560075467ff000000000000001967ff000000000000008360381b16911617600755604051908152a1005b6124a1614523565b61242f565b3461000e57600060031936011261000e576124bf614465565b60ff806005541615612580575b806007548181161561254b575b60101c161561253e575b8060075460201c1615612531575b8060075460301c1615612524575b60075460401c161561250d57005b612515614bbc565b505050505061252261463a565b005b61252c6145f6565b6124ff565b6125396145b6565b6124f1565b61254661457a565b6124e3565b60006006557f854df3eb95564502c8bc871ebdd15310ee26270f955f6c6bd8cea68e75045bc0602060405160008152a16124d9565b60006004557fbf1ce7fb3a8e648b70ea830f99b52f7ea31554186d29763280751f42e77f6386602060405160008152a16124cc565b3461000e57600060031936011261000e57601c54604080516001600160801b038316815260809290921c602083015290f35b3461000e57602060031936011261000e5760043563ffffffff81169081810361000e57612612615bb7565b60ff60075460381c166126c45761c350821161269a577f58a58c712558f3d6e20bed57421eb8f73048d881dea9e5bb80efb37c49680d1c91602091612655614bbc565b50505050507fffffffffffffffffffffffffffffffffffffffffffffffff00000000ffffffff67ffffffff0000000060165492851b16911617601655604051908152a1005b60046040517fda0afa57000000000000000000000000000000000000000000000000000000008152fd5b60046040517fa02a2bcd000000000000000000000000000000000000000000000000000000008152fd5b3461000e57600060031936011261000e57612707615bb7565b66010000000000007fffffffffffffffffffffffffffffffffffffffffffffffffff00ffffffffffff60075416176007557f60c2acdf5b421891c8cc7302420292f2680f0e835fc76dd15f35a7bb0dd5cbc8600080a1005b3461000e57602061276f36613cbd565b1561278b5761075d91612780614767565b945050505050615f7f565b611f7a9161279761423c565b615f7f565b3461000e57600060031936011261000e57602060ff815460101c166040519015158152f35b3461000e5760206127d136613cbd565b156127ed5761075d916127e2614767565b509350505050615f7f565b611f7a91612797614216565b3461000e57600060031936011261000e576003546001600160a01b0333818316036128705773ffffffffffffffffffffffffffffffffffffffff198092166003556002549133908316176002553391167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0600080a3005b608460405162461bcd60e51b815260206004820152602960248201527f4f776e61626c6532537465703a2063616c6c6572206973206e6f74207468652060448201527f6e6577206f776e657200000000000000000000000000000000000000000000006064820152fd5b3461000e57606060031936011261000e576128f3613df3565b602435906128ff613c84565b916129086146c6565b60ff60075460281c16612c1f57804211612be85750612925614bbc565b5050505050612932614e17565b50905061293f8184615554565b612bbe5761294b61423c565b906001600160a01b0384169384600052601e60205260406000205492601f60205261297a6040600020546143e7565b926000936001600160801b0396670de0b6b3a76400006129a6898316946129a18688615f13565b614817565b0498600f5497620186a0988901808a11611428576129c58a918d614817565b046129d86129d283615a64565b91615a64565b90600082820392128183128116918313901516176114285760001280159190612b9e5760209b505b809960115480612b5c575b50505086959293612ab4979486938b6116629c9d612a9a958d612a39612a34612aa29d8c615f44565b6143e7565b98600097600093612abc575b509160c093917f821de4e13fff1938b3806eb2859b6a5d55111f00dcf286f8a793584228ff36f895936040519485526020850152828b166040850152606084015281881660808401521660a0820152a261485f565b903392615918565b612aad813387615850565b3083615850565b30903061576e565b7f821de4e13fff1938b3806eb2859b6a5d55111f00dcf286f8a793584228ff36f8959391985091612af08860c09694614682565b98838d818c1680612b0a575b505050919395509193612a45565b82955090612b1e612a34612b289383615f13565b9586915116614682565b168d52601b546fffffffffffffffffffffffffffffffff1985612b4d86828516614682565b16911617601b55838d38612afc565b6116629b50829a5093612a9a93612b8b8b9a9793612b83612ab49d9a96612aa29c99614817565b04809d614262565b9c50935093968296508195989950612a0b565b506010548901808a1161142857612bb88a9160209d614817565b04612a00565b60046040517f75e595fa000000000000000000000000000000000000000000000000000000008152fd5b604490604051907f5ba2a8d50000000000000000000000000000000000000000000000000000000082524260048301526024820152fd5b60046040517f6d2c35dc000000000000000000000000000000000000000000000000000000008152fd5b3461000e57600060031936011261000e57612c62613ec7565b60006001600160a01b0373ffffffffffffffffffffffffffffffffffffffff198060035416600355600254908116600255167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e08280a3005b3461000e57602060031936011261000e576001600160a01b03612cdb613c58565b1660005260086020526020604060002054604051908152f35b3461000e57604060031936011261000e57600435612d10613c6e565b90612d196146c6565b612d21614bbc565b5050505050612d2e614216565b600654612d45836001600160801b03845116613f1f565b116121505760209281612d64612d5e8561166295615e79565b946143e7565b61214a856143e7565b3461000e57602060031936011261000e57612d86613c58565b612d8e615bb7565b60ff60205460101c1661231f57601254604080516001600160a01b0380841682528481166020830152929373ffffffffffffffffffffffffffffffffffffffff19939290917faeae842c8b3cd009fbb602e1ed072dc1aec69750e431ceae97f7543b466cd04c9190a116911617601255600080f35b3461000e57600060031936011261000e57612e1c614465565b60ff600754166108275760006006557f854df3eb95564502c8bc871ebdd15310ee26270f955f6c6bd8cea68e75045bc0602060405160008152a1005b3461000e57600060031936011261000e57602060ff60075460181c166040519015158152f35b3461000e57600060031936011261000e57612e97615bb7565b680100000000000000007fffffffffffffffffffffffffffffffffffffffffffffff00ffffffffffffffff60075416176007557f16c0a933c76f28f1abdcef88bcea1650397c5f4bb4bf491a0d451a65cae016b6600080a1005b3461000e57600060031936011261000e576060604051600381526000602082015260006040820152f35b3461000e57602060031936011261000e576001600160a01b03612f3c613c58565b16600052601f6020526020604060002054604051908152f35b3461000e57600060031936011261000e57612f6e615bb7565b612f76615bf5565b73ffffffffffffffffffffffffffffffffffffffff19806000541660005560015460006001600160a01b03821681817f162998b90abc2507f3953aa797827b03a14c42dbd9a35f09feaf02e0d592773a8280a37f31b6c5a04b069b6ec1b3cef44c4e7c1eadd721349cda9823d0b1877b3551cdc68280a316600155005b3461000e57600060031936011261000e5761300c615bb7565b6201000062ff00001960075416176007557f269ac55859865c2ff127a862e95c81ce7e3b9b13582036d3df419df5c07ec8b4600080a1005b3461000e57602060031936011261000e5761305d613cae565b80156130c55761306b614465565b6007549060ff8260301c16610827577f28bc4f9e24da61e7ba3aa697dfaefd0167093d2425c00b6190a7d3152ee6dfaa9160209115159065ff00000000001965ff00000000008360281b16911617600755604051908152a1005b6130cd614523565b61306b565b3461000e57600060031936011261000e5760206001600160a01b0360015416604051908152f35b3461000e57602060031936011261000e57613112613cae565b801561317257613120614465565b6007549060ff8260101c16610827577f34a71a12fa81891b738d910d4d44ffabeeb12f8bc026844db237ea8bf8ebe8be9160209115159061ff001961ff008360081b16911617600755604051908152a1005b61317a614523565b613120565b3461000e57600060031936011261000e576020601d54604051908152f35b3461000e57600060031936011261000e576020601054604051908152f35b3461000e57600060031936011261000e57602060ff60075460401c166040519015158152f35b3461000e57602060031936011261000e576131fa613c58565b613202615bb7565b6001600160a01b03809116908173ffffffffffffffffffffffffffffffffffffffff196000541617600055600154167f162998b90abc2507f3953aa797827b03a14c42dbd9a35f09feaf02e0d592773a600080a3005b3461000e57602060031936011261000e57613271613c58565b5060206001600160801b03613284614767565b505160065495169350505050818110156132a15750604051908152f35b611f7a91614262565b3461000e57600060031936011261000e576132c3614523565b60ff806005541615613343575b806007548181161561330d5760101c161561253e578060075460201c1615612531578060075460301c16156125245760075460401c161561250d57005b7f854df3eb95564502c8bc871ebdd15310ee26270f955f6c6bd8cea68e75045bc0602060001980600655604051908152a16124d9565b7fbf1ce7fb3a8e648b70ea830f99b52f7ea31554186d29763280751f42e77f6386602060001980600455604051908152a16132d0565b3461000e57604060031936011261000e57613392613c58565b6024359081151580920361000e577fea1eefb4fd58778d7b274fe54045a9feeec8f2847899c2e71126d3a74d486da5916001600160a01b036040926133d5613ec7565b16908160005260136020528260002060ff1981541660ff831617905582519182526020820152a1005b3461000e57604060031936011261000e5760043561341a613c6e565b906134236146c6565b60ff60075460081c166134625761166260209261343e614bbc565b505050505061344b61423c565b6134558482615f44565b93612a9a611656866143e7565b60046040517f3cc383d2000000000000000000000000000000000000000000000000000000008152fd5b3461000e57604060031936011261000e5761184c6134a8613c58565b33600052600960205260406000206001600160a01b0382166000526020526134d7602435604060002054613f1f565b90336140e2565b3461000e57600060031936011261000e576134f7615bb7565b61010061ff001960205416176020557f0af6d9d6ea0e3f0cdb71562ce1fce30aa597445ea04f5b25a939cfe0a252171c600080a1005b3461000e57600060031936011261000e5760206040516001600160a01b037f0000000000000000000000000000000000000000000000000000000000000000168152f35b3461000e57600060031936011261000e57602060ff60075460301c166040519015158152f35b3461000e57600060031936011261000e57602060405160ff7f0000000000000000000000000000000000000000000000000000000000000000168152f35b3461000e57606060031936011261000e576135ee613c58565b6135f6613c6e565b604435906001600160a01b0383166000526009602052604060002033600052602052604060002054926000198403613633575b61184c9350613f2c565b82841061364f5761364a8361184c950333836140e2565b613629565b606460405162461bcd60e51b815260206004820152601d60248201527f45524332303a20696e73756666696369656e7420616c6c6f77616e63650000006044820152fd5b3461000e57602060031936011261000e576136ac613cae565b6136b461469b565b506040516136c181613e09565b60008152600060208201526105fe604051916136dc83613e09565b60008352600060208401526136ef6146c6565b916136f8614bbc565b929693509096613714575b6001600d5560405196879687613ce9565b92509050613720614216565b9061372961423c565b92613703565b3461000e57602061373f36613cbd565b1561375b5761075d91613750614767565b509350505050615ea6565b611f7a916121b2614216565b3461000e57606060031936011261000e57604435602435600435613789615bb7565b60ff60205460181c1661231f577fc9aa62b60be8f25ac9f285edbb80bde64199b3c53e1da1027058551d32695fca60c0600f5460105460115490604051928352602083015260408201528360608201528460808201528560a0820152a1600f55601055601155005b3461000e57600060031936011261000e576020601b5460801c604051908152f35b3461000e57600060031936011261000e576020600f54604051908152f35b3461000e57600060031936011261000e57602060ff60075460081c166040519015158152f35b3461000e57600060031936011261000e57602060ff60075460101c166040519015158152f35b3461000e57600060031936011261000e57602060ff8154166040519015158152f35b3461000e57600060031936011261000e57602060ff600554166040519015158152f35b3461000e57602060031936011261000e57602061075d6138df614767565b50935050505060043590615e0f565b3461000e57604060031936011261000e5761184c61390a613c58565b60243590336140e2565b3461000e57600060031936011261000e5760206001600160a01b0360005416604051908152f35b3461000e57602060031936011261000e57600435613957615bb7565b60ff60205460081c1661231f577fe796e9ae748449310fcd1cc6718aab236c9b8d2e0e04dacb232ba564d5b338cc6040600e548151908152836020820152a1600e55005b3461000e57600060031936011261000e576139b4615bb7565b630100000063ff0000001960205416176020557f1cd8398e5a04411acbddcb6451a57b51c242322c538947cea5e4a1a506700b87600080a1005b3461000e57602060031936011261000e57602061075d613a0c614767565b50935050505060043590615f13565b3461000e57600060031936011261000e57604051600090601454600181811c90808316928315613acf575b6020938484108114612060578386529081156120405750600114613a74576105fe84611fd981880382613e57565b601460009081529294507fce6d7b5282bd9a3661ae061feed1dbda4e52ab073b1f9285be6e155d9c38d4ec5b828410613abc57505050816105fe93611fd99282010193611fc9565b8054858501870152928501928101613aa0565b91607f1691613a46565b3461000e57600060031936011261000e57613af2615bb7565b6401000000007fffffffffffffffffffffffffffffffffffffffffffffffffffffff00ffffffff60075416176007557fb949af551d0c88280e648f9205b986bb5f1d899c425498238655ee37617c0c39600080a1005b3461000e57600060031936011261000e57613b616146c6565b6060613b6b614e17565b906001600d55604051921515835260208301526040820152f35b3461000e57600060031936011261000e5760206001600160801b03613ba8614767565b50516040519516855250505050f35b3461000e57602060031936011261000e577fee4b3f9e70b2c6499288c7b5fbef140756009cf8839be64c473b1c7cb6d616c46020600435613bf6615bb7565b600160ff19600554161760055580600455604051908152a1005b60208082528251818301819052939260005b858110613c4457505050601f19601f8460006040809697860101520116010190565b818101830151848201604001528201613c22565b600435906001600160a01b038216820361000e57565b602435906001600160a01b038216820361000e57565b604435906001600160a01b038216820361000e57565b35906001600160a01b038216820361000e57565b60043590811515820361000e57565b600319606091011261000e5760043590602435801515810361000e5790604435801515810361000e5790565b9194613d729197969461014094613d8f9761018086019a86526020860152604085015263ffffffff8082511660608601526020820151166080850152608067ffffffffffffffff918260408201511660a08701528260608201511660c087015201511660e0840152610100830190602090816001600160801b0391828151168552015116910152565b0190602090816001600160801b0391828151168552015116910152565b565b503461000e57600060031936011261000e576020604051670de0b6b3a76400008152f35b503461000e57600060031936011261000e576020604051620186a08152f35b503461000e57602060031936011261000e57602061075d613a0c614767565b600435906001600160801b038216820361000e57565b6040810190811067ffffffffffffffff821117613e2557604052565b634e487b7160e01b600052604160045260246000fd5b60a0810190811067ffffffffffffffff821117613e2557604052565b90601f601f19910116810190811067ffffffffffffffff821117613e2557604052565b67ffffffffffffffff8111613e255760051b60200190565b600319606091011261000e57600435906001600160a01b0390602435828116810361000e5791604435908116810361000e5790565b6001600160a01b03600254163303613edb57565b606460405162461bcd60e51b815260206004820152602060248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e65726044820152fd5b9190820180921161142857565b6001600160a01b03809116918215614078571691821561400e5760008281526008602052604081205491808310613fa457604082827fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef95876020965260088652038282205586815220818154019055604051908152a3565b608460405162461bcd60e51b815260206004820152602660248201527f45524332303a207472616e7366657220616d6f756e742065786365656473206260448201527f616c616e636500000000000000000000000000000000000000000000000000006064820152fd5b608460405162461bcd60e51b815260206004820152602360248201527f45524332303a207472616e7366657220746f20746865207a65726f206164647260448201527f65737300000000000000000000000000000000000000000000000000000000006064820152fd5b608460405162461bcd60e51b815260206004820152602560248201527f45524332303a207472616e736665722066726f6d20746865207a65726f20616460448201527f64726573730000000000000000000000000000000000000000000000000000006064820152fd5b6001600160a01b038091169182156141ad57169182156141435760207f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b925918360005260098252604060002085600052825280604060002055604051908152a3565b608460405162461bcd60e51b815260206004820152602260248201527f45524332303a20617070726f766520746f20746865207a65726f20616464726560448201527f73730000000000000000000000000000000000000000000000000000000000006064820152fd5b608460405162461bcd60e51b8152602060048201526024808201527f45524332303a20617070726f76652066726f6d20746865207a65726f2061646460448201527f72657373000000000000000000000000000000000000000000000000000000006064820152fd5b6040519061422382613e09565b601b546001600160801b038116835260801c6020830152565b6040519061424982613e09565b601c546001600160801b038116835260801c6020830152565b9190820391821161142857565b60ff60075460181c16614305576142e06001600160a01b0361428f614767565b9593509691935050169060003083146000146142f157506142bd916000526008602052604060002054613f1f565b905b6142d96001600160801b0391828087511691511690614682565b1692615f13565b808210156142ec575090565b905090565b9050604091815260086020522054906142bf565b50600090565b60ff60075460181c16614305576143536001600160a01b0361432b614767565b9296945092505061434c6001600160801b0391828085511691511690614682565b1690615e79565b921690600030830361437757506142e0916000526008602052604060002054613f1f565b90506040918152600860205220546142e0565b6040519061439782613e3b565b6017546001600160a01b038116835260a01c63ffffffff16602083015260185476ffffffffffffffffffffffffffffffffffffffffffffff1660408301526019546060830152601a546080830152565b6001600160801b03908181116143fb571690565b608460405162461bcd60e51b815260206004820152602760248201527f53616665436173743a2076616c756520646f65736e27742066697420696e203160448201527f32382062697473000000000000000000000000000000000000000000000000006064820152fd5b6001600160a01b03807f0000000000000000000000000000000000000000000000000000000000000000163314159081614513575b816144e6575b816144d7575b506144ad57565b60046040517f1d1e647b000000000000000000000000000000000000000000000000000000008152fd5b905060015416331415386144a6565b337f00000000000000000000000000000000000000000000000000000000000000008216141591506144a0565b809150600254163314159061449a565b6001600160a01b038060025416331415908161456b575b5061454157565b60046040517f6f545269000000000000000000000000000000000000000000000000000000008152fd5b9050600154163314153861453a565b61010061ff001960075416176007557f34a71a12fa81891b738d910d4d44ffabeeb12f8bc026844db237ea8bf8ebe8be602060405160018152a1565b630100000063ff0000001960075416176007557fc56dd3e14f5af3a74c61b7cdf855a3d8ab4401c78c0622a4d312de8a8f8736a2602060405160018152a1565b6501000000000065ff00000000001960075416176007557f28bc4f9e24da61e7ba3aa697dfaefd0167093d2425c00b6190a7d3152ee6dfaa602060405160018152a1565b67010000000000000067ff000000000000001960075416176007557fdea8bb46eee4300a7d2de86939c245f568dc5994576194cbfb69969e010dcb67602060405160018152a1565b6001600160801b03918216908216039190821161142857565b604051906146a882613e3b565b60006080838281528260208201528260408201528260608201520152565b6002600d54146146d7576002600d55565b606460405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152fd5b6040519061472882613e3b565b81608060165463ffffffff80821684528160201c16602084015267ffffffffffffffff808260401c16604085015281831c16606084015260c01c910152565b6000908190818061477661469b565b5080602060405161478681613e09565b8281520152602060405161479981613e09565b82815201526147a661471b565b936147b08561487a565b8051909490156147fc5750505060608201519260808301519260a08101519267ffffffffffffffff806020840151166060850152604083015116608084015260e060c083015192015190565b9290935061480b949194614216565b9061481461423c565b90565b8181029291811591840414171561142857565b8115614834570490565b634e487b7160e01b600052601260045260246000fd5b519067ffffffffffffffff8216820361000e57565b9190916001600160801b038080941691160191821161142857565b6040805192919067ffffffffffffffff610100850181811186821017613e25578252600091828652602080870184815282880195858752606089019386855260808a019587875260a08b0198888a5260c08c019483516148d981613e09565b8a81528a88820152865260e08d019380516148f381613e09565b8b81528b8982015285528d818701908582511642141580614bac575b614924575b5050505050505050505050505050565b614949916001879252614935614216565b8a5261493f61423c565b8852511642614262565b926001600160801b039b8c8951511615600014614b6f57805b836001600160a01b03601254169160648960808d01511691835194859384927fcd3181d50000000000000000000000000000000000000000000000000000000084528c6004850152602484015260448301525afa918215614b645780948193614aff575b50505092856149f1936149fe969382670de0b6b3a764000099971690521690528b8651511690614817565b9060608601511690614817565b04808652878115159182614ae5575b505080614acc575b614a25575b808080808080614914565b63ffffffff91849188614a41818951169251928284511661485f565b1690528786511688614a588651928284511661485f565b1690520151169283614a6b575b80614a1a565b614abd94614aa7614a99620186a0614a888a98614aad9651614817565b048084528787875101511690614817565b918685515116905190614262565b9061482a565b809652510193168284511661485f565b16905238808080808080614a65565b5086614ade8651828651511690613f1f565b1115614a15565b81614af69293508451511690613f1f565b11158738614a0d565b919450915083813d8111614b5d575b614b188183613e57565b81010312614b5a575092849283836149fe96670de0b6b3a764000098614b4b8e614b446149f19961484a565b940161484a565b949750985093968195506149c6565b80fd5b503d614b0e565b8451903d90823e3d90fd5b8c87515116620186a09080820291820403614b9857614b93908e8b5151169061482a565b614962565b602482634e487b7160e01b81526011600452fd5b5060ff60075460381c161561490f565b600090600090600090600090614bd061469b565b50614bd961471b565b90614be38261487a565b8051151580614bf0575050565b93509550925092509260608101519260808201519260a083015192602081015167ffffffffffffffff90818116606086015260408301519180831660808701524216604086015277ffffffffffffffff0000000000000000000000000000000063ffffffff4316928387527fffffffffffffffff00000000000000000000000000000000000000000000000067ffffffff00000000602089015160201b169160c01b16936fffffffffffffffff00000000000000004260401b169117179160801b16171760165560c0810151602060e0816001600160801b0393614cf6858251166001600160801b03166fffffffffffffffffffffffffffffffff19601b541617601b55565b015193601b54846fffffffffffffffffffffffffffffffff19809760801b16911617601b550151614d49838251166001600160801b03166fffffffffffffffffffffffffffffffff19601c541617601c55565b015191601c549260801b16911617601c5582614d6157565b613d8f83305b6001600160a01b0316908115614dc6577fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef602082614da9600094600a54613f1f565b600a558484526008825260408420818154019055604051908152a3565b606460405162461bcd60e51b815260206004820152601f60248201527f45524332303a206d696e7420746f20746865207a65726f2061646472657373006044820152fd5b5190811515820361000e57565b600090600090614e2561438a565b906040918281019376ffffffffffffffffffffffffffffffffffffffffffffff93848651164214156000146150b15760046001600160a01b03966060888651168451938480927fbd9a548b0000000000000000000000000000000000000000000000000000000082525afa9081156150a45784988593869361502d575b5090614f1283927fc1f41e029acf5127d111625602160c4cee3e1a4d38e691e50544d1f7c68b77be9695949a859c614ffb575b42168093528460608a01528360808a01528851166001600160a01b031673ffffffffffffffffffffffffffffffffffffffff196017541617601755565b60208701517fffffffffffffffff00000000ffffffffffffffffffffffffffffffffffffffff77ffffffff00000000000000000000000000000000000000006017549260a01b169116176017557fffffffffffffffffff000000000000000000000000000000000000000000000060185416176018558160195580601a5582519182526020820152a15b60808201614fb08151606085015190614262565b91620186a09280840293840403614fe75750614fd563ffffffff92602092519061482a565b920151161015614fe157565b60019350565b80634e487b7160e01b602492526011600452fd5b7ffc131c36b7e444dacda44901fd43641dcdcfdc43fe9e2601b3c1dd87061db9e56020838c51168951908152a1614ed5565b9950915091506060883d821161509c575b8161504b60609383613e57565b8101031261509857907fc1f41e029acf5127d111625602160c4cee3e1a4d38e691e50544d1f7c68b77be929161508089614e0a565b60208a01519984015190999394509190614f12614ea2565b8380fd5b3d915061503e565b50505051903d90823e3d90fd5b50606082015160808301519095509350614f9c565b7fdcbc1c05240f31ff3ad067ef1ee35ce4997762752e3a095284754544f4c709d79192936151ef6001600160a01b03926001600160801b03908161510d888284511661485f565b16815261515d826020830192816151278c8287511661485f565b168452615136828c1688614d67565b51166001600160801b03166fffffffffffffffffffffffffffffffff19601b541617601b55565b51816fffffffffffffffffffffffffffffffff19601b549260801b16911617601b55604051907f23b872dd00000000000000000000000000000000000000000000000000000000602083015233602483015230604483015286166064820152606481526151c981613e3b565b7f0000000000000000000000000000000000000000000000000000000000000000615c33565b604080516001600160801b03958616815295909416602086015216923392819081015b0390a3565b9091926152546001600160a01b039283871696873303615502575b5061523b61423c565b6001600160801b03928391828085511691511690614682565b1691808616928381106154c05750806152708782855116614682565b1682526020916152b8828483019261528b8b83865116614682565b93828516905251166001600160801b03166fffffffffffffffffffffffffffffffff19601b541617601b55565b816fffffffffffffffffffffffffffffffff19601b549260801b16911617601b55861691871561545657876000526008825260409283600020548181106153ed576151c9846153b67ffbde797d201c681b91056529119e0b02407c7bb96a4a2c75c01fc9667232c8db9998979560008e7fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef6153e89a896153c49a7fa9059cbb000000000000000000000000000000000000000000000000000000009a85875260088452038c86205580600a5403600a558b51908152a386519485938401528860248401602090939291936001600160a01b0360408201951681520152565b03601f198101835282613e57565b516001600160801b0395861681529590941660208601521692339281906040820190565b0390a4565b60848486519062461bcd60e51b82526004820152602260248201527f45524332303a206275726e20616d6f756e7420657863656564732062616c616e60448201527f63650000000000000000000000000000000000000000000000000000000000006064820152fd5b6084826040519062461bcd60e51b82526004820152602160248201527f45524332303a206275726e2066726f6d20746865207a65726f2061646472657360448201527f73000000000000000000000000000000000000000000000000000000000000006064820152fd5b6040517fc5bb6dae00000000000000000000000000000000000000000000000000000000815260048101919091526001600160801b0387166024820152604490fd5b87600052600960205260406000203360005260205260406000205490600019820361552e575b50615232565b61554561554d926001600160801b038a1690614262565b9033906140e2565b3880615528565b90600e549182156155f1576001600160a01b0361556f61423c565b91169161558b600092848452601f602052604084205490615f44565b9283156155e7578252601e60205260408220549283156155df57670de0b6b3a7640000916155b891614817565b0490620186a091828102928184041490151715614fe75750906155da9161482a565b111590565b505091505090565b5050505050600190565b505050600190565b919061560361423c565b6001600160801b03908161562581615619614216565b51168284511690614682565b1691808616928381106154c057508061564b6156418585615e0f565b978285511661485f565b168252615693816020840193615666828a168387511661485f565b94828616905251166001600160801b03166fffffffffffffffffffffffffffffffff19601c541617601c55565b6fffffffffffffffffffffffffffffffff19601c549260801b16911617601c5533600052601f60205260406000206156cc858254613f1f565b90556001600160a01b0382169181308403615718575b50506040519081528360208201527f01348584ec81ac7acd52b7d66d9ade986dd909f3d513881c190fc31c90527efe60403392a3565b6040517fa9059cbb0000000000000000000000000000000000000000000000000000000060208201526001600160a01b0390921660248301526044820152615767906151c981606481016153b6565b38816156e2565b9160207fa32435755c235de2976ed44a75a2f85cb01faf0c894f639fe0c32bb9455fea8f916001600160a01b038091169485600052601e835260406000206157b7868254613f1f565b90556157c585601d54613f1f565b601d5516923084036157db575b604051908152a3565b61584b6040517f23b872dd00000000000000000000000000000000000000000000000000000000848201528560248201523060448201528260648201526064815261582581613e3b565b7f0000000000000000000000000000000000000000000000000000000000000000615c33565b6157d2565b6001600160a01b038093169283600052601e6020526040600020615875838254614262565b905561588382601d54614262565b601d55821691813084036158c2575b50506040519081527fbc290bb45104f73cf92115c9603987c3f8fd30c182a13603d8cffa49b5f5995260203392a4565b6040517fa9059cbb0000000000000000000000000000000000000000000000000000000060208201526001600160a01b03909216602483015260448201526159119061582581606481016153b6565b3881615892565b93907f9dc1449a0ff0c152e18e8289d865b47acc6e1b76b1ecb239c13d6ee22a9206a792916001600160801b03948561595484828a5116614682565b16875260208701866159698682845116614682565b1681526159c0876001600160a01b03809516998a600052601f6020526040600020615997838a168254614262565b905551166001600160801b03166fffffffffffffffffffffffffffffffff19601c541617601c55565b51866fffffffffffffffffffffffffffffffff19601c549260801b16911617601c551693308503615a10575b50604080516001600160801b03928316815292909116602083015281908101615212565b615a5e90604051907f23b872dd00000000000000000000000000000000000000000000000000000000602083015286602483015230604483015283166064820152606481526151c981613e3b565b386159ec565b7f7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff8111615a8e5790565b608460405162461bcd60e51b815260206004820152602860248201527f53616665436173743a2076616c756520646f65736e27742066697420696e206160448201527f6e20696e743235360000000000000000000000000000000000000000000000006064820152fd5b8051156114735760200190565b80518210156114735760209160051b010190565b602090818184031261000e5780519067ffffffffffffffff821161000e57019180601f8401121561000e578251615b4f81613e7a565b93615b5d6040519586613e57565b818552838086019260051b82010192831161000e578301905b828210615b84575050505090565b81518152908301908301615b76565b91908110156114735760051b0190565b356001600160a01b038116810361000e5790565b6001600160a01b03600154163303615bcb57565b60046040517f1c0be90a000000000000000000000000000000000000000000000000000000008152fd5b6001600160a01b03600054163303615c0957565b60046040517ff5c49e64000000000000000000000000000000000000000000000000000000008152fd5b6001600160a01b031690604051615c4981613e09565b6020928382527f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564848301526000808486829651910182855af13d15615d75573d9167ffffffffffffffff8311615d615790615cc493929160405192615cb788601f19601f8401160185613e57565b83523d868885013e615d7f565b90815180615cd3575b50505050565b82849181010312614b5a575081615cea9101614e0a565b15615cf757808080615ccd565b6084906040519062461bcd60e51b82526004820152602a60248201527f5361666545524332303a204552433230206f7065726174696f6e20646964206e60448201527f6f742073756363656564000000000000000000000000000000000000000000006064820152fd5b602485634e487b7160e01b81526041600452fd5b90615cc492916060915b91929015615de05750815115615d93575090565b3b15615d9c5790565b606460405162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e74726163740000006044820152fd5b825190915015615df35750805190602001fd5b610a4e9060405191829162461bcd60e51b835260048301613c10565b91906001600160801b038084511615600014615e2a57509150565b615e63906020850190615e5a81615e51615e478286511688614817565b828a51169061482a565b97511687614817565b9151169061482a565b10615e6a57565b90600181018091116114285790565b6001600160801b038082511615600014615e9257505090565b615e5a614814938260208501511690614817565b90916001600160801b038083511615600014615ec25750505090565b602083959492930190615ee5615edb8284511685614817565b828851169061482a565b9584615ef6575b50505050615e6a57565b615f0993945081615e5a91511687614817565b1038808080615eec565b60208101906001600160801b03908183511615600014615f335750505090565b6148149382615e5a92511690614817565b919060208301926001600160801b038085511615600014615f66575090925050565b9081615e5a81615e51615e47615e639686511688614817565b909160208201916001600160801b038084511615600014615fa1575050505090565b615ee5615edb8284989795969851168561481756fea164736f6c6343000811000a55f448fdea98c4d29eb340757ef0a66cd03dbb9538908a6a81d96026b71ec475c1f41e029acf5127d111625602160c4cee3e1a4d38e691e50544d1f7c68b77be000000000000000000000000000000000000000000000000000000000000006000000000000000000000000000000000000000000000000000000000000001a000000000000000000000000000000000000000000000000000000000000002200000000000000000000000000000000000000000000000000000000000000120000000000000000000000000853d955acef822db058eb8505911ed77f175b99e000000000000000000000000ac3e018457b222d93114458476f3e3416abbe38f000000000000000000000000d2f0fa7f2e6a60eecf4b78c5b6d81002b9789f2c000000000000000000000000000000000000000000000000000000000000138800000000000000000000000018500cb1f2fe7a40ebda393383a0b8548a31f2610000000000000000000000000000000000000000000000000000000235ef7f6800000000000000000000000000000000000000000000000000000000000124f8000000000000000000000000000000000000000000000000000000000000271000000000000000000000000000000000000000000000000000000000000007d00000000000000000000000000000000000000000000000000000000000000060000000000000000000000000fd3065c629ee890fd74f43b802c2fea4b7279b8c000000000000000000000000168200cf227d4543302686124ac28ae0eaf2ca0b0000000000000000000000008412ebf45bac1b340bbe8f318b928c466c4e39ca0000000000000000000000000000000000000000000000000000000000000100000000000000000000000000000000000000000000000000000000000000006000000000000000000000000000000000000000000000000000000000000000c000000000000000000000000000000000000000000000000000000000000000120000000000000000000000000000000000000000000000000000000000000037467261786c656e6420496e7465726573742042656172696e67204652415820285374616b6564204672617820457468657229202d203133000000000000000000000000000000000000000000000000000000000000000000000000000000001166465241582873667278455448292d3133000000000000000000000000000000
Deployed Bytecode
0x60806040526004361015610013575b600080fd5b60003560e01c8062348d1814613bb757806301e1d11414613b8557806302ce728f14613b485780630475260e14613ad957806306fdde0314613a1b57806307a2d13a146139ee5780630880b2f01461399b57806308a0c3751461393b578063090f3f5014613914578063095ea7b3146138ee5780630a28a477146138c15780630a67918c1461389e5780630b4aecf01461387c5780630c70661d14613856578063115a334c1461383057806311a2e4bc1461381257806318160ddd146137f15780631bc23cf9146137675780631c2591d31461372f5780631c6c95971461369357806323b872dd146135d55780632b3ba68114611579578063313ce5671461359757806334680fe51461357157806338d52e0f1461352d57806339030864146134de578063395093511461348c5780633d417d2d146133fe5780633f2617cb146133795780633f4ba83a146132aa578063402d267d1461325857806345014095146131e157806345757e4a146131bb5780634732428c1461319d57806349eb7af6146109285780634ac8eb5f1461317f5780634b4b418e146130f95780634bc66f32146130d25780634c18a4fb146130445780634c41799514612ff35780634cdad506146117d95780634f8b4ae714612f555780634fd422df14612f1b57806354fd4d5014612ef157806364e51d5d14612e7e57806367800b5f14612e5857806369026e8814612e035780636b96668f14612d6d5780636cd3cc77146109285780636e553f6514612cf457806370a0823114612cba578063715018a614612c49578063721b0a47146128da57806379ba5097146127f95780637d37bdd7146127c15780637d63fbc21461279c5780637ec4b5711461275f5780637ecefa6e146126ee5780638142dd53146125e75780638285ef40146125b55780638456cb59146124a6578063858f1e6814612408578063886c033a146123af5780638cad7fbe146123705780638da5cb5b146123495780638f791f8b146121b757806393f46f641461217a57806394bf804d146120d657806395d14ca81461208057806395d89b4114611f7f57806399530b0614611f055780639a295e7314611eb15780639bc6ab8614611e8c5780639fe34bdc14610928578063a053db6814611916578063a457c2d714611857578063a9059cbb14611826578063ad0c3bb5146117e2578063b3d7f6b9146117d9578063b460af941461177c578063b5af306214611742578063b68d0a09146116db578063b78294dd14610928578063b7db54f5146116b6578063b8ca3b8314611699578063ba087652146115fe578063bbb09624146115d9578063bdc8144b14611582578063c0a7e89214611579578063c58e4df614611553578063c63d75b6146114f7578063c6e1c7c9146114b3578063c6e6f59214610730578063ca2298fe14610f15578063cacf3b5814610ee6578063cadac47914610ea5578063cdd72d5214610e44578063ce96cb7714610e21578063d2a156e014610ddd578063d41ddc9614610d28578063d905777e14610d05578063daf33f2a14610c10578063dd62ed3e14610bbe578063e1e9277514610b71578063e203641714610b4e578063e30c397814610b27578063e4b0007014610ad6578063e551d11d14610ab8578063e5f13b1614610931578063e63a391f14610928578063e7a33174146108d1578063e8596f721461087c578063eafecffa1461085e578063ebd462cb146107aa578063ecf708581461078c578063eee2421914610765578063ef8b30f714610730578063f211c3901461070a578063f2fde38b14610693578063f384bd0514610675578063f6ccaad414610602578063f9557ccb146105cc5763fbbbf94c1461056557600080fd5b3461000e57600060031936011261000e5760a060175476ffffffffffffffffffffffffffffffffffffffffffffff60185416601954601a549163ffffffff604051946001600160a01b0381168652861c166020850152604084015260608301526080820152f35b3461000e57600060031936011261000e57601b54604080516001600160801b038316815260809290921c602083015290f35b0390f35b3461000e57600060031936011261000e5761061b615bf5565b73ffffffffffffffffffffffffffffffffffffffff19806000541660005560015490336001600160a01b0383167f31b6c5a04b069b6ec1b3cef44c4e7c1eadd721349cda9823d0b1877b3551cdc6600080a3163317600155005b3461000e57600060031936011261000e576020600e54604051908152f35b3461000e57602060031936011261000e576106ac613c58565b6106b4613ec7565b6001600160a01b03809116908173ffffffffffffffffffffffffffffffffffffffff196003541617600355600254167f38d16b8cac22d99fc7c124b9cd0de2d3fa1faef420bfe791d8c362d765e22700600080a3005b3461000e57600060031936011261000e57602060ff60075460381c166040519015158152f35b3461000e57602060031936011261000e57602061075d61074e614767565b50935050505060043590615e79565b604051908152f35b3461000e57600060031936011261000e5760206001600160a01b0360125416604051908152f35b3461000e57600060031936011261000e576020600654604051908152f35b3461000e57602060031936011261000e576107c3613cae565b8015610851576107d1614465565b6007549060ff8260201c16610827577fc56dd3e14f5af3a74c61b7cdf855a3d8ab4401c78c0622a4d312de8a8f8736a29160209115159063ff0000001963ff0000008360181b16911617600755604051908152a1005b60046040517f1ada47b8000000000000000000000000000000000000000000000000000000008152fd5b610859614523565b6107d1565b3461000e57600060031936011261000e576020601154604051908152f35b3461000e57600060031936011261000e57610895614465565b60ff600554166108275760006004557fbf1ce7fb3a8e648b70ea830f99b52f7ea31554186d29763280751f42e77f6386602060405160008152a1005b3461000e57602060031936011261000e576004356108ed615bb7565b60ff60055416610827576020817fbf1ce7fb3a8e648b70ea830f99b52f7ea31554186d29763280751f42e77f638692600455604051908152a1005b5061000e613db5565b3461000e57606060031936011261000e57600435602435610950613c84565b906109596146c6565b610961614bbc565b505050505060045461097e846001600160801b03601c5416613f1f565b11610a8e5761098b614e17565b505015610a64576109ab92816109a692610a52575b506143e7565b6155f9565b6109b361438a565b50601a54906109c28233615554565b156109d7576020906001600d55604051908152f35b506109f86109e361423c565b33600052601f60205260406000205490615f44565b33600052601e602052610a4e604060002054926040519384937fed27783c000000000000000000000000000000000000000000000000000000008552600485016040919493926060820195825260208201520152565b0390fd5b610a5e9033903361576e565b846109a0565b60046040517f345513d9000000000000000000000000000000000000000000000000000000008152fd5b60046040517f97ba4de3000000000000000000000000000000000000000000000000000000008152fd5b3461000e57600060031936011261000e576020600454604051908152f35b3461000e57600060031936011261000e57610aef615bb7565b6201000062ff00001960205416176020557f70328969870b42d0fc62ce5946530c50a466c5ad21af004a9f2f3a8eda9a4a0c600080a1005b3461000e57600060031936011261000e5760206001600160a01b0360035416604051908152f35b3461000e57600060031936011261000e57602060ff600754166040519015158152f35b3461000e57600060031936011261000e57610b8a615bb7565b600160ff1960205416176020557faa96740f913149dce2173396218295b4c082e86fabacc37ac8d45305239d26f3600080a1005b3461000e57604060031936011261000e57610bd7613c58565b610bdf613c6e565b906001600160a01b038091166000526009602052604060002091166000526020526020604060002054604051908152f35b3461000e57604060031936011261000e576020610c2b613df3565b610c33613c6e565b90610c3c613ec7565b8091610c46614216565b916001600160801b0380911615610ced575b91608091610cac7faf48306b6b4f0ba30d00f05b21559d8d29934142980a553d8a014780c6c7e4529486169282610c8f8583615f44565b97610c9b8633306140e2565b3092610ca68a6143e7565b90615217565b30600052601e86526001600160a01b0360406000205491610cce308285615850565b60405193845216868301528460408301526060820152a1604051908152f35b30600090815260088652604090205481169350610c58565b3461000e57602060031936011261000e57602061075d610d23613c58565b61430b565b3461000e57604060031936011261000e57610d41613c6e565b610d496146c6565b610d51614bbc565b505050505033600052601f602052604060002054610da5575b610d78903390600435615850565b610d8061438a565b50601a54610d8e8133615554565b15610d9a576001600d55005b6109f86109e361423c565b610dad614e17565b5050610d6a5760046040517f345513d9000000000000000000000000000000000000000000000000000000008152fd5b3461000e57600060031936011261000e5760206040516001600160a01b037f000000000000000000000000aa913c26dd7723fcae9dbd2036d28171a56c6251168152f35b3461000e57602060031936011261000e57602061075d610e3f613c58565b61426f565b3461000e57600060031936011261000e5760a0610e5f614767565b9350935050506001600160801b03908160208185511694015116916020818351169201511690601d54926040519485526020850152604084015260608301526080820152f35b3461000e57604060031936011261000e57610edf610ec1613c6e565b610ec96146c6565b610ed1614bbc565b50505050506004353361576e565b6001600d55005b3461000e57600060031936011261000e576105fe610f02614767565b9260409694969291925196879687613ce9565b3461000e57608060031936011261000e57610f2e613c58565b60643567ffffffffffffffff80821161000e573660238301121561000e5781600401351161000e57366024826004013560051b8301011161000e57610f716146c6565b610f79614bbc565b5050505050610f86614e17565b505015610a64576001600160a01b038216600052601360205260ff604060002054161561148957806004013515611473576001600160a01b03610fcb60248301615ba3565b817f000000000000000000000000ac3e018457b222d93114458476f3e3416abbe38f16918291160361143e57600482013560001981019081116114285761102261101d82856004013560248701615b93565b615ba3565b6001600160a01b03807f000000000000000000000000853d955acef822db058eb8505911ed77f175b99e169116036113bf57506110623330602435615850565b6040517f095ea7b30000000000000000000000000000000000000000000000000000000081526001600160a01b0384166004820152602480359082015290602090829060449082906000905af180156112f757611386575b50604051906370a0823160e01b82523060048301526020826024816001600160a01b037f000000000000000000000000853d955acef822db058eb8505911ed77f175b99e165afa9182156112f757600092611352575b5060405180917f38ed17390000000000000000000000000000000000000000000000000000000082526024356004830152604435602483015260a06044830152806004013560a483015260c4820190602481019060005b816004013581106113265750505090806000923060648301524260848301520381836001600160a01b0388165af180156112f757611303575b50604051906370a0823160e01b82523060048301526020826024816001600160a01b037f000000000000000000000000853d955acef822db058eb8505911ed77f175b99e165afa80156112f7576000906112c3575b6111ff9250614262565b90604435821061128b5761121161423c565b61123b61121e8483615e79565b91611228856143e7565b33913091611235866143e7565b91615918565b6001600160a01b0360405192168252602435602083015282604083015260608201527fe947f0f9b6255bdcf76d13d1257d34fbe380e0d5d4daa75e61c783a41e1607ba60803392a26109b361438a565b604482604051907f76baadda000000000000000000000000000000000000000000000000000000008252823560048301526024820152fd5b506020823d6020116112ef575b816112dd60209383613e57565b8101031261000e576111ff91516111f5565b3d91506112d0565b6040513d6000823e3d90fd5b61131f903d806000833e6113178183613e57565b810190615b19565b50826111a0565b919350916020806001926001600160a01b0361134188613c9a565b168152019401910191849392611167565b9091506020813d60201161137e575b8161136e60209383613e57565b8101031261000e57519083611110565b3d9150611361565b6020813d6020116113b7575b8161139f60209383613e57565b8101031261000e576113b090614e0a565b50826110ba565b3d9150611392565b6113d861101d610a4e9285602481600401359101615b93565b60405163b0b3262d60e01b81527f000000000000000000000000853d955acef822db058eb8505911ed77f175b99e6001600160a01b03908116600483015290911660248201529081906044820190565b634e487b7160e01b600052601160045260246000fd5b61144a60248301615ba3565b60405163b0b3262d60e01b81526001600160a01b03928316600482015291166024820152604490fd5b634e487b7160e01b600052603260045260246000fd5b60046040517f1311dc6d000000000000000000000000000000000000000000000000000000008152fd5b3461000e57600060031936011261000e5760206040516001600160a01b037f000000000000000000000000ac3e018457b222d93114458476f3e3416abbe38f168152f35b3461000e57602060031936011261000e57611510613c58565b50602061075d61151e614767565b5093505050506001600160801b0381511660065490818110600014611544575090615e79565b61154d91614262565b90615e79565b3461000e57600060031936011261000e57602060ff60075460281c166040519015158152f35b5061000e613d91565b3461000e57602060031936011261000e5760043561159e615bb7565b60ff60075416610827576020817f854df3eb95564502c8bc871ebdd15310ee26270f955f6c6bd8cea68e75045bc092600655604051908152a1005b3461000e57600060031936011261000e57602060075460ff60405191831c1615158152f35b3461000e5761160c36613e92565b91906116166146c6565b60ff60075460181c1661166f5760209261166291611632614bbc565b505050505061163f614216565b6116498582615f13565b9461165c611656876143e7565b916143e7565b91615217565b6001600d55604051908152f35b60046040517fe0a39803000000000000000000000000000000000000000000000000000000008152fd5b3461000e57600060031936011261000e57602060405161c3508152f35b3461000e57600060031936011261000e57602060ff815460081c166040519015158152f35b3461000e57602060031936011261000e576001600160a01b036116fc613c58565b166000526008602052604060002054601f60205260406000205490601e6020526105fe604060002054604051938493846040919493926060820195825260208201520152565b3461000e57602060031936011261000e576001600160a01b03611763613c58565b16600052601e6020526020604060002054604051908152f35b3461000e5761178a36613e92565b91906117946146c6565b60ff60075460181c1661166f57602092611662916117b0614bbc565b50505050506117bd614216565b6117d06117ca8683615e0f565b956143e7565b61165c866143e7565b5061000e613dd4565b3461000e57600060031936011261000e5760206040516001600160a01b037f000000000000000000000000fd3065c629ee890fd74f43b802c2fea4b7279b8c168152f35b3461000e57604060031936011261000e5761184c611842613c58565b6024359033613f2c565b602060405160018152f35b3461000e57604060031936011261000e57611870613c58565b6024359033600052600960205260406000206001600160a01b038216600052602052604060002054918083106118ac5761184c920390336140e2565b608460405162461bcd60e51b815260206004820152602560248201527f45524332303a2064656372656173656420616c6c6f77616e63652062656c6f7760448201527f207a65726f0000000000000000000000000000000000000000000000000000006064820152fd5b3461000e5760a060031936011261000e5761192f613c58565b6084359067ffffffffffffffff821161000e573660238301121561000e57816004013561195b81613e7a565b926119696040519485613e57565b818452602084016024819360051b8301019136831161000e57602401905b828210611e74575050506119996146c6565b6119a1614bbc565b50505050506119ae614e17565b505015610a64576001600160a01b038216600052601360205260ff6040600020541615611489576001600160a01b036119e684615af8565b5116926001600160a01b037f000000000000000000000000853d955acef822db058eb8505911ed77f175b99e16809403611e355780516000199081810190811161142857611a3c6001600160a01b039184615b05565b51166001600160a01b037f000000000000000000000000ac3e018457b222d93114458476f3e3416abbe38f1603611dc05750604435611dae575b611a8b611a846024356143e7565b30906155f9565b6040517f095ea7b30000000000000000000000000000000000000000000000000000000081526001600160a01b038516600482015260248035908201529094602090829060449082906000905af180156112f757611d75575b50604051916370a0823160e01b83523060048401526020836024816001600160a01b037f000000000000000000000000ac3e018457b222d93114458476f3e3416abbe38f165afa9283156112f757600093611d41575b506040517f38ed17390000000000000000000000000000000000000000000000000000000081526024803560048301526064359082015260a06044820152915160a48301819052829160c483019160005b818110611d1f5750505090806000923060648301524260848301520381836001600160a01b0388165af180156112f757611d04575b50604051906370a0823160e01b82523060048301526020826024816001600160a01b037f000000000000000000000000ac3e018457b222d93114458476f3e3416abbe38f165afa80156112f757600090611cd0575b611c1f9250614262565b6064358110611c9757611c3333823061576e565b611c3f81604435613f1f565b926001600160a01b036040519316835260243560208401526040830152604435606083015260808201527fb19ca0df3f3a01af950d8e6ad62aeff167cf14c73e98af6c52afef1add5c97ed60a03392a26109b361438a565b604490604051907f76baadda00000000000000000000000000000000000000000000000000000000825260643560048301526024820152fd5b506020823d602011611cfc575b81611cea60209383613e57565b8101031261000e57611c1f9151611c15565b3d9150611cdd565b611d18903d806000833e6113178183613e57565b5083611bc0565b82516001600160a01b0316845285945060209384019390920191600101611b8b565b9092506020813d602011611d6d575b81611d5d60209383613e57565b8101031261000e57519185611b3a565b3d9150611d50565b6020813d602011611da6575b81611d8e60209383613e57565b8101031261000e57611d9f90614e0a565b5084611ae4565b3d9150611d81565b611dbb336044353361576e565b611a76565b815190810190811161142857611de16001600160a01b0391610a4e93615b05565b5160405163b0b3262d60e01b81526001600160a01b037f000000000000000000000000ac3e018457b222d93114458476f3e3416abbe38f811660048301529290911690911660248201529081906044820190565b6001600160a01b03611e478592615af8565b5160405163b0b3262d60e01b81526001600160a01b03938416600482015291169091166024820152604490fd5b60208091611e8184613c9a565b815201910190611987565b3461000e57600060031936011261000e57602060ff815460181c166040519015158152f35b3461000e57600060031936011261000e57610100604051620186a0808252806020830152806040830152806060830152670de0b6b3a76400009081608084015260a083015260c082015261c35060e0820152f35b3461000e57600060031936011261000e57611f1e614767565b509350505050602081016001600160801b03918282511615600014611f53575050506020670de0b6b3a7640000604051908152f35b51670de0b6b3a76400009083168181029182040361142857602092611f7a9251169061482a565b61075d565b3461000e57600060031936011261000e57604051600090601554600181811c90808316928315612076575b6020938484108114612060578386529081156120405750600114611fe5575b6105fe84611fd981880382613e57565b60405191829182613c10565b601560009081529294507f55f448fdea98c4d29eb340757ef0a66cd03dbb9538908a6a81d96026b71ec4755b82841061202d57505050816105fe93611fd99282010193611fc9565b8054858501870152928501928101612011565b60ff1916858501525050151560051b8201019150611fd9816105fe611fc9565b634e487b7160e01b600052602260045260246000fd5b91607f1691611faa565b3461000e57600060031936011261000e5760a06016546040519063ffffffff80821683528160201c16602083015267ffffffffffffffff808260401c1660408401528160801c16606083015260c01c6080820152f35b3461000e57604060031936011261000e576004356120f2613c6e565b6120fa6146c6565b612102614bbc565b505050505061210f614216565b9161211a8184615f13565b91600654612132846001600160801b03875116613f1f565b11612150576020936116629261214a611656866143e7565b916150c6565b60046040517f2ab4a214000000000000000000000000000000000000000000000000000000008152fd5b3461000e57602061218a36613cbd565b156121a65761075d9161219b614767565b945050505050615ea6565b611f7a916121b261423c565b615ea6565b3461000e57604060031936011261000e576121d0613c58565b63ffffffff602435818116929183820361000e576121ec615bb7565b60ff6020541661231f576080937f78ba1c32ac8ea4b3d51133dd0b6f5d8f98e23797aade6afc381ea317d5d4f28b8561227e9361222761438a565b966001600160a01b0390818951169260208a015116604051938452602084015216938460408301526060820152a16001600160a01b031673ffffffffffffffffffffffffffffffffffffffff196017541617601755565b7fffffffffffffffff00000000ffffffffffffffffffffffffffffffffffffffff77ffffffff00000000000000000000000000000000000000006017549260a01b1691161760175576ffffffffffffffffffffffffffffffffffffffffffffff6040820151167fffffffffffffffffff0000000000000000000000000000000000000000000000601854161760185560608101516019550151601a55600080f35b60046040517f8c34a9b8000000000000000000000000000000000000000000000000000000008152fd5b3461000e57600060031936011261000e5760206001600160a01b0360025416604051908152f35b3461000e57602060031936011261000e576001600160a01b03612391613c58565b166000526013602052602060ff604060002054166040519015158152f35b3461000e57602060031936011261000e577f3ff713beec3d10b4dfe28953471682eab1f857ba2fdb6367366252381888a75060206004356123ee615bb7565b600160ff19600754161760075580600655604051908152a1005b3461000e57602060031936011261000e57612421613cae565b80156124995761242f614465565b60ff60075460401c166108275760207fdea8bb46eee4300a7d2de86939c245f568dc5994576194cbfb69969e010dcb6791612468614bbc565b5050505050151560075467ff000000000000001967ff000000000000008360381b16911617600755604051908152a1005b6124a1614523565b61242f565b3461000e57600060031936011261000e576124bf614465565b60ff806005541615612580575b806007548181161561254b575b60101c161561253e575b8060075460201c1615612531575b8060075460301c1615612524575b60075460401c161561250d57005b612515614bbc565b505050505061252261463a565b005b61252c6145f6565b6124ff565b6125396145b6565b6124f1565b61254661457a565b6124e3565b60006006557f854df3eb95564502c8bc871ebdd15310ee26270f955f6c6bd8cea68e75045bc0602060405160008152a16124d9565b60006004557fbf1ce7fb3a8e648b70ea830f99b52f7ea31554186d29763280751f42e77f6386602060405160008152a16124cc565b3461000e57600060031936011261000e57601c54604080516001600160801b038316815260809290921c602083015290f35b3461000e57602060031936011261000e5760043563ffffffff81169081810361000e57612612615bb7565b60ff60075460381c166126c45761c350821161269a577f58a58c712558f3d6e20bed57421eb8f73048d881dea9e5bb80efb37c49680d1c91602091612655614bbc565b50505050507fffffffffffffffffffffffffffffffffffffffffffffffff00000000ffffffff67ffffffff0000000060165492851b16911617601655604051908152a1005b60046040517fda0afa57000000000000000000000000000000000000000000000000000000008152fd5b60046040517fa02a2bcd000000000000000000000000000000000000000000000000000000008152fd5b3461000e57600060031936011261000e57612707615bb7565b66010000000000007fffffffffffffffffffffffffffffffffffffffffffffffffff00ffffffffffff60075416176007557f60c2acdf5b421891c8cc7302420292f2680f0e835fc76dd15f35a7bb0dd5cbc8600080a1005b3461000e57602061276f36613cbd565b1561278b5761075d91612780614767565b945050505050615f7f565b611f7a9161279761423c565b615f7f565b3461000e57600060031936011261000e57602060ff815460101c166040519015158152f35b3461000e5760206127d136613cbd565b156127ed5761075d916127e2614767565b509350505050615f7f565b611f7a91612797614216565b3461000e57600060031936011261000e576003546001600160a01b0333818316036128705773ffffffffffffffffffffffffffffffffffffffff198092166003556002549133908316176002553391167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0600080a3005b608460405162461bcd60e51b815260206004820152602960248201527f4f776e61626c6532537465703a2063616c6c6572206973206e6f74207468652060448201527f6e6577206f776e657200000000000000000000000000000000000000000000006064820152fd5b3461000e57606060031936011261000e576128f3613df3565b602435906128ff613c84565b916129086146c6565b60ff60075460281c16612c1f57804211612be85750612925614bbc565b5050505050612932614e17565b50905061293f8184615554565b612bbe5761294b61423c565b906001600160a01b0384169384600052601e60205260406000205492601f60205261297a6040600020546143e7565b926000936001600160801b0396670de0b6b3a76400006129a6898316946129a18688615f13565b614817565b0498600f5497620186a0988901808a11611428576129c58a918d614817565b046129d86129d283615a64565b91615a64565b90600082820392128183128116918313901516176114285760001280159190612b9e5760209b505b809960115480612b5c575b50505086959293612ab4979486938b6116629c9d612a9a958d612a39612a34612aa29d8c615f44565b6143e7565b98600097600093612abc575b509160c093917f821de4e13fff1938b3806eb2859b6a5d55111f00dcf286f8a793584228ff36f895936040519485526020850152828b166040850152606084015281881660808401521660a0820152a261485f565b903392615918565b612aad813387615850565b3083615850565b30903061576e565b7f821de4e13fff1938b3806eb2859b6a5d55111f00dcf286f8a793584228ff36f8959391985091612af08860c09694614682565b98838d818c1680612b0a575b505050919395509193612a45565b82955090612b1e612a34612b289383615f13565b9586915116614682565b168d52601b546fffffffffffffffffffffffffffffffff1985612b4d86828516614682565b16911617601b55838d38612afc565b6116629b50829a5093612a9a93612b8b8b9a9793612b83612ab49d9a96612aa29c99614817565b04809d614262565b9c50935093968296508195989950612a0b565b506010548901808a1161142857612bb88a9160209d614817565b04612a00565b60046040517f75e595fa000000000000000000000000000000000000000000000000000000008152fd5b604490604051907f5ba2a8d50000000000000000000000000000000000000000000000000000000082524260048301526024820152fd5b60046040517f6d2c35dc000000000000000000000000000000000000000000000000000000008152fd5b3461000e57600060031936011261000e57612c62613ec7565b60006001600160a01b0373ffffffffffffffffffffffffffffffffffffffff198060035416600355600254908116600255167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e08280a3005b3461000e57602060031936011261000e576001600160a01b03612cdb613c58565b1660005260086020526020604060002054604051908152f35b3461000e57604060031936011261000e57600435612d10613c6e565b90612d196146c6565b612d21614bbc565b5050505050612d2e614216565b600654612d45836001600160801b03845116613f1f565b116121505760209281612d64612d5e8561166295615e79565b946143e7565b61214a856143e7565b3461000e57602060031936011261000e57612d86613c58565b612d8e615bb7565b60ff60205460101c1661231f57601254604080516001600160a01b0380841682528481166020830152929373ffffffffffffffffffffffffffffffffffffffff19939290917faeae842c8b3cd009fbb602e1ed072dc1aec69750e431ceae97f7543b466cd04c9190a116911617601255600080f35b3461000e57600060031936011261000e57612e1c614465565b60ff600754166108275760006006557f854df3eb95564502c8bc871ebdd15310ee26270f955f6c6bd8cea68e75045bc0602060405160008152a1005b3461000e57600060031936011261000e57602060ff60075460181c166040519015158152f35b3461000e57600060031936011261000e57612e97615bb7565b680100000000000000007fffffffffffffffffffffffffffffffffffffffffffffff00ffffffffffffffff60075416176007557f16c0a933c76f28f1abdcef88bcea1650397c5f4bb4bf491a0d451a65cae016b6600080a1005b3461000e57600060031936011261000e576060604051600381526000602082015260006040820152f35b3461000e57602060031936011261000e576001600160a01b03612f3c613c58565b16600052601f6020526020604060002054604051908152f35b3461000e57600060031936011261000e57612f6e615bb7565b612f76615bf5565b73ffffffffffffffffffffffffffffffffffffffff19806000541660005560015460006001600160a01b03821681817f162998b90abc2507f3953aa797827b03a14c42dbd9a35f09feaf02e0d592773a8280a37f31b6c5a04b069b6ec1b3cef44c4e7c1eadd721349cda9823d0b1877b3551cdc68280a316600155005b3461000e57600060031936011261000e5761300c615bb7565b6201000062ff00001960075416176007557f269ac55859865c2ff127a862e95c81ce7e3b9b13582036d3df419df5c07ec8b4600080a1005b3461000e57602060031936011261000e5761305d613cae565b80156130c55761306b614465565b6007549060ff8260301c16610827577f28bc4f9e24da61e7ba3aa697dfaefd0167093d2425c00b6190a7d3152ee6dfaa9160209115159065ff00000000001965ff00000000008360281b16911617600755604051908152a1005b6130cd614523565b61306b565b3461000e57600060031936011261000e5760206001600160a01b0360015416604051908152f35b3461000e57602060031936011261000e57613112613cae565b801561317257613120614465565b6007549060ff8260101c16610827577f34a71a12fa81891b738d910d4d44ffabeeb12f8bc026844db237ea8bf8ebe8be9160209115159061ff001961ff008360081b16911617600755604051908152a1005b61317a614523565b613120565b3461000e57600060031936011261000e576020601d54604051908152f35b3461000e57600060031936011261000e576020601054604051908152f35b3461000e57600060031936011261000e57602060ff60075460401c166040519015158152f35b3461000e57602060031936011261000e576131fa613c58565b613202615bb7565b6001600160a01b03809116908173ffffffffffffffffffffffffffffffffffffffff196000541617600055600154167f162998b90abc2507f3953aa797827b03a14c42dbd9a35f09feaf02e0d592773a600080a3005b3461000e57602060031936011261000e57613271613c58565b5060206001600160801b03613284614767565b505160065495169350505050818110156132a15750604051908152f35b611f7a91614262565b3461000e57600060031936011261000e576132c3614523565b60ff806005541615613343575b806007548181161561330d5760101c161561253e578060075460201c1615612531578060075460301c16156125245760075460401c161561250d57005b7f854df3eb95564502c8bc871ebdd15310ee26270f955f6c6bd8cea68e75045bc0602060001980600655604051908152a16124d9565b7fbf1ce7fb3a8e648b70ea830f99b52f7ea31554186d29763280751f42e77f6386602060001980600455604051908152a16132d0565b3461000e57604060031936011261000e57613392613c58565b6024359081151580920361000e577fea1eefb4fd58778d7b274fe54045a9feeec8f2847899c2e71126d3a74d486da5916001600160a01b036040926133d5613ec7565b16908160005260136020528260002060ff1981541660ff831617905582519182526020820152a1005b3461000e57604060031936011261000e5760043561341a613c6e565b906134236146c6565b60ff60075460081c166134625761166260209261343e614bbc565b505050505061344b61423c565b6134558482615f44565b93612a9a611656866143e7565b60046040517f3cc383d2000000000000000000000000000000000000000000000000000000008152fd5b3461000e57604060031936011261000e5761184c6134a8613c58565b33600052600960205260406000206001600160a01b0382166000526020526134d7602435604060002054613f1f565b90336140e2565b3461000e57600060031936011261000e576134f7615bb7565b61010061ff001960205416176020557f0af6d9d6ea0e3f0cdb71562ce1fce30aa597445ea04f5b25a939cfe0a252171c600080a1005b3461000e57600060031936011261000e5760206040516001600160a01b037f000000000000000000000000853d955acef822db058eb8505911ed77f175b99e168152f35b3461000e57600060031936011261000e57602060ff60075460301c166040519015158152f35b3461000e57600060031936011261000e57602060405160ff7f0000000000000000000000000000000000000000000000000000000000000012168152f35b3461000e57606060031936011261000e576135ee613c58565b6135f6613c6e565b604435906001600160a01b0383166000526009602052604060002033600052602052604060002054926000198403613633575b61184c9350613f2c565b82841061364f5761364a8361184c950333836140e2565b613629565b606460405162461bcd60e51b815260206004820152601d60248201527f45524332303a20696e73756666696369656e7420616c6c6f77616e63650000006044820152fd5b3461000e57602060031936011261000e576136ac613cae565b6136b461469b565b506040516136c181613e09565b60008152600060208201526105fe604051916136dc83613e09565b60008352600060208401526136ef6146c6565b916136f8614bbc565b929693509096613714575b6001600d5560405196879687613ce9565b92509050613720614216565b9061372961423c565b92613703565b3461000e57602061373f36613cbd565b1561375b5761075d91613750614767565b509350505050615ea6565b611f7a916121b2614216565b3461000e57606060031936011261000e57604435602435600435613789615bb7565b60ff60205460181c1661231f577fc9aa62b60be8f25ac9f285edbb80bde64199b3c53e1da1027058551d32695fca60c0600f5460105460115490604051928352602083015260408201528360608201528460808201528560a0820152a1600f55601055601155005b3461000e57600060031936011261000e576020601b5460801c604051908152f35b3461000e57600060031936011261000e576020600f54604051908152f35b3461000e57600060031936011261000e57602060ff60075460081c166040519015158152f35b3461000e57600060031936011261000e57602060ff60075460101c166040519015158152f35b3461000e57600060031936011261000e57602060ff8154166040519015158152f35b3461000e57600060031936011261000e57602060ff600554166040519015158152f35b3461000e57602060031936011261000e57602061075d6138df614767565b50935050505060043590615e0f565b3461000e57604060031936011261000e5761184c61390a613c58565b60243590336140e2565b3461000e57600060031936011261000e5760206001600160a01b0360005416604051908152f35b3461000e57602060031936011261000e57600435613957615bb7565b60ff60205460081c1661231f577fe796e9ae748449310fcd1cc6718aab236c9b8d2e0e04dacb232ba564d5b338cc6040600e548151908152836020820152a1600e55005b3461000e57600060031936011261000e576139b4615bb7565b630100000063ff0000001960205416176020557f1cd8398e5a04411acbddcb6451a57b51c242322c538947cea5e4a1a506700b87600080a1005b3461000e57602060031936011261000e57602061075d613a0c614767565b50935050505060043590615f13565b3461000e57600060031936011261000e57604051600090601454600181811c90808316928315613acf575b6020938484108114612060578386529081156120405750600114613a74576105fe84611fd981880382613e57565b601460009081529294507fce6d7b5282bd9a3661ae061feed1dbda4e52ab073b1f9285be6e155d9c38d4ec5b828410613abc57505050816105fe93611fd99282010193611fc9565b8054858501870152928501928101613aa0565b91607f1691613a46565b3461000e57600060031936011261000e57613af2615bb7565b6401000000007fffffffffffffffffffffffffffffffffffffffffffffffffffffff00ffffffff60075416176007557fb949af551d0c88280e648f9205b986bb5f1d899c425498238655ee37617c0c39600080a1005b3461000e57600060031936011261000e57613b616146c6565b6060613b6b614e17565b906001600d55604051921515835260208301526040820152f35b3461000e57600060031936011261000e5760206001600160801b03613ba8614767565b50516040519516855250505050f35b3461000e57602060031936011261000e577fee4b3f9e70b2c6499288c7b5fbef140756009cf8839be64c473b1c7cb6d616c46020600435613bf6615bb7565b600160ff19600554161760055580600455604051908152a1005b60208082528251818301819052939260005b858110613c4457505050601f19601f8460006040809697860101520116010190565b818101830151848201604001528201613c22565b600435906001600160a01b038216820361000e57565b602435906001600160a01b038216820361000e57565b604435906001600160a01b038216820361000e57565b35906001600160a01b038216820361000e57565b60043590811515820361000e57565b600319606091011261000e5760043590602435801515810361000e5790604435801515810361000e5790565b9194613d729197969461014094613d8f9761018086019a86526020860152604085015263ffffffff8082511660608601526020820151166080850152608067ffffffffffffffff918260408201511660a08701528260608201511660c087015201511660e0840152610100830190602090816001600160801b0391828151168552015116910152565b0190602090816001600160801b0391828151168552015116910152565b565b503461000e57600060031936011261000e576020604051670de0b6b3a76400008152f35b503461000e57600060031936011261000e576020604051620186a08152f35b503461000e57602060031936011261000e57602061075d613a0c614767565b600435906001600160801b038216820361000e57565b6040810190811067ffffffffffffffff821117613e2557604052565b634e487b7160e01b600052604160045260246000fd5b60a0810190811067ffffffffffffffff821117613e2557604052565b90601f601f19910116810190811067ffffffffffffffff821117613e2557604052565b67ffffffffffffffff8111613e255760051b60200190565b600319606091011261000e57600435906001600160a01b0390602435828116810361000e5791604435908116810361000e5790565b6001600160a01b03600254163303613edb57565b606460405162461bcd60e51b815260206004820152602060248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e65726044820152fd5b9190820180921161142857565b6001600160a01b03809116918215614078571691821561400e5760008281526008602052604081205491808310613fa457604082827fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef95876020965260088652038282205586815220818154019055604051908152a3565b608460405162461bcd60e51b815260206004820152602660248201527f45524332303a207472616e7366657220616d6f756e742065786365656473206260448201527f616c616e636500000000000000000000000000000000000000000000000000006064820152fd5b608460405162461bcd60e51b815260206004820152602360248201527f45524332303a207472616e7366657220746f20746865207a65726f206164647260448201527f65737300000000000000000000000000000000000000000000000000000000006064820152fd5b608460405162461bcd60e51b815260206004820152602560248201527f45524332303a207472616e736665722066726f6d20746865207a65726f20616460448201527f64726573730000000000000000000000000000000000000000000000000000006064820152fd5b6001600160a01b038091169182156141ad57169182156141435760207f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b925918360005260098252604060002085600052825280604060002055604051908152a3565b608460405162461bcd60e51b815260206004820152602260248201527f45524332303a20617070726f766520746f20746865207a65726f20616464726560448201527f73730000000000000000000000000000000000000000000000000000000000006064820152fd5b608460405162461bcd60e51b8152602060048201526024808201527f45524332303a20617070726f76652066726f6d20746865207a65726f2061646460448201527f72657373000000000000000000000000000000000000000000000000000000006064820152fd5b6040519061422382613e09565b601b546001600160801b038116835260801c6020830152565b6040519061424982613e09565b601c546001600160801b038116835260801c6020830152565b9190820391821161142857565b60ff60075460181c16614305576142e06001600160a01b0361428f614767565b9593509691935050169060003083146000146142f157506142bd916000526008602052604060002054613f1f565b905b6142d96001600160801b0391828087511691511690614682565b1692615f13565b808210156142ec575090565b905090565b9050604091815260086020522054906142bf565b50600090565b60ff60075460181c16614305576143536001600160a01b0361432b614767565b9296945092505061434c6001600160801b0391828085511691511690614682565b1690615e79565b921690600030830361437757506142e0916000526008602052604060002054613f1f565b90506040918152600860205220546142e0565b6040519061439782613e3b565b6017546001600160a01b038116835260a01c63ffffffff16602083015260185476ffffffffffffffffffffffffffffffffffffffffffffff1660408301526019546060830152601a546080830152565b6001600160801b03908181116143fb571690565b608460405162461bcd60e51b815260206004820152602760248201527f53616665436173743a2076616c756520646f65736e27742066697420696e203160448201527f32382062697473000000000000000000000000000000000000000000000000006064820152fd5b6001600160a01b03807f000000000000000000000000fd3065c629ee890fd74f43b802c2fea4b7279b8c163314159081614513575b816144e6575b816144d7575b506144ad57565b60046040517f1d1e647b000000000000000000000000000000000000000000000000000000008152fd5b905060015416331415386144a6565b337f000000000000000000000000aa913c26dd7723fcae9dbd2036d28171a56c62518216141591506144a0565b809150600254163314159061449a565b6001600160a01b038060025416331415908161456b575b5061454157565b60046040517f6f545269000000000000000000000000000000000000000000000000000000008152fd5b9050600154163314153861453a565b61010061ff001960075416176007557f34a71a12fa81891b738d910d4d44ffabeeb12f8bc026844db237ea8bf8ebe8be602060405160018152a1565b630100000063ff0000001960075416176007557fc56dd3e14f5af3a74c61b7cdf855a3d8ab4401c78c0622a4d312de8a8f8736a2602060405160018152a1565b6501000000000065ff00000000001960075416176007557f28bc4f9e24da61e7ba3aa697dfaefd0167093d2425c00b6190a7d3152ee6dfaa602060405160018152a1565b67010000000000000067ff000000000000001960075416176007557fdea8bb46eee4300a7d2de86939c245f568dc5994576194cbfb69969e010dcb67602060405160018152a1565b6001600160801b03918216908216039190821161142857565b604051906146a882613e3b565b60006080838281528260208201528260408201528260608201520152565b6002600d54146146d7576002600d55565b606460405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152fd5b6040519061472882613e3b565b81608060165463ffffffff80821684528160201c16602084015267ffffffffffffffff808260401c16604085015281831c16606084015260c01c910152565b6000908190818061477661469b565b5080602060405161478681613e09565b8281520152602060405161479981613e09565b82815201526147a661471b565b936147b08561487a565b8051909490156147fc5750505060608201519260808301519260a08101519267ffffffffffffffff806020840151166060850152604083015116608084015260e060c083015192015190565b9290935061480b949194614216565b9061481461423c565b90565b8181029291811591840414171561142857565b8115614834570490565b634e487b7160e01b600052601260045260246000fd5b519067ffffffffffffffff8216820361000e57565b9190916001600160801b038080941691160191821161142857565b6040805192919067ffffffffffffffff610100850181811186821017613e25578252600091828652602080870184815282880195858752606089019386855260808a019587875260a08b0198888a5260c08c019483516148d981613e09565b8a81528a88820152865260e08d019380516148f381613e09565b8b81528b8982015285528d818701908582511642141580614bac575b614924575b5050505050505050505050505050565b614949916001879252614935614216565b8a5261493f61423c565b8852511642614262565b926001600160801b039b8c8951511615600014614b6f57805b836001600160a01b03601254169160648960808d01511691835194859384927fcd3181d50000000000000000000000000000000000000000000000000000000084528c6004850152602484015260448301525afa918215614b645780948193614aff575b50505092856149f1936149fe969382670de0b6b3a764000099971690521690528b8651511690614817565b9060608601511690614817565b04808652878115159182614ae5575b505080614acc575b614a25575b808080808080614914565b63ffffffff91849188614a41818951169251928284511661485f565b1690528786511688614a588651928284511661485f565b1690520151169283614a6b575b80614a1a565b614abd94614aa7614a99620186a0614a888a98614aad9651614817565b048084528787875101511690614817565b918685515116905190614262565b9061482a565b809652510193168284511661485f565b16905238808080808080614a65565b5086614ade8651828651511690613f1f565b1115614a15565b81614af69293508451511690613f1f565b11158738614a0d565b919450915083813d8111614b5d575b614b188183613e57565b81010312614b5a575092849283836149fe96670de0b6b3a764000098614b4b8e614b446149f19961484a565b940161484a565b949750985093968195506149c6565b80fd5b503d614b0e565b8451903d90823e3d90fd5b8c87515116620186a09080820291820403614b9857614b93908e8b5151169061482a565b614962565b602482634e487b7160e01b81526011600452fd5b5060ff60075460381c161561490f565b600090600090600090600090614bd061469b565b50614bd961471b565b90614be38261487a565b8051151580614bf0575050565b93509550925092509260608101519260808201519260a083015192602081015167ffffffffffffffff90818116606086015260408301519180831660808701524216604086015277ffffffffffffffff0000000000000000000000000000000063ffffffff4316928387527fffffffffffffffff00000000000000000000000000000000000000000000000067ffffffff00000000602089015160201b169160c01b16936fffffffffffffffff00000000000000004260401b169117179160801b16171760165560c0810151602060e0816001600160801b0393614cf6858251166001600160801b03166fffffffffffffffffffffffffffffffff19601b541617601b55565b015193601b54846fffffffffffffffffffffffffffffffff19809760801b16911617601b550151614d49838251166001600160801b03166fffffffffffffffffffffffffffffffff19601c541617601c55565b015191601c549260801b16911617601c5582614d6157565b613d8f83305b6001600160a01b0316908115614dc6577fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef602082614da9600094600a54613f1f565b600a558484526008825260408420818154019055604051908152a3565b606460405162461bcd60e51b815260206004820152601f60248201527f45524332303a206d696e7420746f20746865207a65726f2061646472657373006044820152fd5b5190811515820361000e57565b600090600090614e2561438a565b906040918281019376ffffffffffffffffffffffffffffffffffffffffffffff93848651164214156000146150b15760046001600160a01b03966060888651168451938480927fbd9a548b0000000000000000000000000000000000000000000000000000000082525afa9081156150a45784988593869361502d575b5090614f1283927fc1f41e029acf5127d111625602160c4cee3e1a4d38e691e50544d1f7c68b77be9695949a859c614ffb575b42168093528460608a01528360808a01528851166001600160a01b031673ffffffffffffffffffffffffffffffffffffffff196017541617601755565b60208701517fffffffffffffffff00000000ffffffffffffffffffffffffffffffffffffffff77ffffffff00000000000000000000000000000000000000006017549260a01b169116176017557fffffffffffffffffff000000000000000000000000000000000000000000000060185416176018558160195580601a5582519182526020820152a15b60808201614fb08151606085015190614262565b91620186a09280840293840403614fe75750614fd563ffffffff92602092519061482a565b920151161015614fe157565b60019350565b80634e487b7160e01b602492526011600452fd5b7ffc131c36b7e444dacda44901fd43641dcdcfdc43fe9e2601b3c1dd87061db9e56020838c51168951908152a1614ed5565b9950915091506060883d821161509c575b8161504b60609383613e57565b8101031261509857907fc1f41e029acf5127d111625602160c4cee3e1a4d38e691e50544d1f7c68b77be929161508089614e0a565b60208a01519984015190999394509190614f12614ea2565b8380fd5b3d915061503e565b50505051903d90823e3d90fd5b50606082015160808301519095509350614f9c565b7fdcbc1c05240f31ff3ad067ef1ee35ce4997762752e3a095284754544f4c709d79192936151ef6001600160a01b03926001600160801b03908161510d888284511661485f565b16815261515d826020830192816151278c8287511661485f565b168452615136828c1688614d67565b51166001600160801b03166fffffffffffffffffffffffffffffffff19601b541617601b55565b51816fffffffffffffffffffffffffffffffff19601b549260801b16911617601b55604051907f23b872dd00000000000000000000000000000000000000000000000000000000602083015233602483015230604483015286166064820152606481526151c981613e3b565b7f000000000000000000000000853d955acef822db058eb8505911ed77f175b99e615c33565b604080516001600160801b03958616815295909416602086015216923392819081015b0390a3565b9091926152546001600160a01b039283871696873303615502575b5061523b61423c565b6001600160801b03928391828085511691511690614682565b1691808616928381106154c05750806152708782855116614682565b1682526020916152b8828483019261528b8b83865116614682565b93828516905251166001600160801b03166fffffffffffffffffffffffffffffffff19601b541617601b55565b816fffffffffffffffffffffffffffffffff19601b549260801b16911617601b55861691871561545657876000526008825260409283600020548181106153ed576151c9846153b67ffbde797d201c681b91056529119e0b02407c7bb96a4a2c75c01fc9667232c8db9998979560008e7fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef6153e89a896153c49a7fa9059cbb000000000000000000000000000000000000000000000000000000009a85875260088452038c86205580600a5403600a558b51908152a386519485938401528860248401602090939291936001600160a01b0360408201951681520152565b03601f198101835282613e57565b516001600160801b0395861681529590941660208601521692339281906040820190565b0390a4565b60848486519062461bcd60e51b82526004820152602260248201527f45524332303a206275726e20616d6f756e7420657863656564732062616c616e60448201527f63650000000000000000000000000000000000000000000000000000000000006064820152fd5b6084826040519062461bcd60e51b82526004820152602160248201527f45524332303a206275726e2066726f6d20746865207a65726f2061646472657360448201527f73000000000000000000000000000000000000000000000000000000000000006064820152fd5b6040517fc5bb6dae00000000000000000000000000000000000000000000000000000000815260048101919091526001600160801b0387166024820152604490fd5b87600052600960205260406000203360005260205260406000205490600019820361552e575b50615232565b61554561554d926001600160801b038a1690614262565b9033906140e2565b3880615528565b90600e549182156155f1576001600160a01b0361556f61423c565b91169161558b600092848452601f602052604084205490615f44565b9283156155e7578252601e60205260408220549283156155df57670de0b6b3a7640000916155b891614817565b0490620186a091828102928184041490151715614fe75750906155da9161482a565b111590565b505091505090565b5050505050600190565b505050600190565b919061560361423c565b6001600160801b03908161562581615619614216565b51168284511690614682565b1691808616928381106154c057508061564b6156418585615e0f565b978285511661485f565b168252615693816020840193615666828a168387511661485f565b94828616905251166001600160801b03166fffffffffffffffffffffffffffffffff19601c541617601c55565b6fffffffffffffffffffffffffffffffff19601c549260801b16911617601c5533600052601f60205260406000206156cc858254613f1f565b90556001600160a01b0382169181308403615718575b50506040519081528360208201527f01348584ec81ac7acd52b7d66d9ade986dd909f3d513881c190fc31c90527efe60403392a3565b6040517fa9059cbb0000000000000000000000000000000000000000000000000000000060208201526001600160a01b0390921660248301526044820152615767906151c981606481016153b6565b38816156e2565b9160207fa32435755c235de2976ed44a75a2f85cb01faf0c894f639fe0c32bb9455fea8f916001600160a01b038091169485600052601e835260406000206157b7868254613f1f565b90556157c585601d54613f1f565b601d5516923084036157db575b604051908152a3565b61584b6040517f23b872dd00000000000000000000000000000000000000000000000000000000848201528560248201523060448201528260648201526064815261582581613e3b565b7f000000000000000000000000ac3e018457b222d93114458476f3e3416abbe38f615c33565b6157d2565b6001600160a01b038093169283600052601e6020526040600020615875838254614262565b905561588382601d54614262565b601d55821691813084036158c2575b50506040519081527fbc290bb45104f73cf92115c9603987c3f8fd30c182a13603d8cffa49b5f5995260203392a4565b6040517fa9059cbb0000000000000000000000000000000000000000000000000000000060208201526001600160a01b03909216602483015260448201526159119061582581606481016153b6565b3881615892565b93907f9dc1449a0ff0c152e18e8289d865b47acc6e1b76b1ecb239c13d6ee22a9206a792916001600160801b03948561595484828a5116614682565b16875260208701866159698682845116614682565b1681526159c0876001600160a01b03809516998a600052601f6020526040600020615997838a168254614262565b905551166001600160801b03166fffffffffffffffffffffffffffffffff19601c541617601c55565b51866fffffffffffffffffffffffffffffffff19601c549260801b16911617601c551693308503615a10575b50604080516001600160801b03928316815292909116602083015281908101615212565b615a5e90604051907f23b872dd00000000000000000000000000000000000000000000000000000000602083015286602483015230604483015283166064820152606481526151c981613e3b565b386159ec565b7f7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff8111615a8e5790565b608460405162461bcd60e51b815260206004820152602860248201527f53616665436173743a2076616c756520646f65736e27742066697420696e206160448201527f6e20696e743235360000000000000000000000000000000000000000000000006064820152fd5b8051156114735760200190565b80518210156114735760209160051b010190565b602090818184031261000e5780519067ffffffffffffffff821161000e57019180601f8401121561000e578251615b4f81613e7a565b93615b5d6040519586613e57565b818552838086019260051b82010192831161000e578301905b828210615b84575050505090565b81518152908301908301615b76565b91908110156114735760051b0190565b356001600160a01b038116810361000e5790565b6001600160a01b03600154163303615bcb57565b60046040517f1c0be90a000000000000000000000000000000000000000000000000000000008152fd5b6001600160a01b03600054163303615c0957565b60046040517ff5c49e64000000000000000000000000000000000000000000000000000000008152fd5b6001600160a01b031690604051615c4981613e09565b6020928382527f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564848301526000808486829651910182855af13d15615d75573d9167ffffffffffffffff8311615d615790615cc493929160405192615cb788601f19601f8401160185613e57565b83523d868885013e615d7f565b90815180615cd3575b50505050565b82849181010312614b5a575081615cea9101614e0a565b15615cf757808080615ccd565b6084906040519062461bcd60e51b82526004820152602a60248201527f5361666545524332303a204552433230206f7065726174696f6e20646964206e60448201527f6f742073756363656564000000000000000000000000000000000000000000006064820152fd5b602485634e487b7160e01b81526041600452fd5b90615cc492916060915b91929015615de05750815115615d93575090565b3b15615d9c5790565b606460405162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e74726163740000006044820152fd5b825190915015615df35750805190602001fd5b610a4e9060405191829162461bcd60e51b835260048301613c10565b91906001600160801b038084511615600014615e2a57509150565b615e63906020850190615e5a81615e51615e478286511688614817565b828a51169061482a565b97511687614817565b9151169061482a565b10615e6a57565b90600181018091116114285790565b6001600160801b038082511615600014615e9257505090565b615e5a614814938260208501511690614817565b90916001600160801b038083511615600014615ec25750505090565b602083959492930190615ee5615edb8284511685614817565b828851169061482a565b9584615ef6575b50505050615e6a57565b615f0993945081615e5a91511687614817565b1038808080615eec565b60208101906001600160801b03908183511615600014615f335750505090565b6148149382615e5a92511690614817565b919060208301926001600160801b038085511615600014615f66575090925050565b9081615e5a81615e51615e47615e639686511688614817565b909160208201916001600160801b038084511615600014615fa1575050505090565b615ee5615edb8284989795969851168561481756fea164736f6c6343000811000a
Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)
000000000000000000000000000000000000000000000000000000000000006000000000000000000000000000000000000000000000000000000000000001a000000000000000000000000000000000000000000000000000000000000002200000000000000000000000000000000000000000000000000000000000000120000000000000000000000000853d955acef822db058eb8505911ed77f175b99e000000000000000000000000ac3e018457b222d93114458476f3e3416abbe38f000000000000000000000000d2f0fa7f2e6a60eecf4b78c5b6d81002b9789f2c000000000000000000000000000000000000000000000000000000000000138800000000000000000000000018500cb1f2fe7a40ebda393383a0b8548a31f2610000000000000000000000000000000000000000000000000000000235ef7f6800000000000000000000000000000000000000000000000000000000000124f8000000000000000000000000000000000000000000000000000000000000271000000000000000000000000000000000000000000000000000000000000007d00000000000000000000000000000000000000000000000000000000000000060000000000000000000000000fd3065c629ee890fd74f43b802c2fea4b7279b8c000000000000000000000000168200cf227d4543302686124ac28ae0eaf2ca0b0000000000000000000000008412ebf45bac1b340bbe8f318b928c466c4e39ca0000000000000000000000000000000000000000000000000000000000000100000000000000000000000000000000000000000000000000000000000000006000000000000000000000000000000000000000000000000000000000000000c000000000000000000000000000000000000000000000000000000000000000120000000000000000000000000000000000000000000000000000000000000037467261786c656e6420496e7465726573742042656172696e67204652415820285374616b6564204672617820457468657229202d203133000000000000000000000000000000000000000000000000000000000000000000000000000000001166465241582873667278455448292d3133000000000000000000000000000000
-----Decoded View---------------
Arg [0] : _configData (bytes): 0x000000000000000000000000853d955acef822db058eb8505911ed77f175b99e000000000000000000000000ac3e018457b222d93114458476f3e3416abbe38f000000000000000000000000d2f0fa7f2e6a60eecf4b78c5b6d81002b9789f2c000000000000000000000000000000000000000000000000000000000000138800000000000000000000000018500cb1f2fe7a40ebda393383a0b8548a31f2610000000000000000000000000000000000000000000000000000000235ef7f6800000000000000000000000000000000000000000000000000000000000124f8000000000000000000000000000000000000000000000000000000000000271000000000000000000000000000000000000000000000000000000000000007d0
Arg [1] : _immutables (bytes): 0x000000000000000000000000fd3065c629ee890fd74f43b802c2fea4b7279b8c000000000000000000000000168200cf227d4543302686124ac28ae0eaf2ca0b0000000000000000000000008412ebf45bac1b340bbe8f318b928c466c4e39ca
Arg [2] : _customConfigData (bytes): 0x000000000000000000000000000000000000000000000000000000000000006000000000000000000000000000000000000000000000000000000000000000c000000000000000000000000000000000000000000000000000000000000000120000000000000000000000000000000000000000000000000000000000000037467261786c656e6420496e7465726573742042656172696e67204652415820285374616b6564204672617820457468657229202d203133000000000000000000000000000000000000000000000000000000000000000000000000000000001166465241582873667278455448292d3133000000000000000000000000000000
-----Encoded View---------------
26 Constructor Arguments found :
Arg [0] : 0000000000000000000000000000000000000000000000000000000000000060
Arg [1] : 00000000000000000000000000000000000000000000000000000000000001a0
Arg [2] : 0000000000000000000000000000000000000000000000000000000000000220
Arg [3] : 0000000000000000000000000000000000000000000000000000000000000120
Arg [4] : 000000000000000000000000853d955acef822db058eb8505911ed77f175b99e
Arg [5] : 000000000000000000000000ac3e018457b222d93114458476f3e3416abbe38f
Arg [6] : 000000000000000000000000d2f0fa7f2e6a60eecf4b78c5b6d81002b9789f2c
Arg [7] : 0000000000000000000000000000000000000000000000000000000000001388
Arg [8] : 00000000000000000000000018500cb1f2fe7a40ebda393383a0b8548a31f261
Arg [9] : 0000000000000000000000000000000000000000000000000000000235ef7f68
Arg [10] : 00000000000000000000000000000000000000000000000000000000000124f8
Arg [11] : 0000000000000000000000000000000000000000000000000000000000002710
Arg [12] : 00000000000000000000000000000000000000000000000000000000000007d0
Arg [13] : 0000000000000000000000000000000000000000000000000000000000000060
Arg [14] : 000000000000000000000000fd3065c629ee890fd74f43b802c2fea4b7279b8c
Arg [15] : 000000000000000000000000168200cf227d4543302686124ac28ae0eaf2ca0b
Arg [16] : 0000000000000000000000008412ebf45bac1b340bbe8f318b928c466c4e39ca
Arg [17] : 0000000000000000000000000000000000000000000000000000000000000100
Arg [18] : 0000000000000000000000000000000000000000000000000000000000000060
Arg [19] : 00000000000000000000000000000000000000000000000000000000000000c0
Arg [20] : 0000000000000000000000000000000000000000000000000000000000000012
Arg [21] : 0000000000000000000000000000000000000000000000000000000000000037
Arg [22] : 467261786c656e6420496e7465726573742042656172696e6720465241582028
Arg [23] : 5374616b6564204672617820457468657229202d203133000000000000000000
Arg [24] : 0000000000000000000000000000000000000000000000000000000000000011
Arg [25] : 66465241582873667278455448292d3133000000000000000000000000000000
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 30 Chains
Chain | Token | Portfolio % | Price | Amount | Value |
---|
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.