Contract Overview
Balance:
0 Ether
EtherValue:
$0.00
More Info
Txn Hash |
Method
|
Block
|
From
|
To
|
Value | ||||
---|---|---|---|---|---|---|---|---|---|
0xd7315e8418a664a8021b822654380fa3a53245a1639a4b69436f48dccf349d32 | Exchange | 16121381 | 53 days 6 hrs ago | 0xc3be098f9594e57a3e71f485a53d990fe3961fe5 | IN | 0xbd871de345b2408f48c1b249a1dac7e0d7d4f8f9 | 0 Ether | 0.00316329 | |
0xa0fbf77f85d4168da6a71703e20a192d9d8b00bb2ac1fe83a91f1e85270774d9 | Exchange | 15939092 | 78 days 17 hrs ago | 0x3e909f4eba38ddc71aa72e1ba3be7edd312818c2 | IN | 0xbd871de345b2408f48c1b249a1dac7e0d7d4f8f9 | 0 Ether | 0.00588403 | |
0xaf511fa2b80a050cc963f8c1ea60c4f6672053499d1caab58a1a9e9b0ad75632 | Exchange | 15878696 | 87 days 3 hrs ago | IN | 0xbd871de345b2408f48c1b249a1dac7e0d7d4f8f9 | 0 Ether | 0.00393559 | ||
0x7ba7990a4bd032d1081fbbd9c0b016636d2c6144a0279c01664380c8cfe80682 | 0x61026060 | 15839982 | 92 days 13 hrs ago | Gearbox Protocol: Deployer | IN | Create: CurveV1Adapter3Assets | 0 Ether | 0.08611965 |
[ Download CSV Export ]
View more zero value Internal Transactions in Advanced View mode
Contract Name:
CurveV1Adapter3Assets
Compiler Version
v0.8.10+commit.fc410830
Optimization Enabled:
Yes with 1000000 runs
Other Settings:
default evmVersion
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: GPL-2.0-or-later // Gearbox Protocol. Generalized leverage for DeFi protocols // (c) Gearbox Holdings, 2022 pragma solidity ^0.8.10; // LIBRARIES import { CurveV1AdapterBase } from "./CurveV1_Base.sol"; // INTERFACES import { N_COINS, ICurvePool3Assets } from "../../integrations/curve/ICurvePool_3.sol"; import { IAdapter, AdapterType } from "@gearbox-protocol/core-v2/contracts/interfaces/adapters/IAdapter.sol"; /// @title CurveV1Adapter3Assets adapter /// @dev Implements logic for interacting with a Curve pool with 3 assets contract CurveV1Adapter3Assets is CurveV1AdapterBase, ICurvePool3Assets { AdapterType public constant override _gearboxAdapterType = AdapterType.CURVE_V1_3ASSETS; /// @dev Constructor /// @param _creditManager Address of the Credit manager /// @param _curvePool Address of the target contract Curve pool /// @param _lp_token Address of the pool's LP token /// @param _metapoolBase The base pool if this pool is a metapool, otherwise 0x0 constructor( address _creditManager, address _curvePool, address _lp_token, address _metapoolBase ) CurveV1AdapterBase( _creditManager, _curvePool, _lp_token, _metapoolBase, N_COINS ) {} /// @dev Sends an order to add liquidity to a Curve pool /// @param amounts Amounts of tokens to add /// @notice 'min_mint_amount' is ignored since the calldata is routed directly to the target /// @notice Internal implementation details in CurveV1Base function add_liquidity(uint256[N_COINS] calldata amounts, uint256) external nonReentrant { _add_liquidity(amounts[0] > 1, amounts[1] > 1, amounts[2] > 1, false); // F:[ACV1_3-4] } /// @dev Sends an order to remove liquidity from a Curve pool /// @notice '_amount' and 'min_amounts' are ignored since the calldata is routed directly to the target /// @notice Internal implementation details in CurveV1Base function remove_liquidity(uint256, uint256[N_COINS] calldata) external virtual nonReentrant { _remove_liquidity(); // F:[ACV1_3-5] } /// @dev Sends an order to remove liquidity from a Curve pool in exact token amounts /// @param amounts Amounts of coins to withdraw /// @notice `max_burn_amount` is ignored since the calldata is routed directly to the target /// @notice Internal implementation details in CurveV1Base function remove_liquidity_imbalance( uint256[N_COINS] calldata amounts, uint256 ) external virtual override nonReentrant { _remove_liquidity_imbalance( amounts[0] > 1, amounts[1] > 1, amounts[2] > 1, false ); // F:[ACV1_3-6] } /// @dev Calculates the amount of LP token minted or burned based on added/removed coin amounts /// @param _amounts Amounts of coins to be added or removed from the pool /// @param _is_deposit Whether the tokens are added or removed function calc_token_amount( uint256[N_COINS] calldata _amounts, bool _is_deposit ) external view returns (uint256) { return ICurvePool3Assets(targetContract).calc_token_amount( _amounts, _is_deposit ); } /// @dev Calculates the time-weighted average of initial and final balances /// @param _first_balances Initial cumulative balances /// @param _last_balances Final cumulative balances /// @param _time_elapsed Amount of time between initial and final balances function get_twap_balances( uint256[N_COINS] calldata _first_balances, uint256[N_COINS] calldata _last_balances, uint256 _time_elapsed ) external view returns (uint256[N_COINS] memory) { return ICurvePool3Assets(targetContract).get_twap_balances( _first_balances, _last_balances, _time_elapsed ); } /// @dev Returns the current balances of coins in the pool function get_balances() external view returns (uint256[N_COINS] memory) { return ICurvePool3Assets(targetContract).get_balances(); } /// @dev Returns the balances of coins in the pool from the last block where it was updated function get_previous_balances() external view returns (uint256[N_COINS] memory) { return ICurvePool3Assets(targetContract).get_previous_balances(); } /// @dev Returns cumulative balances for TWAP calculation function get_price_cumulative_last() external view returns (uint256[N_COINS] memory) { return ICurvePool3Assets(targetContract).get_price_cumulative_last(); } }
// SPDX-License-Identifier: UNLICENSED pragma solidity ^0.8.10; import { ICurvePool } from "./ICurvePool.sol"; uint256 constant N_COINS = 3; /// @title ICurvePool3Assets /// @dev Extends original pool contract with liquidity functions interface ICurvePool3Assets is ICurvePool { function add_liquidity( uint256[N_COINS] memory amounts, uint256 min_mint_amount ) external; function remove_liquidity( uint256 _amount, uint256[N_COINS] memory min_amounts ) external; function remove_liquidity_imbalance( uint256[N_COINS] memory amounts, uint256 max_burn_amount ) external; function calc_token_amount( uint256[N_COINS] calldata _amounts, bool _is_deposit ) external view returns (uint256); function get_twap_balances( uint256[N_COINS] calldata _first_balances, uint256[N_COINS] calldata _last_balances, uint256 _time_elapsed ) external view returns (uint256[N_COINS] memory); function get_balances() external view returns (uint256[N_COINS] memory); function get_previous_balances() external view returns (uint256[N_COINS] memory); function get_price_cumulative_last() external view returns (uint256[N_COINS] memory); }
// SPDX-License-Identifier: MIT // Gearbox Protocol. Generalized leverage for DeFi protocols // (c) Gearbox Holdings, 2022 pragma solidity ^0.8.10; import { ICreditManagerV2 } from "../ICreditManagerV2.sol"; enum AdapterType { ABSTRACT, UNISWAP_V2_ROUTER, UNISWAP_V3_ROUTER, CURVE_V1_EXCHANGE_ONLY, YEARN_V2, CURVE_V1_2ASSETS, CURVE_V1_3ASSETS, CURVE_V1_4ASSETS, CURVE_V1_STECRV_POOL, CURVE_V1_WRAPPER, CONVEX_V1_BASE_REWARD_POOL, CONVEX_V1_BOOSTER, CONVEX_V1_CLAIM_ZAP, LIDO_V1, UNIVERSAL, LIDO_WSTETH_V1 } interface IAdapterExceptions { /// @dev Thrown when the adapter attempts to use a token /// that is not recognized as collateral in the connected /// Credit Manager error TokenIsNotInAllowedList(address); } interface IAdapter is IAdapterExceptions { /// @dev Returns the Credit Manager connected to the adapter function creditManager() external view returns (ICreditManagerV2); /// @dev Returns the Credit Facade connected to the adapter's Credit Manager function creditFacade() external view returns (address); /// @dev Returns the address of the contract the adapter is interacting with function targetContract() external view returns (address); /// @dev Returns the adapter type function _gearboxAdapterType() external pure returns (AdapterType); /// @dev Returns the adapter version function _gearboxAdapterVersion() external pure returns (uint16); }
// SPDX-License-Identifier: GPL-2.0-or-later // Gearbox Protocol. Generalized leverage for DeFi protocols // (c) Gearbox Holdings, 2022 pragma solidity ^0.8.10; import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol"; import { ReentrancyGuard } from "@openzeppelin/contracts/security/ReentrancyGuard.sol"; import { SafeCast } from "@openzeppelin/contracts/utils/math/SafeCast.sol"; import { AbstractAdapter } from "@gearbox-protocol/core-v2/contracts/adapters/AbstractAdapter.sol"; import { ICurvePool2Assets } from "../../integrations/curve/ICurvePool_2.sol"; import { ICurvePool3Assets } from "../../integrations/curve/ICurvePool_3.sol"; import { ICurvePool4Assets } from "../../integrations/curve/ICurvePool_4.sol"; import { ICurveV1Adapter } from "../../interfaces/curve/ICurveV1Adapter.sol"; import { IAdapter, AdapterType } from "@gearbox-protocol/core-v2/contracts/interfaces/adapters/IAdapter.sol"; import { ICurvePool } from "../../integrations/curve/ICurvePool.sol"; import { RAY } from "@gearbox-protocol/core-v2/contracts/libraries/Constants.sol"; // EXCEPTIONS import { ZeroAddressException, NotImplementedException } from "@gearbox-protocol/core-v2/contracts/interfaces/IErrors.sol"; uint256 constant ZERO = 0; /// @title CurveV1Base adapter /// @dev Implements common logic for interacting with all Curve pools, regardless of N_COINS contract CurveV1AdapterBase is AbstractAdapter, ICurveV1Adapter, ReentrancyGuard { using SafeCast for uint256; using SafeCast for int256; // LP token, it could be named differently in some Curve Pools, // so we set the same value to cover all possible cases // coins /// @dev Token in the pool under index 0 address public immutable token0; /// @dev Token in the pool under index 1 address public immutable token1; /// @dev Token in the pool under index 2 address public immutable token2; /// @dev Token in the pool under index 3 address public immutable token3; // underlying coins /// @dev Underlying in the pool under index 0 address public immutable underlying0; /// @dev Underlying in the pool under index 1 address public immutable underlying1; /// @dev Underlying in the pool under index 2 address public immutable underlying2; /// @dev Underlying in the pool under index 3 address public immutable underlying3; /// @dev The pool LP token address public immutable override token; /// @dev The pool LP token /// @notice The LP token can be named differently in different Curve pools, /// so 2 getters are needed for backward compatibility address public immutable override lp_token; /// @dev Address of the base pool (for metapools only) address public immutable override metapoolBase; /// @dev Number of coins in the pool uint256 public immutable nCoins; uint16 public constant _gearboxAdapterVersion = 2; function _gearboxAdapterType() external pure virtual override returns (AdapterType) { return AdapterType.CURVE_V1_EXCHANGE_ONLY; } /// @dev Constructor /// @param _creditManager Address of the Credit manager /// @param _curvePool Address of the target contract Curve pool /// @param _lp_token Address of the pool's LP token /// @param _metapoolBase The base pool if this pool is a metapool, otherwise 0x0 constructor( address _creditManager, address _curvePool, address _lp_token, address _metapoolBase, uint256 _nCoins ) AbstractAdapter(_creditManager, _curvePool) { if (_lp_token == address(0)) revert ZeroAddressException(); // F:[ACV1-1] if (creditManager.tokenMasksMap(_lp_token) == 0) revert TokenIsNotInAllowedList(_lp_token); // F:[ACV1-1] token = _lp_token; // F:[ACV1-2] lp_token = _lp_token; // F:[ACV1-2] metapoolBase = _metapoolBase; // F:[ACV1-2] nCoins = _nCoins; // F:[ACV1-2] address[4] memory tokens; for (uint256 i = 0; i < nCoins; ) { address currentCoin; try ICurvePool(targetContract).coins(i) returns ( address tokenAddress ) { currentCoin = tokenAddress; } catch { try ICurvePool(targetContract).coins(i.toInt256().toInt128()) returns (address tokenAddress) { currentCoin = tokenAddress; } catch {} } if (currentCoin == address(0)) revert ZeroAddressException(); if (creditManager.tokenMasksMap(currentCoin) == 0) revert TokenIsNotInAllowedList(currentCoin); tokens[i] = currentCoin; unchecked { ++i; } } token0 = tokens[0]; // F:[ACV1-2] token1 = tokens[1]; // F:[ACV1-2] token2 = tokens[2]; // F:[ACV1-2] token3 = tokens[3]; // F:[ACV1-2] tokens = [address(0), address(0), address(0), address(0)]; for (uint256 i = 0; i < 4; ) { address currentCoin; if (metapoolBase != address(0)) { if (i == 0) { currentCoin = token0; } else { try ICurvePool(metapoolBase).coins(i - 1) returns ( address tokenAddress ) { currentCoin = tokenAddress; } catch {} } } else { try ICurvePool(targetContract).underlying_coins(i) returns ( address tokenAddress ) { currentCoin = tokenAddress; } catch { try ICurvePool(targetContract).underlying_coins( i.toInt256().toInt128() ) returns (address tokenAddress) { currentCoin = tokenAddress; } catch {} } } if ( currentCoin != address(0) && creditManager.tokenMasksMap(currentCoin) == 0 ) { revert TokenIsNotInAllowedList(currentCoin); // F:[ACV1-1] } tokens[i] = currentCoin; unchecked { ++i; } } underlying0 = tokens[0]; // F:[ACV1-2] underlying1 = tokens[1]; // F:[ACV1-2] underlying2 = tokens[2]; // F:[ACV1-2] underlying3 = tokens[3]; // F:[ACV1-2] } /// @dev Sends an order to exchange one asset to another /// @param i Index for the coin sent /// @param j Index for the coin received /// @notice Fast check parameters: /// Input token: Coin under index i /// Output token: Coin under index j /// Input token is allowed, since the target does a transferFrom for coin i /// The input token does not need to be disabled, because this does not spend the entire /// balance, generally function exchange( int128 i, int128 j, uint256, uint256 ) external override nonReentrant { address tokenIn = _get_token(i); // F:[ACV1-4,ACV1S-3] address tokenOut = _get_token(j); // F:[ACV1-4,ACV1S-3] _executeMaxAllowanceFastCheck(tokenIn, tokenOut, msg.data, true, false); // F:[ACV1-4,ACV1S-3] } /// @dev Sends an order to exchange the entire balance of one asset to another /// @param i Index for the coin sent /// @param j Index for the coin received /// @param rateMinRAY Minimum exchange rate between coins i and j /// @notice Fast check parameters: /// Input token: Coin under index i /// Output token: Coin under index j /// Input token is allowed, since the target does a transferFrom for coin i /// The input token does need to be disabled, because this spends the entire balance /// @notice Calls `exchange` under the hood, passing current balance - 1 as the amount function exchange_all( int128 i, int128 j, uint256 rateMinRAY ) external override nonReentrant { address creditAccount = creditManager.getCreditAccountOrRevert( msg.sender ); //F:[ACV1-3] address tokenIn = _get_token(i); //F:[ACV1-5] address tokenOut = _get_token(j); // F:[ACV1-5] uint256 dx = IERC20(tokenIn).balanceOf(creditAccount); //F:[ACV1-5] if (dx > 1) { unchecked { dx--; } uint256 min_dy = (dx * rateMinRAY) / RAY; //F:[ACV1-5] _executeMaxAllowanceFastCheck( creditAccount, tokenIn, tokenOut, abi.encodeWithSelector( ICurvePool.exchange.selector, i, j, dx, min_dy ), true, true ); //F:[ACV1-5] } } /// @dev Sends an order to exchange one underlying asset to another /// @param i Index for the underlying coin sent /// @param j Index for the underlying coin received /// @notice Fast check parameters: /// Input token: Underlying coin under index i /// Output token: Underlying coin under index j /// Input token is allowed, since the target does a transferFrom for underlying i /// The input token does not need to be disabled, because this does not spend the entire /// balance, generally function exchange_underlying( int128 i, int128 j, uint256, uint256 ) external override nonReentrant { address tokenIn = _get_underlying(i); // F:[ACV1-6] address tokenOut = _get_underlying(j); // F:[ACV1-6] _executeMaxAllowanceFastCheck(tokenIn, tokenOut, msg.data, true, false); // F:[ACV1-6] } /// @dev Sends an order to exchange the entire balance of one underlying asset to another /// @param i Index for the underlying coin sent /// @param j Index for the underlying coin received /// @param rateMinRAY Minimum exchange rate between underlyings i and j /// @notice Fast check parameters: /// Input token: Underlying coin under index i /// Output token: Underlying coin under index j /// Input token is allowed, since the target does a transferFrom for underlying i /// The input token does need to be disabled, because this spends the entire balance /// @notice Calls `exchange_underlying` under the hood, passing current balance - 1 as the amount function exchange_all_underlying( int128 i, int128 j, uint256 rateMinRAY ) external nonReentrant { address creditAccount = creditManager.getCreditAccountOrRevert( msg.sender ); //F:[ACV1-3] address tokenIn = _get_underlying(i); //F:[ACV1-7] address tokenOut = _get_underlying(j); // F:[ACV1-7] uint256 dx = IERC20(tokenIn).balanceOf(creditAccount); //F:[ACV1-7] if (dx > 1) { unchecked { dx--; } uint256 min_dy = (dx * rateMinRAY) / RAY; //F:[ACV1-7] _executeMaxAllowanceFastCheck( creditAccount, tokenIn, tokenOut, abi.encodeWithSelector( ICurvePool.exchange_underlying.selector, i, j, dx, min_dy ), true, true ); //F:[ACV1-7] } } /// @dev Internal implementation for `add_liquidity` /// - Sets allowances for tokens that are added /// - Enables the pool LP token on the CA /// - Executes the order with a full check (this is required since >2 tokens are involved) /// - Resets allowance for tokens that are added function _add_liquidity( bool t0Approve, bool t1Approve, bool t2Approve, bool t3Approve ) internal { address creditAccount = creditManager.getCreditAccountOrRevert( msg.sender ); // F:[ACV1_2-3, ACV1_3-3, ACV1_3-4] _approve_coins(t0Approve, t1Approve, t2Approve, t3Approve); // F:[ACV1_2-4, ACV1_3-4, ACV1_4-4] _enableToken(creditAccount, address(lp_token)); // F:[ACV1_2-4, ACV1_3-4, ACV1_4-4] _execute(msg.data); // F:[ACV1_2-4, ACV1_3-4, ACV1_4-4] _approve_coins(t0Approve, t1Approve, t2Approve, t3Approve); /// F:[ACV1_2-4, ACV1_3-4, ACV1_4-4] _fullCheck(creditAccount); } /// @dev Sends an order to add liquidity with only 1 input asset /// - Picks a selector based on the number of coins /// - Makes a fast check call to target /// @param amount Amount of asset to deposit /// @param i Index of the asset to deposit /// @param minAmount Minimal number of LP tokens to receive /// @notice Fast check parameters: /// Input token: Pool asset under index i /// Output token: Pool LP token /// Input token is allowed, since the target does a transferFrom for the deposited asset /// The input token does not need to be disabled, because this does not spend the entire /// balance, generally /// @notice Calls `add_liquidity` under the hood with only one amount being non-zero function add_liquidity_one_coin( uint256 amount, int128 i, uint256 minAmount ) external override nonReentrant { address tokenIn = _get_token(i); address creditAccount = creditManager.getCreditAccountOrRevert( msg.sender ); // F:[ACV1-8A] _executeMaxAllowanceFastCheck( creditAccount, tokenIn, lp_token, _getAddLiquidityCallData(i, amount, minAmount), true, false ); // F:[ACV1-8A] } /// @dev Sends an order to add liquidity with only 1 input asset, using the entire balance /// - Computes the amount of asset to deposit (balance - 1) /// - Picks a selector based on the number of coins /// - Makes a fast check call to target /// @param i Index of the asset to deposit /// @param rateMinRAY Minimal exchange rate between the deposited asset and the LP token /// @notice Fast check parameters: /// Input token: Pool asset under index i /// Output token: Pool LP token /// Input token is allowed, since the target does a transferFrom for the deposited asset /// The input token does need to be disabled, because this spends the entire balance /// @notice Calls `add_liquidity` under the hood with only one amount being non-zero function add_all_liquidity_one_coin(int128 i, uint256 rateMinRAY) external override nonReentrant { address tokenIn = _get_token(i); address creditAccount = creditManager.getCreditAccountOrRevert( msg.sender ); // F:[ACV1-8] uint256 amount = IERC20(tokenIn).balanceOf(creditAccount); /// F:[ACV1-8] if (amount > 1) { unchecked { amount--; // F:[ACV1-8] } uint256 minAmount = (amount * rateMinRAY) / RAY; // F:[ACV1-8] _executeMaxAllowanceFastCheck( creditAccount, tokenIn, lp_token, _getAddLiquidityCallData(i, amount, minAmount), true, true ); // F:[ACV1-8] } } function _getAddLiquidityCallData( int128 i, uint256 amount, uint256 minAmount ) internal view returns (bytes memory) { if (nCoins == 2) { return i == 0 ? abi.encodeWithSelector( ICurvePool2Assets.add_liquidity.selector, amount, ZERO, minAmount ) : abi.encodeWithSelector( ICurvePool2Assets.add_liquidity.selector, ZERO, amount, minAmount ); // F:[ACV1-8] } if (nCoins == 3) { return i == 0 ? abi.encodeWithSelector( ICurvePool3Assets.add_liquidity.selector, amount, ZERO, ZERO, minAmount ) : i == 1 ? abi.encodeWithSelector( ICurvePool3Assets.add_liquidity.selector, ZERO, amount, ZERO, minAmount ) : abi.encodeWithSelector( ICurvePool3Assets.add_liquidity.selector, ZERO, ZERO, amount, minAmount ); // F:[ACV1-8] } if (nCoins == 4) { return i == 0 ? abi.encodeWithSelector( ICurvePool4Assets.add_liquidity.selector, amount, ZERO, ZERO, ZERO, minAmount ) : i == 1 ? abi.encodeWithSelector( ICurvePool4Assets.add_liquidity.selector, ZERO, amount, ZERO, ZERO, minAmount ) : i == 2 ? abi.encodeWithSelector( ICurvePool4Assets.add_liquidity.selector, ZERO, ZERO, amount, ZERO, minAmount ) : abi.encodeWithSelector( ICurvePool4Assets.add_liquidity.selector, ZERO, ZERO, ZERO, amount, minAmount ); // F:[ACV1-8] } revert("Incorrect nCoins"); } /// @dev Returns the amount of lp token received when adding a single coin to the pool /// @param amount Amount of coin to be deposited /// @param i Index of a coin to be deposited function calc_add_one_coin(uint256 amount, int128 i) external view returns (uint256) { if (nCoins == 2) { return i == 0 ? ICurvePool2Assets(targetContract).calc_token_amount( [amount, 0], true ) : ICurvePool2Assets(targetContract).calc_token_amount( [0, amount], true ); } else if (nCoins == 3) { return i == 0 ? ICurvePool3Assets(targetContract).calc_token_amount( [amount, 0, 0], true ) : i == 1 ? ICurvePool3Assets(targetContract).calc_token_amount( [0, amount, 0], true ) : ICurvePool3Assets(targetContract).calc_token_amount( [0, 0, amount], true ); } else if (nCoins == 4) { return i == 0 ? ICurvePool4Assets(targetContract).calc_token_amount( [amount, 0, 0, 0], true ) : i == 1 ? ICurvePool4Assets(targetContract).calc_token_amount( [0, amount, 0, 0], true ) : i == 2 ? ICurvePool4Assets(targetContract).calc_token_amount( [0, 0, amount, 0], true ) : ICurvePool4Assets(targetContract).calc_token_amount( [0, 0, 0, amount], true ); } else { revert("Incorrect nCoins"); } } /// @dev Internal implementation for `remove_liquidity` /// - Enables all of the pool tokens (since remove_liquidity will always /// return non-zero amounts for all tokens) /// - Executes the order with a full check (this is required since >2 tokens are involved) /// @notice The LP token does not need to be approved since the pool burns it function _remove_liquidity() internal { address creditAccount = creditManager.getCreditAccountOrRevert( msg.sender ); // F:[ACV1_2-3, ACV1_3-3, ACV1_3-4] _enableToken(creditAccount, token0); // F:[ACV1_2-5, ACV1_3-5, ACV1_4-5] _enableToken(creditAccount, token1); // F:[ACV1_2-5, ACV1_3-5, ACV1_4-5] if (token2 != address(0)) { _enableToken(creditAccount, token2); // F:[ACV1_3-5, ACV1_4-5] if (token3 != address(0)) { _enableToken(creditAccount, token3); // F:[ACV1_4-5] } } _execute(msg.data); _fullCheck(creditAccount); //F:[ACV1_2-5, ACV1_3-5, ACV1_4-5] } /// @dev Sends an order to remove liquidity from a pool in a single asset /// - Makes a fast check call to target, with passed calldata /// @param i Index of the asset to withdraw /// @notice `_token_amount` and `min_amount` are ignored since the calldata is routed directly to the target /// @notice Fast check parameters: /// Input token: Pool LP token /// Output token: Coin under index i /// Input token is not approved, since the pool directly burns the LP token /// The input token does not need to be disabled, because this does not spend the entire /// balance, generally function remove_liquidity_one_coin( uint256, // _token_amount, int128 i, uint256 // min_amount ) external virtual override nonReentrant { address tokenOut = _get_token(i); // F:[ACV1-9] _remove_liquidity_one_coin(tokenOut); // F:[ACV1-9] } /// @dev Internal implementation for `remove_liquidity_one_coin` operations /// - Makes a fast check call to target, with passed calldata /// @param tokenOut The coin received from the pool /// @notice Fast check parameters: /// Input token: Pool LP token /// Output token: Coin under index i /// Input token is not approved, since the pool directly burns the LP token /// The input token does not need to be disabled, because this does not spend the entire /// balance, generally function _remove_liquidity_one_coin(address tokenOut) internal { address creditAccount = creditManager.getCreditAccountOrRevert( msg.sender ); // F:[ACV1-9] _executeMaxAllowanceFastCheck( creditAccount, lp_token, tokenOut, msg.data, false, false ); // F:[ACV1-9] } /// @dev Sends an order to remove all liquidity from the pool in a single asset /// @param i Index of the asset to withdraw /// @param minRateRAY Minimal exchange rate between the LP token and the received token function remove_all_liquidity_one_coin(int128 i, uint256 minRateRAY) external virtual override nonReentrant { address tokenOut = _get_token(i); // F:[ACV1-4] _remove_all_liquidity_one_coin(i, tokenOut, minRateRAY); // F:[ACV1-10] } /// @dev Internal implementation for `remove_all_liquidity_one_coin` operations /// - Computes the amount of LP token to burn (balance - 1) /// - Makes a max allowance fast check call to target /// @param i Index of the coin received from the pool /// @param tokenOut The coin received from the pool /// @param rateMinRAY The minimal exchange rate between the LP token and received token /// @notice Fast check parameters: /// Input token: Pool LP token /// Output token: Coin under index i /// Input token is not approved, since the pool directly burns the LP token /// The input token does need to be disabled, because this spends the entire balance function _remove_all_liquidity_one_coin( int128 i, address tokenOut, uint256 rateMinRAY ) internal { address creditAccount = creditManager.getCreditAccountOrRevert( msg.sender ); //F:[ACV1-3] uint256 amount = IERC20(lp_token).balanceOf(creditAccount); // F:[ACV1-10] if (amount > 1) { unchecked { amount--; // F:[ACV1-10] } _executeMaxAllowanceFastCheck( creditAccount, lp_token, tokenOut, abi.encodeWithSelector( ICurvePool.remove_liquidity_one_coin.selector, amount, i, (amount * rateMinRAY) / RAY ), false, true ); // F:[ACV1-10] } } /// @dev Internal implementation for `remove_liquidity_imbalance` /// - Enables tokens with a non-zero amount withdrawn /// - Executes the order with a full check (this is required since >2 tokens are involved) /// @notice The LP token does not need to be approved since the pool burns it function _remove_liquidity_imbalance( bool t0Enable, bool t1Enable, bool t2Enable, bool t3Enable ) internal { address creditAccount = creditManager.getCreditAccountOrRevert( msg.sender ); // F:[ACV1_2-3, ACV1_3-3, ACV1_3-4] if (t0Enable) { _enableToken(creditAccount, token0); // F:[ACV1_2-6, ACV1_3-6, ACV1_4-6] } if (t1Enable) { _enableToken(creditAccount, token1); // F:[ACV1_2-6, ACV1_3-6, ACV1_4-6] } if (t2Enable) { _enableToken(creditAccount, token2); // F:[ACV1_3-6, ACV1_4-6] } if (t3Enable) { _enableToken(creditAccount, token3); // F:[ACV1_4-6] } _execute(msg.data); _fullCheck(creditAccount); // F:[ACV1_2-6, ACV1_3-6, ACV1_4-6] } /// @dev Returns the amount of coin j received by swapping dx of coin i /// @param i Index of the input coin /// @param j Index of the output coin /// @param dx Amount of coin i to be swapped in function get_dy( int128 i, int128 j, uint256 dx ) external view override returns (uint256) { return ICurvePool(targetContract).get_dy(i, j, dx); // F:[ACV1-11] } /// @dev Returns the amount of underlying j received by swapping dx of underlying i /// @param i Index of the input underlying /// @param j Index of the output underlying /// @param dx Amount of underlying i to be swapped in function get_dy_underlying( int128 i, int128 j, uint256 dx ) external view override returns (uint256) { return ICurvePool(targetContract).get_dy_underlying(i, j, dx); // F:[ACV1-11] } /// @dev Returns the price of the pool's LP token function get_virtual_price() external view override returns (uint256) { return ICurvePool(targetContract).get_virtual_price(); // F:[ACV1-13] } /// @dev Returns the address of the coin with index i /// @param i The index of a coin to retrieve the address for function coins(uint256 i) external view override returns (address) { return _get_token(i.toInt256().toInt128()); // F:[ACV1-11] } /// @dev Returns the address of the coin with index i /// @param i The index of a coin to retrieve the address for (type int128) /// @notice Since `i` is int128 in some older Curve pools, /// the function is provided for compatibility function coins(int128 i) external view override returns (address) { return _get_token(i); // F:[ACV1-11] } /// @dev Returns the address of the underlying with index i /// @param i The index of a coin to retrieve the address for function underlying_coins(uint256 i) public view override returns (address) { return _get_underlying(i.toInt256().toInt128()); // F:[ACV1-11] } /// @dev Returns the address of the underlying with index i /// @param i The index of a coin to retrieve the address for (type int128) /// @notice Since `i` is int128 in some older Curve pools, /// the function is provided for compatibility function underlying_coins(int128 i) external view override returns (address) { return _get_underlying(i); // F:[ACV1-11] } /// @dev Returns the pool's balance of the coin with index i /// @param i The index of the coin to retrieve the balance for /// @notice Since `i` is int128 in some older Curve pools, /// the function first tries to call a uin256 variant, /// and then then int128 variant if that fails function balances(uint256 i) public view override returns (uint256) { try ICurvePool(targetContract).balances(i) returns (uint256 balance) { return balance; // F:[ACV1-11] } catch { return ICurvePool(targetContract).balances(i.toInt256().toInt128()); // F:[ACV1-11] } } /// @dev Returns the pool's balance of the coin with index i /// @param i The index of the coin to retrieve the balance for /// @notice Since `i` is int128 in some older Curve pools, /// the function first tries to call a int128 variant, /// and then then uint256 variant if that fails function balances(int128 i) public view override returns (uint256) { return balances(uint256(uint128(i))); } /// @dev Return the token i's address gas-efficiently function _get_token(int128 i) internal view returns (address addr) { if (i == 0) addr = token0; // F:[ACV1-14] else if (i == 1) addr = token1; // F:[ACV1-14] else if (i == 2) addr = token2; // F:[ACV1-14] else if (i == 3) addr = token3; // F:[ACV1-14] if (addr == address(0)) revert IncorrectIndexException(); // F:[ACV1-13] } /// @dev Return the underlying i's address gas-efficiently function _get_underlying(int128 i) internal view returns (address addr) { if (i == 0) addr = underlying0; // F:[ACV1-14] else if (i == 1) addr = underlying1; // F:[ACV1-14] else if (i == 2) addr = underlying2; // F:[ACV1-14] else if (i == 3) addr = underlying3; // F:[ACV1-14] if (addr == address(0)) revert IncorrectIndexException(); // F:[ACV1-13] } /// @dev Gives max approval for a coin to target contract function _approve_coins( bool t0Enable, bool t1Enable, bool t2Enable, bool t3Enable ) internal { if (t0Enable) { _approveToken(token0, type(uint256).max); // F:[ACV1_2-4, ACV1_3-4, ACV1_4-4] } if (t1Enable) { _approveToken(token1, type(uint256).max); // F:[ACV1_2-4, ACV1_3-4, ACV1_4-4] } if (t2Enable) { _approveToken(token2, type(uint256).max); // F:[ACV1_3-4, ACV1_4-4] } if (t3Enable) { _approveToken(token3, type(uint256).max); // F:[ACV1_4-4] } } function _enableToken(address creditAccount, address tokenToEnable) internal { creditManager.checkAndEnableToken(creditAccount, tokenToEnable); } /// @dev Returns the current amplification parameter function A() external view returns (uint256) { return ICurvePool(targetContract).A(); } /// @dev Returns the current amplification parameter scaled function A_precise() external view returns (uint256) { return ICurvePool(targetContract).A_precise(); } /// @dev Returns the amount of coin withdrawn when using remove_liquidity_one_coin /// @param _burn_amount Amount of LP token to be burnt /// @param i Index of a coin to receive function calc_withdraw_one_coin(uint256 _burn_amount, int128 i) external view returns (uint256) { return ICurvePool(targetContract).calc_withdraw_one_coin(_burn_amount, i); } /// @dev Returns the amount of coin that belongs to the admin /// @param i Index of a coin function admin_balances(uint256 i) external view returns (uint256) { return ICurvePool(targetContract).admin_balances(i); } /// @dev Returns the admin of a pool function admin() external view returns (address) { return ICurvePool(targetContract).admin(); } /// @dev Returns the fee amount function fee() external view returns (uint256) { return ICurvePool(targetContract).fee(); } /// @dev Returns the percentage of the fee claimed by the admin function admin_fee() external view returns (uint256) { return ICurvePool(targetContract).admin_fee(); } /// @dev Returns the block in which the pool was last interacted with function block_timestamp_last() external view returns (uint256) { return ICurvePool(targetContract).block_timestamp_last(); } /// @dev Returns the initial A during ramping function initial_A() external view returns (uint256) { return ICurvePool(targetContract).initial_A(); } /// @dev Returns the final A during ramping function future_A() external view returns (uint256) { return ICurvePool(targetContract).future_A(); } /// @dev Returns the ramping start time function initial_A_time() external view returns (uint256) { return ICurvePool(targetContract).initial_A_time(); } /// @dev Returns the ramping end time function future_A_time() external view returns (uint256) { return ICurvePool(targetContract).future_A_time(); } /// @dev Returns the name of the LP token /// @notice Only for pools that implement ERC20 function name() external view returns (string memory) { return ICurvePool(targetContract).name(); } /// @dev Returns the symbol of the LP token /// @notice Only for pools that implement ERC20 function symbol() external view returns (string memory) { return ICurvePool(targetContract).symbol(); } /// @dev Returns the decimals of the LP token /// @notice Only for pools that implement ERC20 function decimals() external view returns (uint256) { return ICurvePool(targetContract).decimals(); } /// @dev Returns the LP token balance of address /// @param account Address to compute the balance for /// @notice Only for pools that implement ERC20 function balanceOf(address account) external view returns (uint256) { return ICurvePool(targetContract).balanceOf(account); } /// @dev Returns the LP token allowance of address /// @param owner Address from which the token is allowed /// @param spender Address to which the token is allowed /// @notice Only for pools that implement ERC20 function allowance(address owner, address spender) external view returns (uint256) { return ICurvePool(targetContract).allowance(owner, spender); } /// @dev Returns the total supply of the LP token /// @notice Only for pools that implement ERC20 function totalSupply() external view returns (uint256) { return ICurvePool(targetContract).totalSupply(); } }
// SPDX-License-Identifier: UNLICENSED pragma solidity ^0.8.10; interface ICurvePool { function coins(uint256 i) external view returns (address); function underlying_coins(uint256 i) external view returns (address); function balances(uint256 i) external view returns (uint256); function coins(int128) external view returns (address); function underlying_coins(int128) external view returns (address); function balances(int128) external view returns (uint256); function exchange( int128 i, int128 j, uint256 dx, uint256 min_dy ) external; function exchange_underlying( int128 i, int128 j, uint256 dx, uint256 min_dy ) external; function get_dy_underlying( int128 i, int128 j, uint256 dx ) external view returns (uint256); function get_dy( int128 i, int128 j, uint256 dx ) external view returns (uint256); function get_virtual_price() external view returns (uint256); function token() external view returns (address); function remove_liquidity_one_coin( uint256 _token_amount, int128 i, uint256 min_amount ) external; function A() external view returns (uint256); function A_precise() external view returns (uint256); function calc_withdraw_one_coin(uint256 _burn_amount, int128 i) external view returns (uint256); function admin_balances(uint256 i) external view returns (uint256); function admin() external view returns (address); function fee() external view returns (uint256); function admin_fee() external view returns (uint256); function block_timestamp_last() external view returns (uint256); function initial_A() external view returns (uint256); function future_A() external view returns (uint256); function initial_A_time() external view returns (uint256); function future_A_time() external view returns (uint256); // Some pools implement ERC20 function name() external view returns (string memory); function symbol() external view returns (string memory); function decimals() external view returns (uint256); function balanceOf(address) external view returns (uint256); function allowance(address, address) external view returns (uint256); function totalSupply() external view returns (uint256); }
// SPDX-License-Identifier: MIT // Gearbox Protocol. Generalized leverage for DeFi protocols // (c) Gearbox Holdings, 2022 pragma solidity ^0.8.10; import { IPriceOracleV2 } from "./IPriceOracle.sol"; import { IVersion } from "./IVersion.sol"; enum ClosureAction { CLOSE_ACCOUNT, LIQUIDATE_ACCOUNT, LIQUIDATE_EXPIRED_ACCOUNT, LIQUIDATE_PAUSED } interface ICreditManagerV2Events { /// @dev Emits when a call to an external contract is made through the Credit Manager event ExecuteOrder(address indexed borrower, address indexed target); /// @dev Emits when a configurator is upgraded event NewConfigurator(address indexed newConfigurator); } interface ICreditManagerV2Exceptions { /// @dev Thrown if an access-restricted function is called by an address that is not /// the connected Credit Facade, or an allowed adapter error AdaptersOrCreditFacadeOnlyException(); /// @dev Thrown if an access-restricted function is called by an address that is not /// the connected Credit Facade error CreditFacadeOnlyException(); /// @dev Thrown if an access-restricted function is called by an address that is not /// the connected Credit Configurator error CreditConfiguratorOnlyException(); /// @dev Thrown on attempting to open a Credit Account for or transfer a Credit Account /// to the zero address or an address that already owns a Credit Account error ZeroAddressOrUserAlreadyHasAccountException(); /// @dev Thrown on attempting to execute an order to an address that is not an allowed /// target contract error TargetContractNotAllowedException(); /// @dev Thrown on failing a full collateral check after an operation error NotEnoughCollateralException(); /// @dev Thrown on attempting to receive a token that is not a collateral token /// or was forbidden error TokenNotAllowedException(); /// @dev Thrown if an attempt to approve a collateral token to a target contract failed error AllowanceFailedException(); /// @dev Thrown on attempting to perform an action for an address that owns no Credit Account error HasNoOpenedAccountException(); /// @dev Thrown on attempting to add a token that is already in a collateral list error TokenAlreadyAddedException(); /// @dev Thrown on configurator attempting to add more than 256 collateral tokens error TooManyTokensException(); /// @dev Thrown if more than the maximal number of tokens were enabled on a Credit Account, /// and there are not enough unused token to disable error TooManyEnabledTokensException(); /// @dev Thrown when a reentrancy into the contract is attempted error ReentrancyLockException(); } /// @notice All Credit Manager functions are access-restricted and can only be called /// by the Credit Facade or allowed adapters. Users are not allowed to /// interact with the Credit Manager directly interface ICreditManagerV2 is ICreditManagerV2Events, ICreditManagerV2Exceptions, IVersion { // // CREDIT ACCOUNT MANAGEMENT // /// @dev Opens credit account and borrows funds from the pool. /// - Takes Credit Account from the factory; /// - Requests the pool to lend underlying to the Credit Account /// /// @param borrowedAmount Amount to be borrowed by the Credit Account /// @param onBehalfOf The owner of the newly opened Credit Account function openCreditAccount(uint256 borrowedAmount, address onBehalfOf) external returns (address); /// @dev Closes a Credit Account - covers both normal closure and liquidation /// - Checks whether the contract is paused, and, if so, if the payer is an emergency liquidator. /// Only emergency liquidators are able to liquidate account while the CM is paused. /// Emergency liquidations do not pay a liquidator premium or liquidation fees. /// - Calculates payments to various recipients on closure: /// + Computes amountToPool, which is the amount to be sent back to the pool. /// This includes the principal, interest and fees, but can't be more than /// total position value /// + Computes remainingFunds during liquidations - these are leftover funds /// after paying the pool and the liquidator, and are sent to the borrower /// + Computes protocol profit, which includes interest and liquidation fees /// + Computes loss if the totalValue is less than borrow amount + interest /// - Checks the underlying token balance: /// + if it is larger than amountToPool, then the pool is paid fully from funds on the Credit Account /// + else tries to transfer the shortfall from the payer - either the borrower during closure, or liquidator during liquidation /// - Send assets to the "to" address, as long as they are not included into skipTokenMask /// - If convertWETH is true, the function converts WETH into ETH before sending /// - Returns the Credit Account back to factory /// /// @param borrower Borrower address /// @param closureActionType Whether the account is closed, liquidated or liquidated due to expiry /// @param totalValue Portfolio value for liqution, 0 for ordinary closure /// @param payer Address which would be charged if credit account has not enough funds to cover amountToPool /// @param to Address to which the leftover funds will be sent /// @param skipTokenMask Tokenmask contains 1 for tokens which needed to be skipped for sending /// @param convertWETH If true converts WETH to ETH function closeCreditAccount( address borrower, ClosureAction closureActionType, uint256 totalValue, address payer, address to, uint256 skipTokenMask, bool convertWETH ) external returns (uint256 remainingFunds); /// @dev Manages debt size for borrower: /// /// - Increase debt: /// + Increases debt by transferring funds from the pool to the credit account /// + Updates the cumulative index to keep interest the same. Since interest /// is always computed dynamically as borrowedAmount * (cumulativeIndexNew / cumulativeIndexOpen - 1), /// cumulativeIndexOpen needs to be updated, as the borrow amount has changed /// /// - Decrease debt: /// + Repays debt partially + all interest and fees accrued thus far /// + Updates cunulativeIndex to cumulativeIndex now /// /// @param creditAccount Address of the Credit Account to change debt for /// @param amount Amount to increase / decrease the principal by /// @param increase True to increase principal, false to decrease /// @return newBorrowedAmount The new debt principal function manageDebt( address creditAccount, uint256 amount, bool increase ) external returns (uint256 newBorrowedAmount); /// @dev Adds collateral to borrower's credit account /// @param payer Address of the account which will be charged to provide additional collateral /// @param creditAccount Address of the Credit Account /// @param token Collateral token to add /// @param amount Amount to add function addCollateral( address payer, address creditAccount, address token, uint256 amount ) external; /// @dev Transfers Credit Account ownership to another address /// @param from Address of previous owner /// @param to Address of new owner function transferAccountOwnership(address from, address to) external; /// @dev Requests the Credit Account to approve a collateral token to another contract. /// @param borrower Borrower's address /// @param targetContract Spender to change allowance for /// @param token Collateral token to approve /// @param amount New allowance amount function approveCreditAccount( address borrower, address targetContract, address token, uint256 amount ) external; /// @dev Requests a Credit Account to make a low-level call with provided data /// This is the intended pathway for state-changing interactions with 3rd-party protocols /// @param borrower Borrower's address /// @param targetContract Contract to be called /// @param data Data to pass with the call function executeOrder( address borrower, address targetContract, bytes memory data ) external returns (bytes memory); // // COLLATERAL VALIDITY AND ACCOUNT HEALTH CHECKS // /// @dev Enables a token on a Credit Account, including it /// into account health and total value calculations /// @param creditAccount Address of a Credit Account to enable the token for /// @param token Address of the token to be enabled function checkAndEnableToken(address creditAccount, address token) external; /// @dev Optimized health check for individual swap-like operations. /// @notice Fast health check assumes that only two tokens (input and output) /// participate in the operation and computes a % change in weighted value between /// inbound and outbound collateral. The cumulative negative change across several /// swaps in sequence cannot be larger than feeLiquidation (a fee that the /// protocol is ready to waive if needed). Since this records a % change /// between just two tokens, the corresponding % change in TWV will always be smaller, /// which makes this check safe. /// More details at https://dev.gearbox.fi/docs/documentation/risk/fast-collateral-check#fast-check-protection /// @param creditAccount Address of the Credit Account /// @param tokenIn Address of the token spent by the swap /// @param tokenOut Address of the token received from the swap /// @param balanceInBefore Balance of tokenIn before the operation /// @param balanceOutBefore Balance of tokenOut before the operation function fastCollateralCheck( address creditAccount, address tokenIn, address tokenOut, uint256 balanceInBefore, uint256 balanceOutBefore ) external; /// @dev Performs a full health check on an account, summing up /// value of all enabled collateral tokens /// @param creditAccount Address of the Credit Account to check function fullCollateralCheck(address creditAccount) external; /// @dev Checks that the number of enabled tokens on a Credit Account /// does not violate the maximal enabled token limit and tries /// to disable unused tokens if it does /// @param creditAccount Account to check enabled tokens for function checkAndOptimizeEnabledTokens(address creditAccount) external; /// @dev Disables a token on a credit account /// @notice Usually called by adapters to disable spent tokens during a multicall, /// but can also be called separately from the Credit Facade to remove /// unwanted tokens /// @return True if token mask was change otherwise False function disableToken(address creditAccount, address token) external returns (bool); // // GETTERS // /// @dev Returns the address of a borrower's Credit Account, or reverts if there is none. /// @param borrower Borrower's address function getCreditAccountOrRevert(address borrower) external view returns (address); /// @dev Computes amounts that must be sent to various addresses before closing an account /// @param totalValue Credit Accounts total value in underlying /// @param closureActionType Type of account closure /// * CLOSE_ACCOUNT: The account is healthy and is closed normally /// * LIQUIDATE_ACCOUNT: The account is unhealthy and is being liquidated to avoid bad debt /// * LIQUIDATE_EXPIRED_ACCOUNT: The account has expired and is being liquidated (lowered liquidation premium) /// * LIQUIDATE_PAUSED: The account is liquidated while the system is paused due to emergency (no liquidation premium) /// @param borrowedAmount Credit Account's debt principal /// @param borrowedAmountWithInterest Credit Account's debt principal + interest /// @return amountToPool Amount of underlying to be sent to the pool /// @return remainingFunds Amount of underlying to be sent to the borrower (only applicable to liquidations) /// @return profit Protocol's profit from fees (if any) /// @return loss Protocol's loss from bad debt (if any) function calcClosePayments( uint256 totalValue, ClosureAction closureActionType, uint256 borrowedAmount, uint256 borrowedAmountWithInterest ) external view returns ( uint256 amountToPool, uint256 remainingFunds, uint256 profit, uint256 loss ); /// @dev Calculates the debt accrued by a Credit Account /// @param creditAccount Address of the Credit Account /// @return borrowedAmount The debt principal /// @return borrowedAmountWithInterest The debt principal + accrued interest /// @return borrowedAmountWithInterestAndFees The debt principal + accrued interest and protocol fees function calcCreditAccountAccruedInterest(address creditAccount) external view returns ( uint256 borrowedAmount, uint256 borrowedAmountWithInterest, uint256 borrowedAmountWithInterestAndFees ); /// @dev Maps Credit Accounts to bit masks encoding their enabled token sets /// Only enabled tokens are counted as collateral for the Credit Account /// @notice An enabled token mask encodes an enabled token by setting /// the bit at the position equal to token's index to 1 function enabledTokensMap(address creditAccount) external view returns (uint256); /// @dev Maps the Credit Account to its current percentage drop across all swaps since /// the last full check, in RAY format function cumulativeDropAtFastCheckRAY(address creditAccount) external view returns (uint256); /// @dev Returns the collateral token at requested index and its liquidation threshold /// @param id The index of token to return function collateralTokens(uint256 id) external view returns (address token, uint16 liquidationThreshold); /// @dev Returns the collateral token with requested mask and its liquidationThreshold /// @param tokenMask Token mask corresponding to the token function collateralTokensByMask(uint256 tokenMask) external view returns (address token, uint16 liquidationThreshold); /// @dev Total number of known collateral tokens. function collateralTokensCount() external view returns (uint256); /// @dev Returns the mask for the provided token /// @param token Token to returns the mask for function tokenMasksMap(address token) external view returns (uint256); /// @dev Bit mask encoding a set of forbidden tokens function forbiddenTokenMask() external view returns (uint256); /// @dev Maps allowed adapters to their respective target contracts. function adapterToContract(address adapter) external view returns (address); /// @dev Maps 3rd party contracts to their respective adapters function contractToAdapter(address targetContract) external view returns (address); /// @dev Address of the underlying asset function underlying() external view returns (address); /// @dev Address of the connected pool function pool() external view returns (address); /// @dev Address of the connected pool /// @notice [DEPRECATED]: use pool() instead. function poolService() external view returns (address); /// @dev A map from borrower addresses to Credit Account addresses function creditAccounts(address borrower) external view returns (address); /// @dev Address of the connected Credit Configurator function creditConfigurator() external view returns (address); /// @dev Address of WETH function wethAddress() external view returns (address); /// @dev Returns the liquidation threshold for the provided token /// @param token Token to retrieve the LT for function liquidationThresholds(address token) external view returns (uint16); /// @dev The maximal number of enabled tokens on a single Credit Account function maxAllowedEnabledTokenLength() external view returns (uint8); /// @dev Maps addresses to their status as emergency liquidator. /// @notice Emergency liquidators are trusted addresses /// that are able to liquidate positions while the contracts are paused, /// e.g. when there is a risk of bad debt while an exploit is being patched. /// In the interest of fairness, emergency liquidators do not receive a premium /// And are compensated by the Gearbox DAO separately. function canLiquidateWhilePaused(address) external view returns (bool); /// @dev Returns the fee parameters of the Credit Manager /// @return feeInterest Percentage of interest taken by the protocol as profit /// @return feeLiquidation Percentage of account value taken by the protocol as profit /// during unhealthy account liquidations /// @return liquidationDiscount Multiplier that reduces the effective totalValue during unhealthy account liquidations, /// allowing the liquidator to take the unaccounted for remainder as premium. Equal to (1 - liquidationPremium) /// @return feeLiquidationExpired Percentage of account value taken by the protocol as profit /// during expired account liquidations /// @return liquidationDiscountExpired Multiplier that reduces the effective totalValue during expired account liquidations, /// allowing the liquidator to take the unaccounted for remainder as premium. Equal to (1 - liquidationPremiumExpired) function fees() external view returns ( uint16 feeInterest, uint16 feeLiquidation, uint16 liquidationDiscount, uint16 feeLiquidationExpired, uint16 liquidationDiscountExpired ); /// @dev Address of the connected Credit Facade function creditFacade() external view returns (address); /// @dev Address of the connected Price Oracle function priceOracle() external view returns (IPriceOracleV2); /// @dev Address of the universal adapter function universalAdapter() external view returns (address); /// @dev Contract's version function version() external view returns (uint256); /// @dev Paused() state function checkEmergencyPausable(address caller, bool state) external returns (bool); }
// SPDX-License-Identifier: MIT // Gearbox Protocol. Generalized leverage for DeFi protocols // (c) Gearbox Holdings, 2022 pragma solidity ^0.8.10; import { IVersion } from "./IVersion.sol"; interface IPriceOracleV2Events { /// @dev Emits when a new price feed is added event NewPriceFeed(address indexed token, address indexed priceFeed); } interface IPriceOracleV2Exceptions { /// @dev Thrown if a price feed returns 0 error ZeroPriceException(); /// @dev Thrown if the last recorded result was not updated in the last round error ChainPriceStaleException(); /// @dev Thrown on attempting to get a result for a token that does not have a price feed error PriceOracleNotExistsException(); } /// @title Price oracle interface interface IPriceOracleV2 is IPriceOracleV2Events, IPriceOracleV2Exceptions, IVersion { /// @dev Converts a quantity of an asset to USD (decimals = 8). /// @param amount Amount to convert /// @param token Address of the token to be converted function convertToUSD(uint256 amount, address token) external view returns (uint256); /// @dev Converts a quantity of USD (decimals = 8) to an equivalent amount of an asset /// @param amount Amount to convert /// @param token Address of the token converted to function convertFromUSD(uint256 amount, address token) external view returns (uint256); /// @dev Converts one asset into another /// /// @param amount Amount to convert /// @param tokenFrom Address of the token to convert from /// @param tokenTo Address of the token to convert to function convert( uint256 amount, address tokenFrom, address tokenTo ) external view returns (uint256); /// @dev Returns collateral values for two tokens, required for a fast check /// @param amountFrom Amount of the outbound token /// @param tokenFrom Address of the outbound token /// @param amountTo Amount of the inbound token /// @param tokenTo Address of the inbound token /// @return collateralFrom Value of the outbound token amount in USD /// @return collateralTo Value of the inbound token amount in USD function fastCheck( uint256 amountFrom, address tokenFrom, uint256 amountTo, address tokenTo ) external view returns (uint256 collateralFrom, uint256 collateralTo); /// @dev Returns token's price in USD (8 decimals) /// @param token The token to compute the price for function getPrice(address token) external view returns (uint256); /// @dev Returns the price feed address for the passed token /// @param token Token to get the price feed for function priceFeeds(address token) external view returns (address priceFeed); /// @dev Returns the price feed for the passed token, /// with additional parameters /// @param token Token to get the price feed for function priceFeedsWithFlags(address token) external view returns ( address priceFeed, bool skipCheck, uint256 decimals ); } interface IPriceOracleV2Ext is IPriceOracleV2 { /// @dev Sets a price feed if it doesn't exist, or updates an existing one /// @param token Address of the token to set the price feed for /// @param priceFeed Address of a USD price feed adhering to Chainlink's interface function addPriceFeed(address token, address priceFeed) external; }
// SPDX-License-Identifier: MIT // Gearbox Protocol. Generalized leverage for DeFi protocols // (c) Gearbox Holdings, 2022 pragma solidity ^0.8.10; /// @title IVersion /// @dev Declares a version function which returns the contract's version interface IVersion { /// @dev Returns contract version function version() external view returns (uint256); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (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() { // On the first call to nonReentrant, _notEntered will be true require(_status != _ENTERED, "ReentrancyGuard: reentrant call"); // Any calls to nonReentrant after this point will fail _status = _ENTERED; _; // 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 v4.4.1 (token/ERC20/IERC20.sol) pragma solidity ^0.8.0; /** * @dev Interface of the ERC20 standard as defined in the EIP. */ interface IERC20 { /** * @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 `recipient`. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transfer(address recipient, 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 `sender` to `recipient` 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 sender, address recipient, uint256 amount ) external returns (bool); /** * @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); }
// SPDX-License-Identifier: GPL-2.0-or-later // Gearbox Protocol. Generalized leverage for DeFi protocols // (c) Gearbox Holdings, 2022 pragma solidity ^0.8.10; // Denominations uint256 constant WAD = 1e18; uint256 constant RAY = 1e27; // 25% of type(uint256).max uint256 constant ALLOWANCE_THRESHOLD = type(uint96).max >> 3; // FEE = 50% uint16 constant DEFAULT_FEE_INTEREST = 50_00; // 50% // LIQUIDATION_FEE 1.5% uint16 constant DEFAULT_FEE_LIQUIDATION = 1_50; // 1.5% // LIQUIDATION PREMIUM 4% uint16 constant DEFAULT_LIQUIDATION_PREMIUM = 4_00; // 4% // LIQUIDATION_FEE_EXPIRED 2% uint16 constant DEFAULT_FEE_LIQUIDATION_EXPIRED = 1_00; // 2% // LIQUIDATION PREMIUM EXPIRED 2% uint16 constant DEFAULT_LIQUIDATION_PREMIUM_EXPIRED = 2_00; // 2% // DEFAULT PROPORTION OF MAX BORROWED PER BLOCK TO MAX BORROWED PER ACCOUNT uint16 constant DEFAULT_LIMIT_PER_BLOCK_MULTIPLIER = 2; // Seconds in a year uint256 constant SECONDS_PER_YEAR = 365 days; uint256 constant SECONDS_PER_ONE_AND_HALF_YEAR = (SECONDS_PER_YEAR * 3) / 2; // OPERATIONS // Leverage decimals - 100 is equal to 2x leverage (100% * collateral amount + 100% * borrowed amount) uint8 constant LEVERAGE_DECIMALS = 100; // Maximum withdraw fee for pool in PERCENTAGE_FACTOR format uint8 constant MAX_WITHDRAW_FEE = 100; uint256 constant EXACT_INPUT = 1; uint256 constant EXACT_OUTPUT = 2; address constant UNIVERSAL_CONTRACT = 0xCcCCccccCCCCcCCCCCCcCcCccCcCCCcCcccccccC;
// SPDX-License-Identifier: GPL-2.0-or-later // Gearbox Protocol. Generalized leverage for DeFi protocols // (c) Gearbox Holdings, 2022 pragma solidity ^0.8.10; import { Address } from "@openzeppelin/contracts/utils/Address.sol"; import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol"; import { ICreditManagerV2 } from "../interfaces/ICreditManagerV2.sol"; import { IAdapter } from "../interfaces/adapters/IAdapter.sol"; import { ZeroAddressException } from "../interfaces/IErrors.sol"; abstract contract AbstractAdapter is IAdapter { using Address for address; ICreditManagerV2 public immutable override creditManager; address public immutable override creditFacade; address public immutable override targetContract; constructor(address _creditManager, address _targetContract) { if (_creditManager == address(0) || _targetContract == address(0)) revert ZeroAddressException(); // F:[AA-2] creditManager = ICreditManagerV2(_creditManager); // F:[AA-1] creditFacade = ICreditManagerV2(_creditManager).creditFacade(); // F:[AA-1] targetContract = _targetContract; // F:[AA-1] } /// @dev Approves a token from the Credit Account to the target contract /// @param token Token to be approved /// @param amount Amount to be approved function _approveToken(address token, uint256 amount) internal { creditManager.approveCreditAccount( msg.sender, targetContract, token, amount ); } /// @dev Sends CallData to call the target contract from the Credit Account /// @param callData Data to be sent to the target contract function _execute(bytes memory callData) internal returns (bytes memory result) { result = creditManager.executeOrder( msg.sender, targetContract, callData ); } /// @dev Calls a target contract with maximal allowance and performs a fast check after /// @param creditAccount A credit account from which a call is made /// @param tokenIn The token that the interaction is expected to spend /// @param tokenOut The token that the interaction is expected to produce /// @param callData Data to call targetContract with /// @param allowTokenIn Whether the input token must be approved beforehand /// @param disableTokenIn Whether the input token should be disable afterwards (for interaction that spend the entire balance) /// @notice Must only be used for highly secure and immutable protocols, such as Uniswap & Curve function _executeMaxAllowanceFastCheck( address creditAccount, address tokenIn, address tokenOut, bytes memory callData, bool allowTokenIn, bool disableTokenIn ) internal returns (bytes memory result) { uint256 balanceInBefore; uint256 balanceOutBefore; if (msg.sender != creditFacade) { balanceInBefore = IERC20(tokenIn).balanceOf(creditAccount); // F:[AA-4A] balanceOutBefore = IERC20(tokenOut).balanceOf(creditAccount); // F:[AA-4A] } if (allowTokenIn) { _approveToken(tokenIn, type(uint256).max); } result = creditManager.executeOrder( msg.sender, targetContract, callData ); if (allowTokenIn) { _approveToken(tokenIn, type(uint256).max); } _fastCheck( creditAccount, tokenIn, tokenOut, balanceInBefore, balanceOutBefore, disableTokenIn ); } /// @dev Wrapper for _executeMaxAllowanceFastCheck that computes the Credit Account on the spot /// See params and other details above function _executeMaxAllowanceFastCheck( address tokenIn, address tokenOut, bytes memory callData, bool allowTokenIn, bool disableTokenIn ) internal returns (bytes memory result) { address creditAccount = creditManager.getCreditAccountOrRevert( msg.sender ); // F:[AA-3] result = _executeMaxAllowanceFastCheck( creditAccount, tokenIn, tokenOut, callData, allowTokenIn, disableTokenIn ); } /// @dev Calls a target contract with maximal allowance, then sets allowance to 1 and performs a fast check /// @param creditAccount A credit account from which a call is made /// @param tokenIn The token that the interaction is expected to spend /// @param tokenOut The token that the interaction is expected to produce /// @param callData Data to call targetContract with /// @param allowTokenIn Whether the input token must be approved beforehand /// @param disableTokenIn Whether the input token should be disable afterwards (for interaction that spend the entire balance) function _safeExecuteFastCheck( address creditAccount, address tokenIn, address tokenOut, bytes memory callData, bool allowTokenIn, bool disableTokenIn ) internal returns (bytes memory result) { uint256 balanceInBefore; uint256 balanceOutBefore; if (msg.sender != creditFacade) { balanceInBefore = IERC20(tokenIn).balanceOf(creditAccount); balanceOutBefore = IERC20(tokenOut).balanceOf(creditAccount); // F:[AA-4A] } if (allowTokenIn) { _approveToken(tokenIn, type(uint256).max); } result = creditManager.executeOrder( msg.sender, targetContract, callData ); if (allowTokenIn) { _approveToken(tokenIn, 1); } _fastCheck( creditAccount, tokenIn, tokenOut, balanceInBefore, balanceOutBefore, disableTokenIn ); } /// @dev Wrapper for _safeExecuteFastCheck that computes the Credit Account on the spot /// See params and other details above function _safeExecuteFastCheck( address tokenIn, address tokenOut, bytes memory callData, bool allowTokenIn, bool disableTokenIn ) internal returns (bytes memory result) { address creditAccount = creditManager.getCreditAccountOrRevert( msg.sender ); result = _safeExecuteFastCheck( creditAccount, tokenIn, tokenOut, callData, allowTokenIn, disableTokenIn ); } // // HEALTH CHECK FUNCTIONS // /// @dev Performs a fast check during ordinary adapter call, or skips /// it for multicalls (since a full collateral check is always performed after a multicall) /// @param creditAccount Credit Account for which the fast check is performed /// @param tokenIn Token that is spent by the operation /// @param tokenOut Token that is received as a result of operation /// @param balanceInBefore Balance of tokenIn before the operation /// @param balanceOutBefore Balance of tokenOut before the operation /// @param disableTokenIn Whether tokenIn needs to be disabled (required for multicalls, where the fast check is skipped) function _fastCheck( address creditAccount, address tokenIn, address tokenOut, uint256 balanceInBefore, uint256 balanceOutBefore, bool disableTokenIn ) private { if (msg.sender != creditFacade) { creditManager.fastCollateralCheck( creditAccount, tokenIn, tokenOut, balanceInBefore, balanceOutBefore ); } else { if (disableTokenIn) creditManager.disableToken(creditAccount, tokenIn); creditManager.checkAndEnableToken(creditAccount, tokenOut); } } /// @dev Performs a full collateral check during ordinary adapter call, or skips /// it for multicalls (since a full collateral check is always performed after a multicall) /// @param creditAccount Credit Account for which the full check is performed function _fullCheck(address creditAccount) internal { if (msg.sender != creditFacade) { creditManager.fullCollateralCheck(creditAccount); } } /// @dev Performs a enabled token optimization on account or skips /// it for multicalls (since a full collateral check is always performed after a multicall, /// and includes enabled token optimization by default) /// @param creditAccount Credit Account for which the full check is performed /// @notice Used when new tokens are added on an account but no tokens are subtracted /// (e.g., claiming rewards) function _checkAndOptimizeEnabledTokens(address creditAccount) internal { if (msg.sender != creditFacade) { creditManager.checkAndOptimizeEnabledTokens(creditAccount); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/math/SafeCast.sol) 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 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 */ 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 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 */ 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 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 */ 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 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 */ 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 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 */ 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 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 */ 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. */ 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. */ function toUint256(int256 value) internal pure returns (uint256) { require(value >= 0, "SafeCast: value must be positive"); return uint256(value); } /** * @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) { require(value >= type(int128).min && value <= type(int128).max, "SafeCast: value doesn't fit in 128 bits"); return int128(value); } /** * @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) { require(value >= type(int64).min && value <= type(int64).max, "SafeCast: value doesn't fit in 64 bits"); return int64(value); } /** * @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) { require(value >= type(int32).min && value <= type(int32).max, "SafeCast: value doesn't fit in 32 bits"); return int32(value); } /** * @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) { require(value >= type(int16).min && value <= type(int16).max, "SafeCast: value doesn't fit in 16 bits"); return int16(value); } /** * @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) { require(value >= type(int8).min && value <= type(int8).max, "SafeCast: value doesn't fit in 8 bits"); return int8(value); } /** * @dev Converts an unsigned uint256 into a signed int256. * * Requirements: * * - input must be less than or equal to maxInt256. */ 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: MIT // Gearbox Protocol. Generalized leverage for DeFi protocols // (c) Gearbox Holdings, 2022 pragma solidity ^0.8.10; /// @dev Common contract exceptions /// @dev Thrown on attempting to set an important address to zero address error ZeroAddressException(); /// @dev Thrown on attempting to call a non-implemented function error NotImplementedException(); /// @dev Thrown on attempting to set an EOA as an important contract in the system error AddressIsNotContractException(address); /// @dev Thrown on attempting to use a non-ERC20 contract or an EOA as a token error IncorrectTokenContractException(); /// @dev Thrown on attempting to set a token price feed to an address that is not a /// correct price feed error IncorrectPriceFeedException(); /// @dev Thrown on attempting to call an access restricted function as a non-Configurator error CallerNotConfiguratorException(); /// @dev Thrown on attempting to pause a contract as a non-Pausable admin error CallerNotPausableAdminException(); /// @dev Thrown on attempting to pause a contract as a non-Unpausable admin error CallerNotUnPausableAdminException(); error TokenIsNotAddedToCreditManagerException(address token);
// SPDX-License-Identifier: UNLICENSED pragma solidity ^0.8.10; import { ICurvePool } from "./ICurvePool.sol"; uint256 constant N_COINS = 2; /// @title ICurvePool2Assets /// @dev Extends original pool contract with liquidity functions interface ICurvePool2Assets is ICurvePool { function add_liquidity( uint256[N_COINS] memory amounts, uint256 min_mint_amount ) external; function remove_liquidity( uint256 _amount, uint256[N_COINS] memory min_amounts ) external; function remove_liquidity_imbalance( uint256[N_COINS] calldata amounts, uint256 max_burn_amount ) external; function calc_token_amount( uint256[N_COINS] calldata _amounts, bool _is_deposit ) external view returns (uint256); function get_twap_balances( uint256[N_COINS] calldata _first_balances, uint256[N_COINS] calldata _last_balances, uint256 _time_elapsed ) external view returns (uint256[N_COINS] memory); function get_balances() external view returns (uint256[N_COINS] memory); function get_previous_balances() external view returns (uint256[N_COINS] memory); function get_price_cumulative_last() external view returns (uint256[N_COINS] memory); }
// SPDX-License-Identifier: UNLICENSED pragma solidity ^0.8.10; import { ICurvePool } from "./ICurvePool.sol"; uint256 constant N_COINS = 4; /// @title ICurvePool4Assets /// @dev Extends original pool contract with liquidity functions interface ICurvePool4Assets is ICurvePool { function add_liquidity( uint256[N_COINS] memory amounts, uint256 min_mint_amount ) external; function remove_liquidity( uint256 _amount, uint256[N_COINS] memory min_amounts ) external; function remove_liquidity_imbalance( uint256[N_COINS] memory amounts, uint256 max_burn_amount ) external; function calc_token_amount( uint256[N_COINS] calldata _amounts, bool _is_deposit ) external view returns (uint256); function get_twap_balances( uint256[N_COINS] calldata _first_balances, uint256[N_COINS] calldata _last_balances, uint256 _time_elapsed ) external view returns (uint256[N_COINS] memory); function get_balances() external view returns (uint256[N_COINS] memory); function get_previous_balances() external view returns (uint256[N_COINS] memory); function get_price_cumulative_last() external view returns (uint256[N_COINS] memory); }
// SPDX-License-Identifier: MIT // Gearbox Protocol. Generalized leverage for DeFi protocols // (c) Gearbox Holdings, 2022 pragma solidity ^0.8.10; import { IAdapter } from "@gearbox-protocol/core-v2/contracts/interfaces/adapters/IAdapter.sol"; import { ICurvePool } from "../../integrations/curve/ICurvePool.sol"; interface ICurveV1AdapterExceptions { error IncorrectIndexException(); } interface ICurveV1Adapter is IAdapter, ICurvePool, ICurveV1AdapterExceptions { /// @dev Sends an order to exchange the entire balance of one asset to another /// @param i Index for the coin sent /// @param j Index for the coin received /// @param rateMinRAY Minimum exchange rate between coins i and j function exchange_all( int128 i, int128 j, uint256 rateMinRAY ) external; /// @dev Sends an order to exchange the entire balance of one underlying asset to another /// @param i Index for the underlying coin sent /// @param j Index for the underlying coin received /// @param rateMinRAY Minimum exchange rate between underlyings i and j function exchange_all_underlying( int128 i, int128 j, uint256 rateMinRAY ) external; /// @dev Sends an order to add liquidity with only 1 input asset /// @param amount Amount of asset to deposit /// @param i Index of the asset to deposit /// @param minAmount Minimal number of LP tokens to receive function add_liquidity_one_coin( uint256 amount, int128 i, uint256 minAmount ) external; /// @dev Sends an order to add liquidity with only 1 input asset, using the entire balance /// @param i Index of the asset to deposit /// @param rateMinRAY Minimal exchange rate between the deposited asset and the LP token function add_all_liquidity_one_coin(int128 i, uint256 rateMinRAY) external; /// @dev Sends an order to remove all liquidity from the pool in a single asset /// @param i Index of the asset to withdraw /// @param minRateRAY Minimal exchange rate between the LP token and the received token function remove_all_liquidity_one_coin(int128 i, uint256 minRateRAY) external; // // GETTERS // /// @dev The pool LP token function lp_token() external view returns (address); /// @dev Address of the base pool (for metapools only) function metapoolBase() external view returns (address); /// @dev Number of coins in the pool function nCoins() external view returns (uint256); /// @dev Token in the pool under index 0 function token0() external view returns (address); /// @dev Token in the pool under index 1 function token1() external view returns (address); /// @dev Token in the pool under index 2 function token2() external view returns (address); /// @dev Token in the pool under index 3 function token3() external view returns (address); /// @dev Underlying in the pool under index 0 function underlying0() external view returns (address); /// @dev Underlying in the pool under index 1 function underlying1() external view returns (address); /// @dev Underlying in the pool under index 2 function underlying2() external view returns (address); /// @dev Underlying in the pool under index 3 function underlying3() external view returns (address); /// @dev Returns the amount of lp token received when adding a single coin to the pool /// @param amount Amount of coin to be deposited /// @param i Index of a coin to be deposited function calc_add_one_coin(uint256 amount, int128 i) external view returns (uint256); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/Address.sol) pragma solidity ^0.8.0; /** * @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 * ==== */ function isContract(address account) internal view returns (bool) { // This method relies on extcodesize, which returns 0 for contracts in // construction, since the code is only stored at the end of the // constructor execution. uint256 size; assembly { size := extcodesize(account) } return size > 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 functionCall(target, data, "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"); require(isContract(target), "Address: call to non-contract"); (bool success, bytes memory returndata) = target.call{value: value}(data); return verifyCallResult(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) { require(isContract(target), "Address: static call to non-contract"); (bool success, bytes memory returndata) = target.staticcall(data); return verifyCallResult(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) { require(isContract(target), "Address: delegate call to non-contract"); (bool success, bytes memory returndata) = target.delegatecall(data); return verifyCallResult(success, returndata, errorMessage); } /** * @dev Tool to verifies that a low level call was successful, and revert if it wasn't, either by bubbling the * revert reason 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 { // 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 assembly { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert(errorMessage); } } } }
{ "optimizer": { "enabled": true, "runs": 1000000 }, "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "devdoc", "userdoc", "metadata", "abi" ] } }, "libraries": {} }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"inputs":[{"internalType":"address","name":"_creditManager","type":"address"},{"internalType":"address","name":"_curvePool","type":"address"},{"internalType":"address","name":"_lp_token","type":"address"},{"internalType":"address","name":"_metapoolBase","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"IncorrectIndexException","type":"error"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"TokenIsNotInAllowedList","type":"error"},{"inputs":[],"name":"ZeroAddressException","type":"error"},{"inputs":[],"name":"A","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"A_precise","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"_gearboxAdapterType","outputs":[{"internalType":"enum AdapterType","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"_gearboxAdapterVersion","outputs":[{"internalType":"uint16","name":"","type":"uint16"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"int128","name":"i","type":"int128"},{"internalType":"uint256","name":"rateMinRAY","type":"uint256"}],"name":"add_all_liquidity_one_coin","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256[3]","name":"amounts","type":"uint256[3]"},{"internalType":"uint256","name":"","type":"uint256"}],"name":"add_liquidity","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"int128","name":"i","type":"int128"},{"internalType":"uint256","name":"minAmount","type":"uint256"}],"name":"add_liquidity_one_coin","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"admin","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"i","type":"uint256"}],"name":"admin_balances","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"admin_fee","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","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":"account","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"int128","name":"i","type":"int128"}],"name":"balances","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"i","type":"uint256"}],"name":"balances","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"block_timestamp_last","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"int128","name":"i","type":"int128"}],"name":"calc_add_one_coin","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256[3]","name":"_amounts","type":"uint256[3]"},{"internalType":"bool","name":"_is_deposit","type":"bool"}],"name":"calc_token_amount","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_burn_amount","type":"uint256"},{"internalType":"int128","name":"i","type":"int128"}],"name":"calc_withdraw_one_coin","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"int128","name":"i","type":"int128"}],"name":"coins","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"i","type":"uint256"}],"name":"coins","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"creditFacade","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"creditManager","outputs":[{"internalType":"contract ICreditManagerV2","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"decimals","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"int128","name":"i","type":"int128"},{"internalType":"int128","name":"j","type":"int128"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"name":"exchange","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"int128","name":"i","type":"int128"},{"internalType":"int128","name":"j","type":"int128"},{"internalType":"uint256","name":"rateMinRAY","type":"uint256"}],"name":"exchange_all","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"int128","name":"i","type":"int128"},{"internalType":"int128","name":"j","type":"int128"},{"internalType":"uint256","name":"rateMinRAY","type":"uint256"}],"name":"exchange_all_underlying","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"int128","name":"i","type":"int128"},{"internalType":"int128","name":"j","type":"int128"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"name":"exchange_underlying","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"fee","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"future_A","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"future_A_time","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"get_balances","outputs":[{"internalType":"uint256[3]","name":"","type":"uint256[3]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"int128","name":"i","type":"int128"},{"internalType":"int128","name":"j","type":"int128"},{"internalType":"uint256","name":"dx","type":"uint256"}],"name":"get_dy","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"int128","name":"i","type":"int128"},{"internalType":"int128","name":"j","type":"int128"},{"internalType":"uint256","name":"dx","type":"uint256"}],"name":"get_dy_underlying","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"get_previous_balances","outputs":[{"internalType":"uint256[3]","name":"","type":"uint256[3]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"get_price_cumulative_last","outputs":[{"internalType":"uint256[3]","name":"","type":"uint256[3]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256[3]","name":"_first_balances","type":"uint256[3]"},{"internalType":"uint256[3]","name":"_last_balances","type":"uint256[3]"},{"internalType":"uint256","name":"_time_elapsed","type":"uint256"}],"name":"get_twap_balances","outputs":[{"internalType":"uint256[3]","name":"","type":"uint256[3]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"get_virtual_price","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"initial_A","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"initial_A_time","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"lp_token","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"metapoolBase","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"nCoins","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"int128","name":"i","type":"int128"},{"internalType":"uint256","name":"minRateRAY","type":"uint256"}],"name":"remove_all_liquidity_one_coin","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256[3]","name":"","type":"uint256[3]"}],"name":"remove_liquidity","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256[3]","name":"amounts","type":"uint256[3]"},{"internalType":"uint256","name":"","type":"uint256"}],"name":"remove_liquidity_imbalance","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"int128","name":"i","type":"int128"},{"internalType":"uint256","name":"","type":"uint256"}],"name":"remove_liquidity_one_coin","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"targetContract","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"token","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"token0","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"token1","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"token2","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"token3","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"underlying0","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"underlying1","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"underlying2","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"underlying3","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"int128","name":"i","type":"int128"}],"name":"underlying_coins","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"i","type":"uint256"}],"name":"underlying_coins","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"}]
Contract Creation Code
6102606040523480156200001257600080fd5b5060405162005bcf38038062005bcf83398101604081905262000035916200083f565b83838383600384846001600160a01b03821615806200005b57506001600160a01b038116155b156200007a57604051635919af9760e11b815260040160405180910390fd5b6001600160a01b038216608081905260408051632f7a188160e01b81529051632f7a1881916004808201926020929091908290030181865afa158015620000c5573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190620000eb91906200089c565b6001600160a01b0390811660a05290811660c05260016000558416151590506200012857604051635919af9760e11b815260040160405180910390fd5b608051604051630f67c5bd60e41b81526001600160a01b0385811660048301529091169063f67c5bd090602401602060405180830381865afa15801562000173573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190620001999190620008c1565b620001c757604051634c607af960e11b81526001600160a01b03841660048201526024015b60405180910390fd5b6001600160a01b038084166101e081905261020052821661022052610240819052620001f262000804565b60005b61024051811015620004155760c05160405163c661065760e01b8152600481018390526000916001600160a01b03169063c661065790602401602060405180830381865afa9250505080156200026a575060408051601f3d908101601f1916820190925262000267918101906200089c565b60015b6200031c5760c0516001600160a01b03166323746eb8620002ab6200029a856200071460201b62002c771760201c565b6200078460201b62002d2d1760201c565b6040516001600160e01b031960e084901b168152600f9190910b6004820152602401602060405180830381865afa92505050801562000309575060408051601f3d908101601f1916820190925262000306918101906200089c565b60015b62000314576200031f565b90506200031f565b90505b6001600160a01b0381166200034757604051635919af9760e11b815260040160405180910390fd5b608051604051630f67c5bd60e41b81526001600160a01b0383811660048301529091169063f67c5bd090602401602060405180830381865afa15801562000392573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190620003b89190620008c1565b620003e257604051634c607af960e11b81526001600160a01b0382166004820152602401620001be565b80838360048110620003f857620003f8620008db565b6001600160a01b03909216602092909202015250600101620001f5565b5080516001600160a01b0390811660e0526020808301518216610100526040808401518316610120526060938401519092166101405281516080810183526000808252918101829052918201819052918101829052905b6004811015620006d157610220516000906001600160a01b0316156200052957816200049c575060e051620005ec565b610220516001600160a01b031663c6610657620004bb600185620008f1565b6040518263ffffffff1660e01b8152600401620004da91815260200190565b602060405180830381865afa92505050801562000516575060408051601f3d908101601f1916820190925262000513918101906200089c565b60015b6200052157620005ec565b9050620005ec565b60c051604051630b9947eb60e41b8152600481018490526001600160a01b039091169063b9947eb090602401602060405180830381865afa92505050801562000591575060408051601f3d908101601f191682019092526200058e918101906200089c565b60015b620005e95760c0516001600160a01b031663b739953e620005c16200029a856200071460201b62002c771760201c565b6040516001600160e01b031960e084901b168152600f9190910b6004820152602401620004da565b90505b6001600160a01b03811615801590620006735750608051604051630f67c5bd60e41b81526001600160a01b0383811660048301529091169063f67c5bd090602401602060405180830381865afa1580156200064b573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190620006719190620008c1565b155b156200069e57604051634c607af960e11b81526001600160a01b0382166004820152602401620001be565b80838360048110620006b457620006b4620008db565b6001600160a01b039092166020929092020152506001016200046c565b5080516001600160a01b03908116610160526020820151811661018052604082015181166101a052606090910151166101c0525062000917975050505050505050565b60006001600160ff1b03821115620007805760405162461bcd60e51b815260206004820152602860248201527f53616665436173743a2076616c756520646f65736e27742066697420696e2061604482015267371034b73a191a9b60c11b6064820152608401620001be565b5090565b600060016001607f1b03198212801590620007a6575060016001607f1b038213155b620007805760405162461bcd60e51b815260206004820152602760248201527f53616665436173743a2076616c756520646f65736e27742066697420696e20316044820152663238206269747360c81b6064820152608401620001be565b60405180608001604052806004906020820280368337509192915050565b80516001600160a01b03811681146200083a57600080fd5b919050565b600080600080608085870312156200085657600080fd5b620008618562000822565b9350620008716020860162000822565b9250620008816040860162000822565b9150620008916060860162000822565b905092959194509250565b600060208284031215620008af57600080fd5b620008ba8262000822565b9392505050565b600060208284031215620008d457600080fd5b5051919050565b634e487b7160e01b600052603260045260246000fd5b6000828210156200091257634e487b7160e01b600052601160045260246000fd5b500390565b60805160a05160c05160e05161010051610120516101405161016051610180516101a0516101c0516101e051610200516102205161024051614f6c62000c63600039600081816107a6015281816110600152818161122f0152818161141501528181613a9f01528181613c560152613d70015260006105fc0152600081816106930152818161261a01528181612a5101528181612fcb01528181613521015281816135bf0152613864015260006108d50152600081816105ba01526130d90152600081816103df01526130a601526000818161062301526130730152600081816106cd015261304001526000818161089e01528181612ea501528181613a2d01528181614067015281816140a7015261455a0152600081816104e201528181612e72015281816139fd0152818161400101528181614041015261450a01526000818161081b01528181612e3f015281816139cd01528181613fdb01526144ba01526000818161045301528181612e0c0152818161399d01528181613fb1015261446a0152600081816107580152818161092601528181610a1e01528181610aad01528181610b4401528181610bd501528181610fe8015281816110e2015281816111ae015281816112c20152818161134f015281816113dc015281816114ba0152818161154e015281816115e2015281816116760152818161171301528181611850015281816119a301528181611ada01528181611b7601528181611c4601528181611cf301528181611d3b01528181611dec01528181611e210152818161214b015281816121c901528181612353015281816123e10152818161249c015281816126640152818161271d01528181612752015281816127ed01528181612b1001528181612b7d01528181612c0e0152818161334a0152818161415b015261469001526000818161051c0152818161316201528181614203015261473701526000818161077f01528181610d7801528181611f2c015281816125a6015281816128f701528181612f430152818161331b0152818161345601528181613719015281816137d00152818161390f01528181613f29015281816141920152818161428b0152818161433f015281816143fb015281816145f20152818161466101526147a10152614f6c6000f3fe608060405234801561001057600080fd5b50600436106103af5760003560e01c806378aa73a4116101f4578063c66106571161011a578063e2e7d264116100ad578063f446c1d01161007c578063f446c1d0146108c0578063f851a440146108c8578063fc0c546a146108d0578063fee3f7f9146108f757600080fd5b8063e2e7d26414610860578063ec026ca714610873578063ecb586a514610886578063ef14101e1461089957600080fd5b8063d21220a7116100e9578063d21220a714610816578063d96c7fce1461083d578063dd62ed3e14610845578063ddca3f431461085857600080fd5b8063c6610657146107c8578063cc2b27d7146107db578063ce30bbdb146107ee578063cf023dd01461080357600080fd5b8063a6417ed611610192578063bb7b8b8011610161578063bb7b8b801461074b578063bd90df7014610753578063c12c21c01461077a578063c21ee162146107a157600080fd5b8063a6417ed61461070a578063b4b577ad1461071d578063b739953e14610725578063b9947eb01461073857600080fd5b806385ca3c6f116101ce57806385ca3c6f146106b55780638ba51dfc146106c857806395d89b41146106ef5780639fdaea0c146106f757600080fd5b806378aa73a41461066057806379bea6641461067b57806382c630661461068e57600080fd5b8063313ce567116102d95780635409491a1161027757806364a89bca1161024657806364a89bca146105f75780636e1d82711461061e57806370a082311461064557806376a2f0f01461065857600080fd5b80635409491a146105ad57806357d78875146105b55780635e0d443f146105dc57806363543f06146105ef57600080fd5b80633df02124116102b35780633df021241461056c5780634469e30e1461057f5780634515cef3146105875780634903b0d11461059a57600080fd5b8063313ce5671461053e57806333d2ebf2146105465780633883e1191461055957600080fd5b806318160ddd1161035157806323746eb81161032057806323746eb8146104ca57806325be124e146104dd5780632c5788d2146105045780632f7a18811461051757600080fd5b806318160ddd146104925780631a4d01d21461049a5780631af4de83146104af5780632081066c146104c257600080fd5b806307211ef71161038d57806307211ef71461043b5780630dfe16811461044e578063140522881461047557806314f059791461047d57600080fd5b8063065a80d8146103b457806306871163146103da57806306fdde0314610426575b600080fd5b6103c76103c2366004614829565b6108ff565b6040519081526020015b60405180910390f35b6104017f000000000000000000000000000000000000000000000000000000000000000081565b60405173ffffffffffffffffffffffffffffffffffffffff90911681526020016103d1565b61042e610922565b6040516103d191906148ba565b6103c76104493660046148cd565b6109da565b6104017f000000000000000000000000000000000000000000000000000000000000000081565b6103c7610aa9565b610485610b3a565b6040516103d1919061492c565b6103c7610bd1565b6104ad6104a836600461493a565b610c3e565b005b6104ad6104bd3660046148cd565b610cd6565b6103c7610fe4565b6104016104d8366004614829565b611051565b6104017f000000000000000000000000000000000000000000000000000000000000000081565b6103c761051236600461495f565b61105c565b6104017f000000000000000000000000000000000000000000000000000000000000000081565b6103c761170f565b6104ad61055436600461498b565b61177c565b6103c76105673660046149d4565b611810565b6104ad61057a366004614a0c565b6118c8565b610485611999565b6104ad610595366004614a4e565b611a0c565b6103c76105a8366004614a79565b611aa8565b6103c7611c42565b6104017f000000000000000000000000000000000000000000000000000000000000000081565b6103c76105ea3660046148cd565b611caf565b6103c7611d37565b6104017f000000000000000000000000000000000000000000000000000000000000000081565b6104017f000000000000000000000000000000000000000000000000000000000000000081565b6103c7610653366004614ab4565b611da4565b6103c7611e1d565b610668600281565b60405161ffff90911681526020016103d1565b6104ad6106893660046148cd565b611e8a565b6104017f000000000000000000000000000000000000000000000000000000000000000081565b6104856106c3366004614ad1565b612106565b6104017f000000000000000000000000000000000000000000000000000000000000000081565b61042e6121c5565b6104ad610705366004614a4e565b612232565b6104ad610718366004614a0c565b6122c5565b6103c761234f565b610401610733366004614829565b6123bc565b610401610746366004614a79565b6123c7565b6103c76123dd565b6104017f000000000000000000000000000000000000000000000000000000000000000081565b6104017f000000000000000000000000000000000000000000000000000000000000000081565b6103c77f000000000000000000000000000000000000000000000000000000000000000081565b6104016107d6366004614a79565b61244a565b6103c76107e936600461495f565b612460565b6107f6600681565b6040516103d19190614b0f565b6104ad61081136600461493a565b6124e0565b6104017f000000000000000000000000000000000000000000000000000000000000000081565b61048561265a565b6103c7610853366004614b50565b6126cd565b6103c761274e565b6103c761086e366004614a79565b6127bb565b6104ad61088136600461498b565b612831565b6104ad610894366004614b7e565b612a92565b6104017f000000000000000000000000000000000000000000000000000000000000000081565b6103c7612b0c565b610401612b79565b6104017f000000000000000000000000000000000000000000000000000000000000000081565b6103c7612c0a565b600061091c826fffffffffffffffffffffffffffffffff16611aa8565b92915050565b60607f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff166306fdde036040518163ffffffff1660e01b8152600401600060405180830381865afa15801561098f573d6000803e3d6000fd5b505050506040513d6000823e601f3d9081017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe01682016040526109d59190810190614c63565b905090565b6040517f07211ef7000000000000000000000000000000000000000000000000000000008152600f84810b600483015283900b6024820152604481018290526000907f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff16906307211ef7906064015b602060405180830381865afa158015610a7b573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610a9f9190614cb4565b90505b9392505050565b60007f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff1663140522886040518163ffffffff1660e01b8152600401602060405180830381865afa158015610b16573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906109d59190614cb4565b610b426147f9565b7f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff166314f059796040518163ffffffff1660e01b8152600401606060405180830381865afa158015610bad573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906109d59190614ccd565b60007f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff166318160ddd6040518163ffffffff1660e01b8152600401602060405180830381865afa158015610b16573d6000803e3d6000fd5b60026000541415610cb0576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c0060448201526064015b60405180910390fd5b60026000908155610cc083612dfb565b9050610ccb81612f12565b505060016000555050565b60026000541415610d43576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152606401610ca7565b600260009081556040517fe958b7040000000000000000000000000000000000000000000000000000000081523360048201527f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff169063e958b70490602401602060405180830381865afa158015610dd4573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610df89190614d4b565b90506000610e058561302f565b90506000610e128561302f565b6040517f70a0823100000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff85811660048301529192506000918416906370a0823190602401602060405180830381865afa158015610e84573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610ea89190614cb4565b90506001811115610fd6577fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0160006b033b2e3c9fd0803ce8000000610eee8784614d68565b610ef89190614dcc565b604051600f8a810b602483015289900b60448201526064810184905260848101829052909150610fd3908690869086907fa6417ed6000000000000000000000000000000000000000000000000000000009060a4015b604080517fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe08184030181529190526020810180517bffffffffffffffffffffffffffffffffffffffffffffffffffffffff167fffffffff0000000000000000000000000000000000000000000000000000000090931692909217909152600180613145565b50505b505060016000555050505050565b60007f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff16632081066c6040518163ffffffff1660e01b8152600401602060405180830381865afa158015610b16573d6000803e3d6000fd5b600061091c82612dfb565b60007f00000000000000000000000000000000000000000000000000000000000000006002141561122d5781600f0b60001461115f57604080518082018252600081526020810185905290517fed8e84f300000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff7f0000000000000000000000000000000000000000000000000000000000000000169163ed8e84f3916111199190600190600401614e07565b602060405180830381865afa158015611136573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061115a9190614cb4565b611226565b6040805180820182528481526000602082015290517fed8e84f300000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff7f0000000000000000000000000000000000000000000000000000000000000000169163ed8e84f3916111e59190600190600401614e07565b602060405180830381865afa158015611202573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906112269190614cb4565b905061091c565b7f0000000000000000000000000000000000000000000000000000000000000000600314156114135781600f0b6000146113865781600f0b6001146112f957604080516060810182526000808252602082015280820185905290517f3883e11900000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff7f00000000000000000000000000000000000000000000000000000000000000001691633883e119916111199190600190600401614e41565b604080516060810182526000808252602082018690528183015290517f3883e11900000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff7f00000000000000000000000000000000000000000000000000000000000000001691633883e119916111199190600190600401614e41565b604080516060810182528481526000602082018190528183015290517f3883e11900000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff7f00000000000000000000000000000000000000000000000000000000000000001691633883e119916111e59190600190600401614e41565b7f0000000000000000000000000000000000000000000000000000000000000000600414156116ad5781600f0b6000146116195781600f0b6001146115855781600f0b6002146114f15760408051608081018252600080825260208201819052818301526060810185905290517fcf701ff700000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff7f0000000000000000000000000000000000000000000000000000000000000000169163cf701ff7916111199190600190600401614e5e565b60408051608081018252600080825260208201819052818301869052606082015290517fcf701ff700000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff7f0000000000000000000000000000000000000000000000000000000000000000169163cf701ff7916111199190600190600401614e5e565b60408051608081018252600080825260208201869052818301819052606082015290517fcf701ff700000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff7f0000000000000000000000000000000000000000000000000000000000000000169163cf701ff7916111199190600190600401614e5e565b60408051608081018252848152600060208201819052818301819052606082015290517fcf701ff700000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff7f0000000000000000000000000000000000000000000000000000000000000000169163cf701ff7916111e59190600190600401614e5e565b6040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601060248201527f496e636f7272656374206e436f696e73000000000000000000000000000000006044820152606401610ca7565b60007f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff1663313ce5676040518163ffffffff1660e01b8152600401602060405180830381865afa158015610b16573d6000803e3d6000fd5b600260005414156117e9576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152606401610ca7565b600260009081556117f983612dfb565b9050611806838284613425565b5050600160005550565b6040517f3883e11900000000000000000000000000000000000000000000000000000000815260009073ffffffffffffffffffffffffffffffffffffffff7f00000000000000000000000000000000000000000000000000000000000000001690633883e119906118879086908690600401614e98565b602060405180830381865afa1580156118a4573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610aa29190614cb4565b60026000541415611935576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152606401610ca7565b6002600090815561194585612dfb565b9050600061195285612dfb565b9050610fd682826000368080601f01602080910402602001604051908101604052809392919081815260200183838082843760009201829052506001935091506136d09050565b6119a16147f9565b7f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff16634469e30e6040518163ffffffff1660e01b8152600401606060405180830381865afa158015610bad573d6000803e3d6000fd5b60026000541415611a79576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152606401610ca7565b60026000908155611a9f906001843581109160208601358210916040870135119061379f565b50506001600055565b6040517f4903b0d1000000000000000000000000000000000000000000000000000000008152600481018290526000907f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff1690634903b0d190602401602060405180830381865afa925050508015611b70575060408051601f3d9081017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0168201909252611b6d91810190614cb4565b60015b61091c577f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff1663065a80d8611bc1611bbc85612c77565b612d2d565b6040517fffffffff0000000000000000000000000000000000000000000000000000000060e084901b168152600f9190910b60048201526024015b602060405180830381865afa158015611c19573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061091c9190614cb4565b919050565b60007f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff16635409491a6040518163ffffffff1660e01b8152600401602060405180830381865afa158015610b16573d6000803e3d6000fd5b6040517f5e0d443f000000000000000000000000000000000000000000000000000000008152600f84810b600483015283900b6024820152604481018290526000907f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff1690635e0d443f90606401610a5e565b60007f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff166363543f066040518163ffffffff1660e01b8152600401602060405180830381865afa158015610b16573d6000803e3d6000fd5b6040517f70a0823100000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff82811660048301526000917f0000000000000000000000000000000000000000000000000000000000000000909116906370a0823190602401611bfc565b60007f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff166376a2f0f06040518163ffffffff1660e01b8152600401602060405180830381865afa158015610b16573d6000803e3d6000fd5b60026000541415611ef7576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152606401610ca7565b600260009081556040517fe958b7040000000000000000000000000000000000000000000000000000000081523360048201527f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff169063e958b70490602401602060405180830381865afa158015611f88573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611fac9190614d4b565b90506000611fb985612dfb565b90506000611fc685612dfb565b6040517f70a0823100000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff85811660048301529192506000918416906370a0823190602401602060405180830381865afa158015612038573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061205c9190614cb4565b90506001811115610fd6577fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0160006b033b2e3c9fd0803ce80000006120a28784614d68565b6120ac9190614dcc565b604051600f8a810b602483015289900b60448201526064810184905260848101829052909150610fd3908690869086907f3df02124000000000000000000000000000000000000000000000000000000009060a401610f4e565b61210e6147f9565b6040517f85ca3c6f00000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff7f000000000000000000000000000000000000000000000000000000000000000016906385ca3c6f9061218490879087908790600401614eb0565b606060405180830381865afa1580156121a1573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610a9f9190614ccd565b60607f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff166395d89b416040518163ffffffff1660e01b8152600401600060405180830381865afa15801561098f573d6000803e3d6000fd5b6002600054141561229f576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152606401610ca7565b60026000908155611a9f90600184358110916020860135821091604087013511906138de565b60026000541415612332576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152606401610ca7565b600260009081556123428561302f565b905060006119528561302f565b60007f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff1663b4b577ad6040518163ffffffff1660e01b8152600401602060405180830381865afa158015610b16573d6000803e3d6000fd5b600061091c8261302f565b600061091c6123d8611bbc84612c77565b61302f565b60007f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff1663bb7b8b806040518163ffffffff1660e01b8152600401602060405180830381865afa158015610b16573d6000803e3d6000fd5b600061091c61245b611bbc84612c77565b612dfb565b6040517fcc2b27d700000000000000000000000000000000000000000000000000000000815260048101839052600f82900b60248201526000907f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff169063cc2b27d790604401611887565b6002600054141561254d576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152606401610ca7565b6002600090815561255d83612dfb565b6040517fe958b70400000000000000000000000000000000000000000000000000000000815233600482015290915060009073ffffffffffffffffffffffffffffffffffffffff7f0000000000000000000000000000000000000000000000000000000000000000169063e958b70490602401602060405180830381865afa1580156125ed573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906126119190614d4b565b905061264d81837f0000000000000000000000000000000000000000000000000000000000000000612644888a89613a9b565b60016000613145565b5050600160005550505050565b6126626147f9565b7f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff1663d96c7fce6040518163ffffffff1660e01b8152600401606060405180830381865afa158015610bad573d6000803e3d6000fd5b6040517fdd62ed3e00000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff838116600483015282811660248301526000917f00000000000000000000000000000000000000000000000000000000000000009091169063dd62ed3e90604401611887565b60007f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff1663ddca3f436040518163ffffffff1660e01b8152600401602060405180830381865afa158015610b16573d6000803e3d6000fd5b6040517fe2e7d264000000000000000000000000000000000000000000000000000000008152600481018290526000907f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff169063e2e7d26490602401611bfc565b6002600054141561289e576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152606401610ca7565b600260009081556128ae83612dfb565b6040517fe958b70400000000000000000000000000000000000000000000000000000000815233600482015290915060009073ffffffffffffffffffffffffffffffffffffffff7f0000000000000000000000000000000000000000000000000000000000000000169063e958b70490602401602060405180830381865afa15801561293e573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906129629190614d4b565b6040517f70a0823100000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff80831660048301529192506000918416906370a0823190602401602060405180830381865afa1580156129d4573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906129f89190614cb4565b90506001811115612a86577fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0160006b033b2e3c9fd0803ce8000000612a3e8684614d68565b612a489190614dcc565b9050612a8383857f0000000000000000000000000000000000000000000000000000000000000000612a7b8a8787613a9b565b600180613145565b50505b50506001600055505050565b60026000541415612aff576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152606401610ca7565b6002600055611a9f613ef8565b60007f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff1663f446c1d06040518163ffffffff1660e01b8152600401602060405180830381865afa158015610b16573d6000803e3d6000fd5b60007f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff1663f851a4406040518163ffffffff1660e01b8152600401602060405180830381865afa158015612be6573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906109d59190614d4b565b60007f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff1663fee3f7f96040518163ffffffff1660e01b8152600401602060405180830381865afa158015610b16573d6000803e3d6000fd5b60007f7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff821115612d29576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602860248201527f53616665436173743a2076616c756520646f65736e27742066697420696e206160448201527f6e20696e743235360000000000000000000000000000000000000000000000006064820152608401610ca7565b5090565b60007fffffffffffffffffffffffffffffffff800000000000000000000000000000008212801590612d6f57506f7fffffffffffffffffffffffffffffff8213155b612d29576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602760248201527f53616665436173743a2076616c756520646f65736e27742066697420696e203160448201527f32382062697473000000000000000000000000000000000000000000000000006064820152608401610ca7565b600081600f0b60001415612e3057507f0000000000000000000000000000000000000000000000000000000000000000612ec5565b81600f0b60011415612e6357507f0000000000000000000000000000000000000000000000000000000000000000612ec5565b81600f0b60021415612e9657507f0000000000000000000000000000000000000000000000000000000000000000612ec5565b81600f0b60031415612ec557507f00000000000000000000000000000000000000000000000000000000000000005b73ffffffffffffffffffffffffffffffffffffffff8116611c3d576040517fd1da79bc00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6040517fe958b7040000000000000000000000000000000000000000000000000000000081523360048201526000907f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff169063e958b70490602401602060405180830381865afa158015612f9f573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612fc39190614d4b565b905061302a817f0000000000000000000000000000000000000000000000000000000000000000846000368080601f016020809104026020016040519081016040528093929190818152602001838380828437600092018290525092508291506131459050565b505050565b600081600f0b6000141561306457507f0000000000000000000000000000000000000000000000000000000000000000612ec5565b81600f0b6001141561309757507f0000000000000000000000000000000000000000000000000000000000000000612ec5565b81600f0b600214156130ca57507f0000000000000000000000000000000000000000000000000000000000000000612ec5565b81600f0b60031415612ec557507f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff8116611c3d576040517fd1da79bc00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b60606000803373ffffffffffffffffffffffffffffffffffffffff7f000000000000000000000000000000000000000000000000000000000000000016146132ae576040517f70a0823100000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff8a811660048301528916906370a0823190602401602060405180830381865afa1580156131f3573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906132179190614cb4565b6040517f70a0823100000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff8b81166004830152919350908816906370a0823190602401602060405180830381865afa158015613287573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906132ab9190614cb4565b90505b84156132de576132de887fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff614118565b6040517f6ce4074a00000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff7f00000000000000000000000000000000000000000000000000000000000000001690636ce4074a906133749033907f0000000000000000000000000000000000000000000000000000000000000000908b90600401614ed7565b6000604051808303816000875af1158015613393573d6000803e3d6000fd5b505050506040513d6000823e601f3d9081017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe01682016040526133d99190810190614c63565b9250841561340b5761340b887fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff614118565b6134198989898585896141eb565b50509695505050505050565b6040517fe958b7040000000000000000000000000000000000000000000000000000000081523360048201526000907f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff169063e958b70490602401602060405180830381865afa1580156134b2573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906134d69190614d4b565b6040517f70a0823100000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff80831660048301529192506000917f000000000000000000000000000000000000000000000000000000000000000016906370a0823190602401602060405180830381865afa158015613568573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061358c9190614cb4565b905060018111156136c9577fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff016136c7827f0000000000000000000000000000000000000000000000000000000000000000867f1a4d01d200000000000000000000000000000000000000000000000000000000858a6b033b2e3c9fd0803ce80000006136198b84614d68565b6136239190614dcc565b6040516024810193909352600f9190910b60448301526064820152608401604080517fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe08184030181529190526020810180517bffffffffffffffffffffffffffffffffffffffffffffffffffffffff167fffffffff000000000000000000000000000000000000000000000000000000009093169290921790915260006001613145565b505b5050505050565b6040517fe958b70400000000000000000000000000000000000000000000000000000000815233600482015260609060009073ffffffffffffffffffffffffffffffffffffffff7f0000000000000000000000000000000000000000000000000000000000000000169063e958b70490602401602060405180830381865afa158015613760573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906137849190614d4b565b9050613794818888888888613145565b979650505050505050565b6040517fe958b7040000000000000000000000000000000000000000000000000000000081523360048201526000907f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff169063e958b70490602401602060405180830381865afa15801561382c573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906138509190614d4b565b905061385e8585858561445f565b613888817f00000000000000000000000000000000000000000000000000000000000000006145a5565b6138c86000368080601f01602080910402602001604051908101604052809392919081815260200183838082843760009201919091525061462192505050565b506138d58585858561445f565b6136c98161471f565b6040517fe958b7040000000000000000000000000000000000000000000000000000000081523360048201526000907f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff169063e958b70490602401602060405180830381865afa15801561396b573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061398f9190614d4b565b905084156139c1576139c1817f00000000000000000000000000000000000000000000000000000000000000006145a5565b83156139f1576139f1817f00000000000000000000000000000000000000000000000000000000000000006145a5565b8215613a2157613a21817f00000000000000000000000000000000000000000000000000000000000000006145a5565b8115613a5157613a51817f00000000000000000000000000000000000000000000000000000000000000006145a5565b613a916000368080601f01602080910402602001604051908101604052809392919081815260200183838082843760009201919091525061462192505050565b506136c98161471f565b60607f000000000000000000000000000000000000000000000000000000000000000060021415613c545783600f0b600014613b91576040516000602482015260448101849052606481018390527f0b4c7e4d00000000000000000000000000000000000000000000000000000000906084015b604080517fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe08184030181529190526020810180517bffffffffffffffffffffffffffffffffffffffffffffffffffffffff167fffffffff0000000000000000000000000000000000000000000000000000000090931692909217909152613c4d565b6040516024810184905260006044820152606481018390527f0b4c7e4d00000000000000000000000000000000000000000000000000000000906084015b604080517fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe08184030181529190526020810180517bffffffffffffffffffffffffffffffffffffffffffffffffffffffff167fffffffff00000000000000000000000000000000000000000000000000000000909316929092179091525b9050610aa2565b7f000000000000000000000000000000000000000000000000000000000000000060031415613d6e5783600f0b600014613d255783600f0b600114613cdc57604051600060248201819052604482015260648101849052608481018390527f4515cef3000000000000000000000000000000000000000000000000000000009060a401613b0f565b604051600060248201819052604482018590526064820152608481018390527f4515cef3000000000000000000000000000000000000000000000000000000009060a401613b0f565b604051602481018490526000604482018190526064820152608481018390527f4515cef3000000000000000000000000000000000000000000000000000000009060a401613bcf565b7f0000000000000000000000000000000000000000000000000000000000000000600414156116ad5783600f0b600014613ea85783600f0b600114613e585783600f0b600214613e08576040516000602482018190526044820181905260648201526084810184905260a481018390527f029b2f34000000000000000000000000000000000000000000000000000000009060c401613b0f565b6040516000602482018190526044820181905260648201859052608482015260a481018390527f029b2f34000000000000000000000000000000000000000000000000000000009060c401613b0f565b6040516000602482018190526044820185905260648201819052608482015260a481018390527f029b2f34000000000000000000000000000000000000000000000000000000009060c401613b0f565b6040516024810184905260006044820181905260648201819052608482015260a481018390527f029b2f34000000000000000000000000000000000000000000000000000000009060c401613bcf565b6040517fe958b7040000000000000000000000000000000000000000000000000000000081523360048201526000907f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff169063e958b70490602401602060405180830381865afa158015613f85573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190613fa99190614d4b565b9050613fd5817f00000000000000000000000000000000000000000000000000000000000000006145a5565b613fff817f00000000000000000000000000000000000000000000000000000000000000006145a5565b7f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff16156140cb57614065817f00000000000000000000000000000000000000000000000000000000000000006145a5565b7f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff16156140cb576140cb817f00000000000000000000000000000000000000000000000000000000000000006145a5565b61410b6000368080601f01602080910402602001604051908101604052809392919081815260200183838082843760009201919091525061462192505050565b506141158161471f565b50565b6040517f46fb371d00000000000000000000000000000000000000000000000000000000815233600482015273ffffffffffffffffffffffffffffffffffffffff7f0000000000000000000000000000000000000000000000000000000000000000811660248301528381166044830152606482018390527f000000000000000000000000000000000000000000000000000000000000000016906346fb371d906084015b600060405180830381600087803b1580156141d757600080fd5b505af11580156136c7573d6000803e3d6000fd5b3373ffffffffffffffffffffffffffffffffffffffff7f000000000000000000000000000000000000000000000000000000000000000016146142ec576040517f654a9eda00000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff87811660048301528681166024830152858116604483015260648201859052608482018490527f0000000000000000000000000000000000000000000000000000000000000000169063654a9eda9060a401600060405180830381600087803b1580156142cf57600080fd5b505af11580156142e3573d6000803e3d6000fd5b505050506136c7565b80156143ae576040517f0d8f9cee00000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff878116600483015286811660248301527f00000000000000000000000000000000000000000000000000000000000000001690630d8f9cee906044016020604051808303816000875af1158015614388573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906143ac9190614f19565b505b6040517f51e3f16000000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff878116600483015285811660248301527f000000000000000000000000000000000000000000000000000000000000000016906351e3f16090604401600060405180830381600087803b15801561443f57600080fd5b505af1158015614453573d6000803e3d6000fd5b50505050505050505050565b83156144af576144af7f00000000000000000000000000000000000000000000000000000000000000007fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff614118565b82156144ff576144ff7f00000000000000000000000000000000000000000000000000000000000000007fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff614118565b811561454f5761454f7f00000000000000000000000000000000000000000000000000000000000000007fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff614118565b801561459f5761459f7f00000000000000000000000000000000000000000000000000000000000000007fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff614118565b50505050565b6040517f51e3f16000000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff838116600483015282811660248301527f000000000000000000000000000000000000000000000000000000000000000016906351e3f160906044016141bd565b6040517f6ce4074a00000000000000000000000000000000000000000000000000000000815260609073ffffffffffffffffffffffffffffffffffffffff7f00000000000000000000000000000000000000000000000000000000000000001690636ce4074a906146ba9033907f0000000000000000000000000000000000000000000000000000000000000000908790600401614ed7565b6000604051808303816000875af11580156146d9573d6000803e3d6000fd5b505050506040513d6000823e601f3d9081017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe016820160405261091c9190810190614c63565b3373ffffffffffffffffffffffffffffffffffffffff7f00000000000000000000000000000000000000000000000000000000000000001614614115576040517f9537301800000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff82811660048301527f00000000000000000000000000000000000000000000000000000000000000001690639537301890602401600060405180830381600087803b1580156147e557600080fd5b505af11580156136c9573d6000803e3d6000fd5b60405180606001604052806003906020820280368337509192915050565b8035600f81900b8114611c3d57600080fd5b60006020828403121561483b57600080fd5b610aa282614817565b60005b8381101561485f578181015183820152602001614847565b8381111561459f5750506000910152565b60008151808452614888816020860160208601614844565b601f017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0169290920160200192915050565b602081526000610aa26020830184614870565b6000806000606084860312156148e257600080fd5b6148eb84614817565b92506148f960208501614817565b9150604084013590509250925092565b8060005b600381101561459f57815184526020938401939091019060010161490d565b6060810161091c8284614909565b60008060006060848603121561494f57600080fd5b833592506148f960208501614817565b6000806040838503121561497257600080fd5b8235915061498260208401614817565b90509250929050565b6000806040838503121561499e57600080fd5b6149a783614817565b946020939093013593505050565b806060810183101561091c57600080fd5b801515811461411557600080fd5b600080608083850312156149e757600080fd5b6149f184846149b5565b91506060830135614a01816149c6565b809150509250929050565b60008060008060808587031215614a2257600080fd5b614a2b85614817565b9350614a3960208601614817565b93969395505050506040820135916060013590565b60008060808385031215614a6157600080fd5b614a6b84846149b5565b946060939093013593505050565b600060208284031215614a8b57600080fd5b5035919050565b73ffffffffffffffffffffffffffffffffffffffff8116811461411557600080fd5b600060208284031215614ac657600080fd5b8135610aa281614a92565b600080600060e08486031215614ae657600080fd5b614af085856149b5565b9250614aff85606086016149b5565b915060c084013590509250925092565b6020810160108310614b4a577f4e487b7100000000000000000000000000000000000000000000000000000000600052602160045260246000fd5b91905290565b60008060408385031215614b6357600080fd5b8235614b6e81614a92565b91506020830135614a0181614a92565b60008060808385031215614b9157600080fd5b8235915061498284602085016149b5565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b600067ffffffffffffffff80841115614bec57614bec614ba2565b604051601f85017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0908116603f01168101908282118183101715614c3257614c32614ba2565b81604052809350858152868686011115614c4b57600080fd5b614c59866020830187614844565b5050509392505050565b600060208284031215614c7557600080fd5b815167ffffffffffffffff811115614c8c57600080fd5b8201601f81018413614c9d57600080fd5b614cac84825160208401614bd1565b949350505050565b600060208284031215614cc657600080fd5b5051919050565b600060608284031215614cdf57600080fd5b82601f830112614cee57600080fd5b6040516060810181811067ffffffffffffffff82111715614d1157614d11614ba2565b604052806060840185811115614d2657600080fd5b845b81811015614d40578051835260209283019201614d28565b509195945050505050565b600060208284031215614d5d57600080fd5b8151610aa281614a92565b6000817fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0483118215151615614dc7577f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b500290565b600082614e02577f4e487b7100000000000000000000000000000000000000000000000000000000600052601260045260246000fd5b500490565b60608101818460005b6002811015614e2f578151835260209283019290910190600101614e10565b50505082151560408301529392505050565b60808101614e4f8285614909565b82151560608301529392505050565b60a08101818460005b6004811015614e86578151835260209283019290910190600101614e67565b50505082151560808301529392505050565b60808101606084833791151560609190910152919050565b60e081016060858337606082016000815260608582375060c0919091019190915292915050565b600073ffffffffffffffffffffffffffffffffffffffff808616835280851660208401525060606040830152614f106060830184614870565b95945050505050565b600060208284031215614f2b57600080fd5b8151610aa2816149c656fea2646970667358221220bc8b74748f92ac3d72405003a7d775392e376d2e21dd711b6e1bfc4f8ed79c2e64736f6c634300080a003300000000000000000000000095357303f995e184a7998da6c6ea35cc728a1900000000000000000000000000bebc44782c7db0a1a60cb6fe97d0b483032ff1c70000000000000000000000006c3f90f043a72fa612cbac8115ee7e52bde6e4900000000000000000000000000000000000000000000000000000000000000000
Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)
00000000000000000000000095357303f995e184a7998da6c6ea35cc728a1900000000000000000000000000bebc44782c7db0a1a60cb6fe97d0b483032ff1c70000000000000000000000006c3f90f043a72fa612cbac8115ee7e52bde6e4900000000000000000000000000000000000000000000000000000000000000000
-----Decoded View---------------
Arg [0] : _creditManager (address): 0x95357303f995e184A7998dA6C6eA35cC728A1900
Arg [1] : _curvePool (address): 0xbEbc44782C7dB0a1A60Cb6fe97d0b483032FF1C7
Arg [2] : _lp_token (address): 0x6c3F90f043a72FA612cbac8115EE7e52BDe6E490
Arg [3] : _metapoolBase (address): 0x0000000000000000000000000000000000000000
-----Encoded View---------------
4 Constructor Arguments found :
Arg [0] : 00000000000000000000000095357303f995e184a7998da6c6ea35cc728a1900
Arg [1] : 000000000000000000000000bebc44782c7db0a1a60cb6fe97d0b483032ff1c7
Arg [2] : 0000000000000000000000006c3f90f043a72fa612cbac8115ee7e52bde6e490
Arg [3] : 0000000000000000000000000000000000000000000000000000000000000000
Make sure to use the "Vote Down" button for any spammy posts, and the "Vote Up" for interesting conversations.
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.