Overview
S Balance
0 S
S Value
$0.00More Info
Private Name Tags
ContractCreator
Loading...
Loading
Similar Match Source Code This contract matches the deployed Bytecode of the Source Code for Contract 0x4Bc73050...FFAA879f8 The constructor portion of the code might be different and could alter the actual behaviour of the contract
Contract Name:
OSonicVaultAdmin
Compiler Version
v0.8.28+commit.7893614a
Optimization Enabled:
Yes with 200 runs
Other Settings:
paris EvmVersion
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import { OETHVaultAdmin } from "./OETHVaultAdmin.sol"; /** * @title Origin Sonic VaultAdmin contract on Sonic * @author Origin Protocol Inc */ contract OSonicVaultAdmin is OETHVaultAdmin { /// @param _wS Sonic's Wrapped S token constructor(address _wS) OETHVaultAdmin(_wS) {} /*************************************** Asset Config ****************************************/ /** * @notice Add a supported asset to the contract, i.e. one that can be to mint OTokens. * @dev Overridden to remove price provider integration * @param _asset Address of asset * @param _unitConversion 0 decimals, 1 exchange rate */ function supportAsset(address _asset, uint8 _unitConversion) external override onlyGovernor { require(!assets[_asset].isSupported, "Asset already supported"); assets[_asset] = Asset({ isSupported: true, unitConversion: UnitConversion(_unitConversion), decimals: 0, // will be overridden in _cacheDecimals allowedOracleSlippageBps: 0 // 0% by default }); _cacheDecimals(_asset); allAssets.push(_asset); emit AssetSupported(_asset); } }
// 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: MIT // OpenZeppelin Contracts v4.4.1 (token/ERC20/utils/SafeERC20.sol) pragma solidity ^0.8.0; import "../IERC20.sol"; import "../../../utils/Address.sol"; /** * @title SafeERC20 * @dev Wrappers around ERC20 operations that throw on failure (when the token * contract returns false). Tokens that return no value (and instead revert or * throw on failure) are also supported, non-reverting calls are assumed to be * successful. * To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract, * which allows you to call the safe operations as `token.safeTransfer(...)`, etc. */ library SafeERC20 { using Address for address; function safeTransfer( IERC20 token, address to, uint256 value ) internal { _callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value)); } function safeTransferFrom( IERC20 token, address from, address to, uint256 value ) internal { _callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value)); } /** * @dev Deprecated. This function has issues similar to the ones found in * {IERC20-approve}, and its usage is discouraged. * * Whenever possible, use {safeIncreaseAllowance} and * {safeDecreaseAllowance} instead. */ function safeApprove( IERC20 token, address spender, uint256 value ) internal { // safeApprove should only be called when setting an initial allowance, // or when resetting it to zero. To increase and decrease it, use // 'safeIncreaseAllowance' and 'safeDecreaseAllowance' require( (value == 0) || (token.allowance(address(this), spender) == 0), "SafeERC20: approve from non-zero to non-zero allowance" ); _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value)); } function safeIncreaseAllowance( IERC20 token, address spender, uint256 value ) internal { uint256 newAllowance = token.allowance(address(this), spender) + value; _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance)); } function safeDecreaseAllowance( IERC20 token, address spender, uint256 value ) internal { unchecked { uint256 oldAllowance = token.allowance(address(this), spender); require(oldAllowance >= value, "SafeERC20: decreased allowance below zero"); uint256 newAllowance = oldAllowance - value; _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance)); } } /** * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement * on the return value: the return value is optional (but if data is returned, it must not be false). * @param token The token targeted by the call. * @param data The call data (encoded using abi.encode or one of its variants). */ function _callOptionalReturn(IERC20 token, bytes memory data) private { // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since // we're implementing it ourselves. We use {Address.functionCall} to perform this call, which verifies that // the target address contains contract code and also asserts for success in the low-level call. bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed"); if (returndata.length > 0) { // Return data is optional require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed"); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts 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); } } } }
// 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 // OpenZeppelin Contracts v4.4.1 (utils/math/SafeMath.sol) pragma solidity ^0.8.0; // CAUTION // This version of SafeMath should only be used with Solidity 0.8 or later, // because it relies on the compiler's built in overflow checks. /** * @dev Wrappers over Solidity's arithmetic operations. * * NOTE: `SafeMath` is generally not needed starting with Solidity 0.8, since the compiler * now has built in overflow checking. */ library SafeMath { /** * @dev Returns the addition of two unsigned integers, with an overflow flag. * * _Available since v3.4._ */ function tryAdd(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { uint256 c = a + b; if (c < a) return (false, 0); return (true, c); } } /** * @dev Returns the substraction of two unsigned integers, with an overflow flag. * * _Available since v3.4._ */ function trySub(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { if (b > a) return (false, 0); return (true, a - b); } } /** * @dev Returns the multiplication of two unsigned integers, with an overflow flag. * * _Available since v3.4._ */ function tryMul(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { // Gas optimization: this is cheaper than requiring 'a' not being zero, but the // benefit is lost if 'b' is also tested. // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522 if (a == 0) return (true, 0); uint256 c = a * b; if (c / a != b) return (false, 0); return (true, c); } } /** * @dev Returns the division of two unsigned integers, with a division by zero flag. * * _Available since v3.4._ */ function tryDiv(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { if (b == 0) return (false, 0); return (true, a / b); } } /** * @dev Returns the remainder of dividing two unsigned integers, with a division by zero flag. * * _Available since v3.4._ */ function tryMod(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { if (b == 0) return (false, 0); return (true, a % b); } } /** * @dev Returns the addition of two unsigned integers, reverting on * overflow. * * Counterpart to Solidity's `+` operator. * * Requirements: * * - Addition cannot overflow. */ function add(uint256 a, uint256 b) internal pure returns (uint256) { return a + b; } /** * @dev Returns the subtraction of two unsigned integers, reverting on * overflow (when the result is negative). * * Counterpart to Solidity's `-` operator. * * Requirements: * * - Subtraction cannot overflow. */ function sub(uint256 a, uint256 b) internal pure returns (uint256) { return a - b; } /** * @dev Returns the multiplication of two unsigned integers, reverting on * overflow. * * Counterpart to Solidity's `*` operator. * * Requirements: * * - Multiplication cannot overflow. */ function mul(uint256 a, uint256 b) internal pure returns (uint256) { return a * b; } /** * @dev Returns the integer division of two unsigned integers, reverting on * division by zero. The result is rounded towards zero. * * Counterpart to Solidity's `/` operator. * * Requirements: * * - The divisor cannot be zero. */ function div(uint256 a, uint256 b) internal pure returns (uint256) { return a / b; } /** * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo), * reverting when dividing by zero. * * Counterpart to Solidity's `%` operator. This function uses a `revert` * opcode (which leaves remaining gas untouched) while Solidity uses an * invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function mod(uint256 a, uint256 b) internal pure returns (uint256) { return a % b; } /** * @dev Returns the subtraction of two unsigned integers, reverting with custom message on * overflow (when the result is negative). * * CAUTION: This function is deprecated because it requires allocating memory for the error * message unnecessarily. For custom revert reasons use {trySub}. * * Counterpart to Solidity's `-` operator. * * Requirements: * * - Subtraction cannot overflow. */ function sub( uint256 a, uint256 b, string memory errorMessage ) internal pure returns (uint256) { unchecked { require(b <= a, errorMessage); return a - b; } } /** * @dev Returns the integer division of two unsigned integers, reverting with custom message on * division by zero. The result is rounded towards zero. * * Counterpart to Solidity's `/` operator. Note: this function uses a * `revert` opcode (which leaves remaining gas untouched) while Solidity * uses an invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function div( uint256 a, uint256 b, string memory errorMessage ) internal pure returns (uint256) { unchecked { require(b > 0, errorMessage); return a / b; } } /** * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo), * reverting with custom message when dividing by zero. * * CAUTION: This function is deprecated because it requires allocating memory for the error * message unnecessarily. For custom revert reasons use {tryMod}. * * Counterpart to Solidity's `%` operator. This function uses a `revert` * opcode (which leaves remaining gas untouched) while Solidity uses an * invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function mod( uint256 a, uint256 b, string memory errorMessage ) internal pure returns (uint256) { unchecked { require(b > 0, errorMessage); return a % b; } } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; /** * @title Base for contracts that are managed by the Origin Protocol's Governor. * @dev Copy of the openzeppelin Ownable.sol contract with nomenclature change * from owner to governor and renounce methods removed. Does not use * Context.sol like Ownable.sol does for simplification. * @author Origin Protocol Inc */ contract Governable { // Storage position of the owner and pendingOwner of the contract // keccak256("OUSD.governor"); bytes32 private constant governorPosition = 0x7bea13895fa79d2831e0a9e28edede30099005a50d652d8957cf8a607ee6ca4a; // keccak256("OUSD.pending.governor"); bytes32 private constant pendingGovernorPosition = 0x44c4d30b2eaad5130ad70c3ba6972730566f3e6359ab83e800d905c61b1c51db; // keccak256("OUSD.reentry.status"); bytes32 private constant reentryStatusPosition = 0x53bf423e48ed90e97d02ab0ebab13b2a235a6bfbe9c321847d5c175333ac4535; // See OpenZeppelin ReentrancyGuard implementation uint256 constant _NOT_ENTERED = 1; uint256 constant _ENTERED = 2; event PendingGovernorshipTransfer( address indexed previousGovernor, address indexed newGovernor ); event GovernorshipTransferred( address indexed previousGovernor, address indexed newGovernor ); /** * @dev Initializes the contract setting the deployer as the initial Governor. */ constructor() { _setGovernor(msg.sender); emit GovernorshipTransferred(address(0), _governor()); } /** * @notice Returns the address of the current Governor. */ function governor() public view returns (address) { return _governor(); } /** * @dev Returns the address of the current Governor. */ function _governor() internal view returns (address governorOut) { bytes32 position = governorPosition; // solhint-disable-next-line no-inline-assembly assembly { governorOut := sload(position) } } /** * @dev Returns the address of the pending Governor. */ function _pendingGovernor() internal view returns (address pendingGovernor) { bytes32 position = pendingGovernorPosition; // solhint-disable-next-line no-inline-assembly assembly { pendingGovernor := sload(position) } } /** * @dev Throws if called by any account other than the Governor. */ modifier onlyGovernor() { require(isGovernor(), "Caller is not the Governor"); _; } /** * @notice Returns true if the caller is the current Governor. */ function isGovernor() public view returns (bool) { return msg.sender == _governor(); } function _setGovernor(address newGovernor) internal { bytes32 position = governorPosition; // solhint-disable-next-line no-inline-assembly assembly { sstore(position, newGovernor) } } /** * @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 make it call a * `private` function that does the actual work. */ modifier nonReentrant() { bytes32 position = reentryStatusPosition; uint256 _reentry_status; // solhint-disable-next-line no-inline-assembly assembly { _reentry_status := sload(position) } // On the first call to nonReentrant, _notEntered will be true require(_reentry_status != _ENTERED, "Reentrant call"); // Any calls to nonReentrant after this point will fail // solhint-disable-next-line no-inline-assembly assembly { sstore(position, _ENTERED) } _; // By storing the original value once again, a refund is triggered (see // https://eips.ethereum.org/EIPS/eip-2200) // solhint-disable-next-line no-inline-assembly assembly { sstore(position, _NOT_ENTERED) } } function _setPendingGovernor(address newGovernor) internal { bytes32 position = pendingGovernorPosition; // solhint-disable-next-line no-inline-assembly assembly { sstore(position, newGovernor) } } /** * @notice Transfers Governance of the contract to a new account (`newGovernor`). * Can only be called by the current Governor. Must be claimed for this to complete * @param _newGovernor Address of the new Governor */ function transferGovernance(address _newGovernor) external onlyGovernor { _setPendingGovernor(_newGovernor); emit PendingGovernorshipTransfer(_governor(), _newGovernor); } /** * @notice Claim Governance of the contract to a new account (`newGovernor`). * Can only be called by the new Governor. */ function claimGovernance() external { require( msg.sender == _pendingGovernor(), "Only the pending Governor can complete the claim" ); _changeGovernor(msg.sender); } /** * @dev Change Governance of the contract to a new account (`newGovernor`). * @param _newGovernor Address of the new Governor */ function _changeGovernor(address _newGovernor) internal { require(_newGovernor != address(0), "New Governor is address(0)"); emit GovernorshipTransferred(_governor(), _newGovernor); _setGovernor(_newGovernor); } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; interface IBasicToken { function symbol() external view returns (string memory); function decimals() external view returns (uint8); }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; interface IOracle { /** * @dev returns the asset price in USD, in 8 decimal digits. * * The version of priceProvider deployed for OETH has 18 decimal digits */ function price(address asset) external view returns (uint256); }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; /** * @title Platform interface to integrate with lending platform like Compound, AAVE etc. */ interface IStrategy { /** * @dev Deposit the given asset to platform * @param _asset asset address * @param _amount Amount to deposit */ function deposit(address _asset, uint256 _amount) external; /** * @dev Deposit the entire balance of all supported assets in the Strategy * to the platform */ function depositAll() external; /** * @dev Withdraw given asset from Lending platform */ function withdraw( address _recipient, address _asset, uint256 _amount ) external; /** * @dev Liquidate all assets in strategy and return them to Vault. */ function withdrawAll() external; /** * @dev Returns the current balance of the given asset. */ function checkBalance(address _asset) external view returns (uint256 balance); /** * @dev Returns bool indicating whether strategy supports asset. */ function supportsAsset(address _asset) external view returns (bool); /** * @dev Collect reward tokens from the Strategy. */ function collectRewardTokens() external; /** * @dev The address array of the reward tokens for the Strategy. */ function getRewardTokenAddresses() external view returns (address[] memory); }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; interface ISwapper { /** * @param fromAsset The token address of the asset being sold. * @param toAsset The token address of the asset being purchased. * @param fromAssetAmount The amount of assets being sold. * @param minToAssetAmmount The minimum amount of assets to be purchased. * @param data tx.data returned from 1Inch's /v5.0/1/swap API */ function swap( address fromAsset, address toAsset, uint256 fromAssetAmount, uint256 minToAssetAmmount, bytes calldata data ) external returns (uint256 toAssetAmount); }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import { VaultStorage } from "../vault/VaultStorage.sol"; interface IVault { event AssetSupported(address _asset); event AssetDefaultStrategyUpdated(address _asset, address _strategy); event AssetAllocated(address _asset, address _strategy, uint256 _amount); event StrategyApproved(address _addr); event StrategyRemoved(address _addr); event Mint(address _addr, uint256 _value); event Redeem(address _addr, uint256 _value); event CapitalPaused(); event CapitalUnpaused(); event RebasePaused(); event RebaseUnpaused(); event VaultBufferUpdated(uint256 _vaultBuffer); event RedeemFeeUpdated(uint256 _redeemFeeBps); event PriceProviderUpdated(address _priceProvider); event AllocateThresholdUpdated(uint256 _threshold); event RebaseThresholdUpdated(uint256 _threshold); event StrategistUpdated(address _address); event MaxSupplyDiffChanged(uint256 maxSupplyDiff); event YieldDistribution(address _to, uint256 _yield, uint256 _fee); event TrusteeFeeBpsChanged(uint256 _basis); event TrusteeAddressChanged(address _address); event SwapperChanged(address _address); event SwapAllowedUndervalueChanged(uint256 _basis); event SwapSlippageChanged(address _asset, uint256 _basis); event Swapped( address indexed _fromAsset, address indexed _toAsset, uint256 _fromAssetAmount, uint256 _toAssetAmount ); event StrategyAddedToMintWhitelist(address indexed strategy); event StrategyRemovedFromMintWhitelist(address indexed strategy); event DripperChanged(address indexed _dripper); event WithdrawalRequested( address indexed _withdrawer, uint256 indexed _requestId, uint256 _amount, uint256 _queued ); event WithdrawalClaimed( address indexed _withdrawer, uint256 indexed _requestId, uint256 _amount ); event WithdrawalClaimable(uint256 _claimable, uint256 _newClaimable); // Governable.sol function transferGovernance(address _newGovernor) external; function claimGovernance() external; function governor() external view returns (address); // VaultAdmin.sol function setPriceProvider(address _priceProvider) external; function priceProvider() external view returns (address); function setRedeemFeeBps(uint256 _redeemFeeBps) external; function redeemFeeBps() external view returns (uint256); function setVaultBuffer(uint256 _vaultBuffer) external; function vaultBuffer() external view returns (uint256); function setAutoAllocateThreshold(uint256 _threshold) external; function autoAllocateThreshold() external view returns (uint256); function setRebaseThreshold(uint256 _threshold) external; function rebaseThreshold() external view returns (uint256); function setStrategistAddr(address _address) external; function strategistAddr() external view returns (address); function setMaxSupplyDiff(uint256 _maxSupplyDiff) external; function maxSupplyDiff() external view returns (uint256); function setTrusteeAddress(address _address) external; function trusteeAddress() external view returns (address); function setTrusteeFeeBps(uint256 _basis) external; function trusteeFeeBps() external view returns (uint256); function ousdMetaStrategy() external view returns (address); function setSwapper(address _swapperAddr) external; function setSwapAllowedUndervalue(uint16 _percentageBps) external; function setOracleSlippage(address _asset, uint16 _allowedOracleSlippageBps) external; function supportAsset(address _asset, uint8 _supportsAsset) external; function approveStrategy(address _addr) external; function removeStrategy(address _addr) external; function setAssetDefaultStrategy(address _asset, address _strategy) external; function assetDefaultStrategies(address _asset) external view returns (address); function pauseRebase() external; function unpauseRebase() external; function rebasePaused() external view returns (bool); function pauseCapital() external; function unpauseCapital() external; function capitalPaused() external view returns (bool); function transferToken(address _asset, uint256 _amount) external; function priceUnitMint(address asset) external view returns (uint256); function priceUnitRedeem(address asset) external view returns (uint256); function withdrawAllFromStrategy(address _strategyAddr) external; function withdrawAllFromStrategies() external; function withdrawFromStrategy( address _strategyFromAddress, address[] calldata _assets, uint256[] calldata _amounts ) external; function depositToStrategy( address _strategyToAddress, address[] calldata _assets, uint256[] calldata _amounts ) external; // VaultCore.sol function mint( address _asset, uint256 _amount, uint256 _minimumOusdAmount ) external; function mintForStrategy(uint256 _amount) external; function redeem(uint256 _amount, uint256 _minimumUnitAmount) external; function burnForStrategy(uint256 _amount) external; function redeemAll(uint256 _minimumUnitAmount) external; function allocate() external; function rebase() external; function swapCollateral( address fromAsset, address toAsset, uint256 fromAssetAmount, uint256 minToAssetAmount, bytes calldata data ) external returns (uint256 toAssetAmount); function totalValue() external view returns (uint256 value); function checkBalance(address _asset) external view returns (uint256); function calculateRedeemOutputs(uint256 _amount) external view returns (uint256[] memory); function getAssetCount() external view returns (uint256); function getAssetConfig(address _asset) external view returns (VaultStorage.Asset memory config); function getAllAssets() external view returns (address[] memory); function getStrategyCount() external view returns (uint256); function swapper() external view returns (address); function allowedSwapUndervalue() external view returns (uint256); function getAllStrategies() external view returns (address[] memory); function isSupportedAsset(address _asset) external view returns (bool); function netOusdMintForStrategyThreshold() external view returns (uint256); function setOusdMetaStrategy(address _ousdMetaStrategy) external; function setNetOusdMintForStrategyThreshold(uint256 _threshold) external; function netOusdMintedForStrategy() external view returns (int256); function setDripper(address _dripper) external; function dripper() external view returns (address); function weth() external view returns (address); function cacheWETHAssetIndex() external; function wethAssetIndex() external view returns (uint256); function initialize(address, address) external; function setAdminImpl(address) external; function removeAsset(address _asset) external; // These are OETH specific functions function addWithdrawalQueueLiquidity() external; function requestWithdrawal(uint256 _amount) external returns (uint256 requestId, uint256 queued); function claimWithdrawal(uint256 requestId) external returns (uint256 amount); function claimWithdrawals(uint256[] memory requestIds) external returns (uint256[] memory amounts, uint256 totalAmount); function withdrawalQueueMetadata() external view returns (VaultStorage.WithdrawalQueueMetadata memory); function withdrawalRequests(uint256 requestId) external view returns (VaultStorage.WithdrawalRequest memory); // OETHb specific functions function addStrategyToMintWhitelist(address strategyAddr) external; function removeStrategyFromMintWhitelist(address strategyAddr) external; function isMintWhitelistedStrategy(address strategyAddr) external view returns (bool); function withdrawalClaimDelay() external view returns (uint256); function setWithdrawalClaimDelay(uint256 newDelay) external; }
// SPDX-License-Identifier: BUSL-1.1 pragma solidity ^0.8.0; /** * @title OUSD Token Contract * @dev ERC20 compatible contract for OUSD * @dev Implements an elastic supply * @author Origin Protocol Inc */ import { Governable } from "../governance/Governable.sol"; import { SafeCast } from "@openzeppelin/contracts/utils/math/SafeCast.sol"; contract OUSD is Governable { using SafeCast for int256; using SafeCast for uint256; /// @dev Event triggered when the supply changes /// @param totalSupply Updated token total supply /// @param rebasingCredits Updated token rebasing credits /// @param rebasingCreditsPerToken Updated token rebasing credits per token event TotalSupplyUpdatedHighres( uint256 totalSupply, uint256 rebasingCredits, uint256 rebasingCreditsPerToken ); /// @dev Event triggered when an account opts in for rebasing /// @param account Address of the account event AccountRebasingEnabled(address account); /// @dev Event triggered when an account opts out of rebasing /// @param account Address of the account event AccountRebasingDisabled(address account); /// @dev Emitted when `value` tokens are moved from one account `from` to /// another `to`. /// @param from Address of the account tokens are moved from /// @param to Address of the account tokens are moved to /// @param value Amount of tokens transferred 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. /// @param owner Address of the owner approving allowance /// @param spender Address of the spender allowance is granted to /// @param value Amount of tokens spender can transfer event Approval( address indexed owner, address indexed spender, uint256 value ); /// @dev Yield resulting from {changeSupply} that a `source` account would /// receive is directed to `target` account. /// @param source Address of the source forwarding the yield /// @param target Address of the target receiving the yield event YieldDelegated(address source, address target); /// @dev Yield delegation from `source` account to the `target` account is /// suspended. /// @param source Address of the source suspending yield forwarding /// @param target Address of the target no longer receiving yield from `source` /// account event YieldUndelegated(address source, address target); enum RebaseOptions { NotSet, StdNonRebasing, StdRebasing, YieldDelegationSource, YieldDelegationTarget } uint256[154] private _gap; // Slots to align with deployed contract uint256 private constant MAX_SUPPLY = type(uint128).max; /// @dev The amount of tokens in existence uint256 public totalSupply; mapping(address => mapping(address => uint256)) private allowances; /// @dev The vault with privileges to execute {mint}, {burn} /// and {changeSupply} address public vaultAddress; mapping(address => uint256) internal creditBalances; // the 2 storage variables below need trailing underscores to not name collide with public functions uint256 private rebasingCredits_; // Sum of all rebasing credits (creditBalances for rebasing accounts) uint256 private rebasingCreditsPerToken_; /// @dev The amount of tokens that are not rebasing - receiving yield uint256 public nonRebasingSupply; mapping(address => uint256) internal alternativeCreditsPerToken; /// @dev A map of all addresses and their respective RebaseOptions mapping(address => RebaseOptions) public rebaseState; mapping(address => uint256) private __deprecated_isUpgraded; /// @dev A map of addresses that have yields forwarded to. This is an /// inverse mapping of {yieldFrom} /// Key Account forwarding yield /// Value Account receiving yield mapping(address => address) public yieldTo; /// @dev A map of addresses that are receiving the yield. This is an /// inverse mapping of {yieldTo} /// Key Account receiving yield /// Value Account forwarding yield mapping(address => address) public yieldFrom; uint256 private constant RESOLUTION_INCREASE = 1e9; uint256[34] private __gap; // including below gap totals up to 200 /// @dev Initializes the contract and sets necessary variables. /// @param _vaultAddress Address of the vault contract /// @param _initialCreditsPerToken The starting rebasing credits per token. function initialize(address _vaultAddress, uint256 _initialCreditsPerToken) external onlyGovernor { require(_vaultAddress != address(0), "Zero vault address"); require(vaultAddress == address(0), "Already initialized"); rebasingCreditsPerToken_ = _initialCreditsPerToken; vaultAddress = _vaultAddress; } /// @dev Returns the symbol of the token, a shorter version /// of the name. function symbol() external pure virtual returns (string memory) { return "OUSD"; } /// @dev Returns the name of the token. function name() external pure virtual returns (string memory) { return "Origin Dollar"; } /// @dev Returns the number of decimals used to get its user representation. function decimals() external pure virtual returns (uint8) { return 18; } /** * @dev Verifies that the caller is the Vault contract */ modifier onlyVault() { require(vaultAddress == msg.sender, "Caller is not the Vault"); _; } /** * @return High resolution rebasingCreditsPerToken */ function rebasingCreditsPerTokenHighres() external view returns (uint256) { return rebasingCreditsPerToken_; } /** * @return Low resolution rebasingCreditsPerToken */ function rebasingCreditsPerToken() external view returns (uint256) { return rebasingCreditsPerToken_ / RESOLUTION_INCREASE; } /** * @return High resolution total number of rebasing credits */ function rebasingCreditsHighres() external view returns (uint256) { return rebasingCredits_; } /** * @return Low resolution total number of rebasing credits */ function rebasingCredits() external view returns (uint256) { return rebasingCredits_ / RESOLUTION_INCREASE; } /** * @notice Gets the balance of the specified address. * @param _account Address to query the balance of. * @return A uint256 representing the amount of base units owned by the * specified address. */ function balanceOf(address _account) public view returns (uint256) { RebaseOptions state = rebaseState[_account]; if (state == RebaseOptions.YieldDelegationSource) { // Saves a slot read when transferring to or from a yield delegating source // since we know creditBalances equals the balance. return creditBalances[_account]; } uint256 baseBalance = (creditBalances[_account] * 1e18) / _creditsPerToken(_account); if (state == RebaseOptions.YieldDelegationTarget) { // creditBalances of yieldFrom accounts equals token balances return baseBalance - creditBalances[yieldFrom[_account]]; } return baseBalance; } /** * @notice Gets the credits balance of the specified address. * @dev Backwards compatible with old low res credits per token. * @param _account The address to query the balance of. * @return (uint256, uint256) Credit balance and credits per token of the * address */ function creditsBalanceOf(address _account) external view returns (uint256, uint256) { uint256 cpt = _creditsPerToken(_account); if (cpt == 1e27) { // For a period before the resolution upgrade, we created all new // contract accounts at high resolution. Since they are not changing // as a result of this upgrade, we will return their true values return (creditBalances[_account], cpt); } else { return ( creditBalances[_account] / RESOLUTION_INCREASE, cpt / RESOLUTION_INCREASE ); } } /** * @notice Gets the credits balance of the specified address. * @param _account The address to query the balance of. * @return (uint256, uint256, bool) Credit balance, credits per token of the * address, and isUpgraded */ function creditsBalanceOfHighres(address _account) external view returns ( uint256, uint256, bool ) { return ( creditBalances[_account], _creditsPerToken(_account), true // all accounts have their resolution "upgraded" ); } // Backwards compatible view function nonRebasingCreditsPerToken(address _account) external view returns (uint256) { return alternativeCreditsPerToken[_account]; } /** * @notice Transfer tokens to a specified address. * @param _to the address to transfer to. * @param _value the amount to be transferred. * @return true on success. */ function transfer(address _to, uint256 _value) external returns (bool) { require(_to != address(0), "Transfer to zero address"); _executeTransfer(msg.sender, _to, _value); emit Transfer(msg.sender, _to, _value); return true; } /** * @notice Transfer tokens from one address to another. * @param _from The address you want to send tokens from. * @param _to The address you want to transfer to. * @param _value The amount of tokens to be transferred. * @return true on success. */ function transferFrom( address _from, address _to, uint256 _value ) external returns (bool) { require(_to != address(0), "Transfer to zero address"); uint256 userAllowance = allowances[_from][msg.sender]; require(_value <= userAllowance, "Allowance exceeded"); unchecked { allowances[_from][msg.sender] = userAllowance - _value; } _executeTransfer(_from, _to, _value); emit Transfer(_from, _to, _value); return true; } function _executeTransfer( address _from, address _to, uint256 _value ) internal { ( int256 fromRebasingCreditsDiff, int256 fromNonRebasingSupplyDiff ) = _adjustAccount(_from, -_value.toInt256()); ( int256 toRebasingCreditsDiff, int256 toNonRebasingSupplyDiff ) = _adjustAccount(_to, _value.toInt256()); _adjustGlobals( fromRebasingCreditsDiff + toRebasingCreditsDiff, fromNonRebasingSupplyDiff + toNonRebasingSupplyDiff ); } function _adjustAccount(address _account, int256 _balanceChange) internal returns (int256 rebasingCreditsDiff, int256 nonRebasingSupplyDiff) { RebaseOptions state = rebaseState[_account]; int256 currentBalance = balanceOf(_account).toInt256(); if (currentBalance + _balanceChange < 0) { revert("Transfer amount exceeds balance"); } uint256 newBalance = (currentBalance + _balanceChange).toUint256(); if (state == RebaseOptions.YieldDelegationSource) { address target = yieldTo[_account]; uint256 targetOldBalance = balanceOf(target); uint256 targetNewCredits = _balanceToRebasingCredits( targetOldBalance + newBalance ); rebasingCreditsDiff = targetNewCredits.toInt256() - creditBalances[target].toInt256(); creditBalances[_account] = newBalance; creditBalances[target] = targetNewCredits; } else if (state == RebaseOptions.YieldDelegationTarget) { uint256 newCredits = _balanceToRebasingCredits( newBalance + creditBalances[yieldFrom[_account]] ); rebasingCreditsDiff = newCredits.toInt256() - creditBalances[_account].toInt256(); creditBalances[_account] = newCredits; } else { _autoMigrate(_account); uint256 alternativeCreditsPerTokenMem = alternativeCreditsPerToken[ _account ]; if (alternativeCreditsPerTokenMem > 0) { nonRebasingSupplyDiff = _balanceChange; if (alternativeCreditsPerTokenMem != 1e18) { alternativeCreditsPerToken[_account] = 1e18; } creditBalances[_account] = newBalance; } else { uint256 newCredits = _balanceToRebasingCredits(newBalance); rebasingCreditsDiff = newCredits.toInt256() - creditBalances[_account].toInt256(); creditBalances[_account] = newCredits; } } } function _adjustGlobals( int256 _rebasingCreditsDiff, int256 _nonRebasingSupplyDiff ) internal { if (_rebasingCreditsDiff != 0) { rebasingCredits_ = (rebasingCredits_.toInt256() + _rebasingCreditsDiff).toUint256(); } if (_nonRebasingSupplyDiff != 0) { nonRebasingSupply = (nonRebasingSupply.toInt256() + _nonRebasingSupplyDiff).toUint256(); } } /** * @notice Function to check the amount of tokens that _owner has allowed * to `_spender`. * @param _owner The address which owns the funds. * @param _spender The address which will spend the funds. * @return The number of tokens still available for the _spender. */ function allowance(address _owner, address _spender) external view returns (uint256) { return allowances[_owner][_spender]; } /** * @notice Approve the passed address to spend the specified amount of * tokens on behalf of msg.sender. * @param _spender The address which will spend the funds. * @param _value The amount of tokens to be spent. * @return true on success. */ function approve(address _spender, uint256 _value) external returns (bool) { allowances[msg.sender][_spender] = _value; emit Approval(msg.sender, _spender, _value); return true; } /** * @notice Creates `_amount` tokens and assigns them to `_account`, * increasing the total supply. */ function mint(address _account, uint256 _amount) external onlyVault { require(_account != address(0), "Mint to the zero address"); // Account ( int256 toRebasingCreditsDiff, int256 toNonRebasingSupplyDiff ) = _adjustAccount(_account, _amount.toInt256()); // Globals _adjustGlobals(toRebasingCreditsDiff, toNonRebasingSupplyDiff); totalSupply = totalSupply + _amount; require(totalSupply < MAX_SUPPLY, "Max supply"); emit Transfer(address(0), _account, _amount); } /** * @notice Destroys `_amount` tokens from `_account`, * reducing the total supply. */ function burn(address _account, uint256 _amount) external onlyVault { require(_account != address(0), "Burn from the zero address"); if (_amount == 0) { return; } // Account ( int256 toRebasingCreditsDiff, int256 toNonRebasingSupplyDiff ) = _adjustAccount(_account, -_amount.toInt256()); // Globals _adjustGlobals(toRebasingCreditsDiff, toNonRebasingSupplyDiff); totalSupply = totalSupply - _amount; emit Transfer(_account, address(0), _amount); } /** * @dev Get the credits per token for an account. Returns a fixed amount * if the account is non-rebasing. * @param _account Address of the account. */ function _creditsPerToken(address _account) internal view returns (uint256) { uint256 alternativeCreditsPerTokenMem = alternativeCreditsPerToken[ _account ]; if (alternativeCreditsPerTokenMem != 0) { return alternativeCreditsPerTokenMem; } else { return rebasingCreditsPerToken_; } } /** * @dev Auto migrate contracts to be non rebasing, * unless they have opted into yield. * @param _account Address of the account. */ function _autoMigrate(address _account) internal { bool isContract = _account.code.length > 0; // In previous code versions, contracts would not have had their // rebaseState[_account] set to RebaseOptions.NonRebasing when migrated // therefore we check the actual accounting used on the account instead. if ( isContract && rebaseState[_account] == RebaseOptions.NotSet && alternativeCreditsPerToken[_account] == 0 ) { _rebaseOptOut(_account); } } /** * @dev Calculates credits from contract's global rebasingCreditsPerToken_, and * also balance that corresponds to those credits. The latter is important * when adjusting the contract's global nonRebasingSupply to circumvent any * possible rounding errors. * * @param _balance Balance of the account. */ function _balanceToRebasingCredits(uint256 _balance) internal view returns (uint256 rebasingCredits) { // Rounds up, because we need to ensure that accounts always have // at least the balance that they should have. // Note this should always be used on an absolute account value, // not on a possibly negative diff, because then the rounding would be wrong. return ((_balance) * rebasingCreditsPerToken_ + 1e18 - 1) / 1e18; } /** * @notice The calling account will start receiving yield after a successful call. * @param _account Address of the account. */ function governanceRebaseOptIn(address _account) external onlyGovernor { require(_account != address(0), "Zero address not allowed"); _rebaseOptIn(_account); } /** * @notice The calling account will start receiving yield after a successful call. */ function rebaseOptIn() external { _rebaseOptIn(msg.sender); } function _rebaseOptIn(address _account) internal { uint256 balance = balanceOf(_account); // prettier-ignore require( alternativeCreditsPerToken[_account] > 0 || // Accounts may explicitly `rebaseOptIn` regardless of // accounting if they have a 0 balance. creditBalances[_account] == 0 , "Account must be non-rebasing" ); RebaseOptions state = rebaseState[_account]; // prettier-ignore require( state == RebaseOptions.StdNonRebasing || state == RebaseOptions.NotSet, "Only standard non-rebasing accounts can opt in" ); uint256 newCredits = _balanceToRebasingCredits(balance); // Account rebaseState[_account] = RebaseOptions.StdRebasing; alternativeCreditsPerToken[_account] = 0; creditBalances[_account] = newCredits; // Globals _adjustGlobals(newCredits.toInt256(), -balance.toInt256()); emit AccountRebasingEnabled(_account); } /** * @notice The calling account will no longer receive yield */ function rebaseOptOut() external { _rebaseOptOut(msg.sender); } function _rebaseOptOut(address _account) internal { require( alternativeCreditsPerToken[_account] == 0, "Account must be rebasing" ); RebaseOptions state = rebaseState[_account]; require( state == RebaseOptions.StdRebasing || state == RebaseOptions.NotSet, "Only standard rebasing accounts can opt out" ); uint256 oldCredits = creditBalances[_account]; uint256 balance = balanceOf(_account); // Account rebaseState[_account] = RebaseOptions.StdNonRebasing; alternativeCreditsPerToken[_account] = 1e18; creditBalances[_account] = balance; // Globals _adjustGlobals(-oldCredits.toInt256(), balance.toInt256()); emit AccountRebasingDisabled(_account); } /** * @notice Distribute yield to users. This changes the exchange rate * between "credits" and OUSD tokens to change rebasing user's balances. * @param _newTotalSupply New total supply of OUSD. */ function changeSupply(uint256 _newTotalSupply) external onlyVault { require(totalSupply > 0, "Cannot increase 0 supply"); if (totalSupply == _newTotalSupply) { emit TotalSupplyUpdatedHighres( totalSupply, rebasingCredits_, rebasingCreditsPerToken_ ); return; } totalSupply = _newTotalSupply > MAX_SUPPLY ? MAX_SUPPLY : _newTotalSupply; uint256 rebasingSupply = totalSupply - nonRebasingSupply; // round up in the favour of the protocol rebasingCreditsPerToken_ = (rebasingCredits_ * 1e18 + rebasingSupply - 1) / rebasingSupply; require(rebasingCreditsPerToken_ > 0, "Invalid change in supply"); emit TotalSupplyUpdatedHighres( totalSupply, rebasingCredits_, rebasingCreditsPerToken_ ); } /* * @notice Send the yield from one account to another account. * Each account keeps its own balances. */ function delegateYield(address _from, address _to) external onlyGovernor { require(_from != address(0), "Zero from address not allowed"); require(_to != address(0), "Zero to address not allowed"); require(_from != _to, "Cannot delegate to self"); require( yieldFrom[_to] == address(0) && yieldTo[_to] == address(0) && yieldFrom[_from] == address(0) && yieldTo[_from] == address(0), "Blocked by existing yield delegation" ); RebaseOptions stateFrom = rebaseState[_from]; RebaseOptions stateTo = rebaseState[_to]; require( stateFrom == RebaseOptions.NotSet || stateFrom == RebaseOptions.StdNonRebasing || stateFrom == RebaseOptions.StdRebasing, "Invalid rebaseState from" ); require( stateTo == RebaseOptions.NotSet || stateTo == RebaseOptions.StdNonRebasing || stateTo == RebaseOptions.StdRebasing, "Invalid rebaseState to" ); if (alternativeCreditsPerToken[_from] == 0) { _rebaseOptOut(_from); } if (alternativeCreditsPerToken[_to] > 0) { _rebaseOptIn(_to); } uint256 fromBalance = balanceOf(_from); uint256 toBalance = balanceOf(_to); uint256 oldToCredits = creditBalances[_to]; uint256 newToCredits = _balanceToRebasingCredits( fromBalance + toBalance ); // Set up the bidirectional links yieldTo[_from] = _to; yieldFrom[_to] = _from; // Local rebaseState[_from] = RebaseOptions.YieldDelegationSource; alternativeCreditsPerToken[_from] = 1e18; creditBalances[_from] = fromBalance; rebaseState[_to] = RebaseOptions.YieldDelegationTarget; creditBalances[_to] = newToCredits; // Global int256 creditsChange = newToCredits.toInt256() - oldToCredits.toInt256(); _adjustGlobals(creditsChange, -(fromBalance).toInt256()); emit YieldDelegated(_from, _to); } /* * @notice Stop sending the yield from one account to another account. */ function undelegateYield(address _from) external onlyGovernor { // Require a delegation, which will also ensure a valid delegation require(yieldTo[_from] != address(0), "Zero address not allowed"); address to = yieldTo[_from]; uint256 fromBalance = balanceOf(_from); uint256 toBalance = balanceOf(to); uint256 oldToCredits = creditBalances[to]; uint256 newToCredits = _balanceToRebasingCredits(toBalance); // Remove the bidirectional links yieldFrom[to] = address(0); yieldTo[_from] = address(0); // Local rebaseState[_from] = RebaseOptions.StdNonRebasing; // alternativeCreditsPerToken[from] already 1e18 from `delegateYield()` creditBalances[_from] = fromBalance; rebaseState[to] = RebaseOptions.StdRebasing; // alternativeCreditsPerToken[to] already 0 from `delegateYield()` creditBalances[to] = newToCredits; // Global int256 creditsChange = newToCredits.toInt256() - oldToCredits.toInt256(); _adjustGlobals(creditsChange, fromBalance.toInt256()); emit YieldUndelegated(_from, to); } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import { IBasicToken } from "../interfaces/IBasicToken.sol"; library Helpers { /** * @notice Fetch the `symbol()` from an ERC20 token * @dev Grabs the `symbol()` from a contract * @param _token Address of the ERC20 token * @return string Symbol of the ERC20 token */ function getSymbol(address _token) internal view returns (string memory) { string memory symbol = IBasicToken(_token).symbol(); return symbol; } /** * @notice Fetch the `decimals()` from an ERC20 token * @dev Grabs the `decimals()` from a contract and fails if * the decimal value does not live within a certain range * @param _token Address of the ERC20 token * @return uint256 Decimals of the ERC20 token */ function getDecimals(address _token) internal view returns (uint256) { uint256 decimals = IBasicToken(_token).decimals(); require( decimals >= 4 && decimals <= 18, "Token must have sufficient decimal places" ); return decimals; } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; /** * @title Base contract any contracts that need to initialize state after deployment. * @author Origin Protocol Inc */ abstract contract Initializable { /** * @dev Indicates that the contract has been initialized. */ bool private initialized; /** * @dev Indicates that the contract is in the process of being initialized. */ bool private initializing; /** * @dev Modifier to protect an initializer function from being invoked twice. */ modifier initializer() { require( initializing || !initialized, "Initializable: contract is already initialized" ); bool isTopLevelCall = !initializing; if (isTopLevelCall) { initializing = true; initialized = true; } _; if (isTopLevelCall) { initializing = false; } } uint256[50] private ______gap; }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import { SafeMath } from "@openzeppelin/contracts/utils/math/SafeMath.sol"; // Based on StableMath from Stability Labs Pty. Ltd. // https://github.com/mstable/mStable-contracts/blob/master/contracts/shared/StableMath.sol library StableMath { using SafeMath for uint256; /** * @dev Scaling unit for use in specific calculations, * where 1 * 10**18, or 1e18 represents a unit '1' */ uint256 private constant FULL_SCALE = 1e18; /*************************************** Helpers ****************************************/ /** * @dev Adjust the scale of an integer * @param to Decimals to scale to * @param from Decimals to scale from */ function scaleBy( uint256 x, uint256 to, uint256 from ) internal pure returns (uint256) { if (to > from) { x = x.mul(10**(to - from)); } else if (to < from) { // slither-disable-next-line divide-before-multiply x = x.div(10**(from - to)); } return x; } /*************************************** Precise Arithmetic ****************************************/ /** * @dev Multiplies two precise units, and then truncates by the full scale * @param x Left hand input to multiplication * @param y Right hand input to multiplication * @return Result after multiplying the two inputs and then dividing by the shared * scale unit */ function mulTruncate(uint256 x, uint256 y) internal pure returns (uint256) { return mulTruncateScale(x, y, FULL_SCALE); } /** * @dev Multiplies two precise units, and then truncates by the given scale. For example, * when calculating 90% of 10e18, (10e18 * 9e17) / 1e18 = (9e36) / 1e18 = 9e18 * @param x Left hand input to multiplication * @param y Right hand input to multiplication * @param scale Scale unit * @return Result after multiplying the two inputs and then dividing by the shared * scale unit */ function mulTruncateScale( uint256 x, uint256 y, uint256 scale ) internal pure returns (uint256) { // e.g. assume scale = fullScale // z = 10e18 * 9e17 = 9e36 uint256 z = x.mul(y); // return 9e36 / 1e18 = 9e18 return z.div(scale); } /** * @dev Multiplies two precise units, and then truncates by the full scale, rounding up the result * @param x Left hand input to multiplication * @param y Right hand input to multiplication * @return Result after multiplying the two inputs and then dividing by the shared * scale unit, rounded up to the closest base unit. */ function mulTruncateCeil(uint256 x, uint256 y) internal pure returns (uint256) { // e.g. 8e17 * 17268172638 = 138145381104e17 uint256 scaled = x.mul(y); // e.g. 138145381104e17 + 9.99...e17 = 138145381113.99...e17 uint256 ceil = scaled.add(FULL_SCALE.sub(1)); // e.g. 13814538111.399...e18 / 1e18 = 13814538111 return ceil.div(FULL_SCALE); } /** * @dev Precisely divides two units, by first scaling the left hand operand. Useful * for finding percentage weightings, i.e. 8e18/10e18 = 80% (or 8e17) * @param x Left hand input to division * @param y Right hand input to division * @return Result after multiplying the left operand by the scale, and * executing the division on the right hand input. */ function divPrecisely(uint256 x, uint256 y) internal pure returns (uint256) { // e.g. 8e18 * 1e18 = 8e36 uint256 z = x.mul(FULL_SCALE); // e.g. 8e36 / 10e18 = 8e17 return z.div(y); } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import { SafeERC20 } from "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol"; import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol"; import { IStrategy } from "../interfaces/IStrategy.sol"; import { IVault } from "../interfaces/IVault.sol"; import { VaultAdmin } from "./VaultAdmin.sol"; /** * @title OETH VaultAdmin Contract * @author Origin Protocol Inc */ contract OETHVaultAdmin is VaultAdmin { using SafeERC20 for IERC20; address public immutable weth; constructor(address _weth) { weth = _weth; } /// @dev Simplified version of the deposit function as WETH is the only supported asset. function _depositToStrategy( address _strategyToAddress, address[] calldata _assets, uint256[] calldata _amounts ) internal override { require( strategies[_strategyToAddress].isSupported, "Invalid to Strategy" ); require( _assets.length == 1 && _amounts.length == 1 && _assets[0] == weth, "Only WETH is supported" ); // Check the there is enough WETH to transfer once the WETH reserved for the withdrawal queue is accounted for require(_amounts[0] <= _wethAvailable(), "Not enough WETH available"); // Send required amount of funds to the strategy IERC20(weth).safeTransfer(_strategyToAddress, _amounts[0]); // Deposit all the funds that have been sent to the strategy IStrategy(_strategyToAddress).depositAll(); } function _withdrawFromStrategy( address _recipient, address _strategyFromAddress, address[] calldata _assets, uint256[] calldata _amounts ) internal override { super._withdrawFromStrategy( _recipient, _strategyFromAddress, _assets, _amounts ); IVault(address(this)).addWithdrawalQueueLiquidity(); } function _withdrawAllFromStrategy(address _strategyAddr) internal override { super._withdrawAllFromStrategy(_strategyAddr); IVault(address(this)).addWithdrawalQueueLiquidity(); } function _withdrawAllFromStrategies() internal override { super._withdrawAllFromStrategies(); IVault(address(this)).addWithdrawalQueueLiquidity(); } /// @dev Calculate how much WETH in the vault is not reserved for the withdrawal queue. // That is, it is available to be redeemed or deposited into a strategy. function _wethAvailable() internal view returns (uint256 wethAvailable) { WithdrawalQueueMetadata memory queue = withdrawalQueueMetadata; // The amount of WETH that is still to be claimed in the withdrawal queue uint256 outstandingWithdrawals = queue.queued - queue.claimed; // The amount of sitting in WETH in the vault uint256 wethBalance = IERC20(weth).balanceOf(address(this)); // If there is not enough WETH in the vault to cover the outstanding withdrawals if (wethBalance <= outstandingWithdrawals) { return 0; } return wethBalance - outstandingWithdrawals; } function _swapCollateral( address, address, uint256, uint256, bytes calldata ) internal pure override returns (uint256) { revert("Collateral swap not supported"); } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; /** * @title OToken VaultAdmin contract * @notice The VaultAdmin contract makes configuration and admin calls on the vault. * @author Origin Protocol Inc */ import { SafeERC20 } from "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol"; import { IOracle } from "../interfaces/IOracle.sol"; import { ISwapper } from "../interfaces/ISwapper.sol"; import { IVault } from "../interfaces/IVault.sol"; import { StableMath } from "../utils/StableMath.sol"; import "./VaultStorage.sol"; contract VaultAdmin is VaultStorage { using SafeERC20 for IERC20; using StableMath for uint256; /** * @dev Verifies that the caller is the Governor or Strategist. */ modifier onlyGovernorOrStrategist() { require( msg.sender == strategistAddr || isGovernor(), "Caller is not the Strategist or Governor" ); _; } /*************************************** Configuration ****************************************/ /** * @notice Set address of price provider. * @param _priceProvider Address of price provider */ function setPriceProvider(address _priceProvider) external onlyGovernor { priceProvider = _priceProvider; emit PriceProviderUpdated(_priceProvider); } /** * @notice Set a fee in basis points to be charged for a redeem. * @param _redeemFeeBps Basis point fee to be charged */ function setRedeemFeeBps(uint256 _redeemFeeBps) external onlyGovernor { require(_redeemFeeBps <= 1000, "Redeem fee should not be over 10%"); redeemFeeBps = _redeemFeeBps; emit RedeemFeeUpdated(_redeemFeeBps); } /** * @notice Set a buffer of assets to keep in the Vault to handle most * redemptions without needing to spend gas unwinding assets from a Strategy. * @param _vaultBuffer Percentage using 18 decimals. 100% = 1e18. */ function setVaultBuffer(uint256 _vaultBuffer) external onlyGovernorOrStrategist { require(_vaultBuffer <= 1e18, "Invalid value"); vaultBuffer = _vaultBuffer; emit VaultBufferUpdated(_vaultBuffer); } /** * @notice Sets the minimum amount of OTokens in a mint to trigger an * automatic allocation of funds afterwords. * @param _threshold OToken amount with 18 fixed decimals. */ function setAutoAllocateThreshold(uint256 _threshold) external onlyGovernor { autoAllocateThreshold = _threshold; emit AllocateThresholdUpdated(_threshold); } /** * @notice Set a minimum amount of OTokens in a mint or redeem that triggers a * rebase * @param _threshold OToken amount with 18 fixed decimals. */ function setRebaseThreshold(uint256 _threshold) external onlyGovernor { rebaseThreshold = _threshold; emit RebaseThresholdUpdated(_threshold); } /** * @notice Set address of Strategist * @param _address Address of Strategist */ function setStrategistAddr(address _address) external onlyGovernor { strategistAddr = _address; emit StrategistUpdated(_address); } /** * @notice Set the default Strategy for an asset, i.e. the one which the asset will be automatically allocated to and withdrawn from * @param _asset Address of the asset * @param _strategy Address of the Strategy */ function setAssetDefaultStrategy(address _asset, address _strategy) external onlyGovernorOrStrategist { emit AssetDefaultStrategyUpdated(_asset, _strategy); // If its a zero address being passed for the strategy we are removing // the default strategy if (_strategy != address(0)) { // Make sure the strategy meets some criteria require(strategies[_strategy].isSupported, "Strategy not approved"); IStrategy strategy = IStrategy(_strategy); require(assets[_asset].isSupported, "Asset is not supported"); require( strategy.supportsAsset(_asset), "Asset not supported by Strategy" ); } assetDefaultStrategies[_asset] = _strategy; } /** * @notice Set maximum amount of OTokens that can at any point be minted and deployed * to strategy (used only by ConvexOUSDMetaStrategy for now). * @param _threshold OToken amount with 18 fixed decimals. */ function setNetOusdMintForStrategyThreshold(uint256 _threshold) external onlyGovernor { /** * Because `netOusdMintedForStrategy` check in vault core works both ways * (positive and negative) the actual impact of the amount of OToken minted * could be double the threshold. E.g.: * - contract has threshold set to 100 * - state of netOusdMinted is -90 * - in effect it can mint 190 OToken and still be within limits * * We are somewhat mitigating this behaviour by resetting the netOusdMinted * counter whenever new threshold is set. So it can only move one threshold * amount in each direction. This also enables us to reduce the threshold * amount and not have problems with current netOusdMinted being near * limits on either side. */ netOusdMintedForStrategy = 0; netOusdMintForStrategyThreshold = _threshold; emit NetOusdMintForStrategyThresholdChanged(_threshold); } /** * @notice Set the Dripper contract that streams harvested rewards to the vault. * @param _dripper Address of the Dripper contract. */ function setDripper(address _dripper) external onlyGovernor { dripper = _dripper; emit DripperChanged(_dripper); } /** * @notice Changes the async withdrawal claim period for OETH & superOETHb * @param _delay Delay period (should be between 10 mins to 7 days). * Set to 0 to disable async withdrawals */ function setWithdrawalClaimDelay(uint256 _delay) external onlyGovernor { require( _delay == 0 || (_delay >= 10 minutes && _delay <= 15 days), "Invalid claim delay period" ); withdrawalClaimDelay = _delay; emit WithdrawalClaimDelayUpdated(_delay); } /*************************************** Swaps ****************************************/ /** * @notice Strategist swaps collateral assets sitting in the vault. * @param _fromAsset The token address of the asset being sold by the vault. * @param _toAsset The token address of the asset being purchased by the vault. * @param _fromAssetAmount The amount of assets being sold by the vault. * @param _minToAssetAmount The minimum amount of assets to be purchased. * @param _data implementation specific data. eg 1Inch swap data * @return toAssetAmount The amount of toAssets that was received from the swap */ function swapCollateral( address _fromAsset, address _toAsset, uint256 _fromAssetAmount, uint256 _minToAssetAmount, bytes calldata _data ) external nonReentrant onlyGovernorOrStrategist returns (uint256 toAssetAmount) { toAssetAmount = _swapCollateral( _fromAsset, _toAsset, _fromAssetAmount, _minToAssetAmount, _data ); } function _swapCollateral( address _fromAsset, address _toAsset, uint256 _fromAssetAmount, uint256 _minToAssetAmount, bytes calldata _data ) internal virtual returns (uint256 toAssetAmount) { // Check fromAsset and toAsset are valid Asset memory fromAssetConfig = assets[_fromAsset]; Asset memory toAssetConfig = assets[_toAsset]; require(fromAssetConfig.isSupported, "From asset is not supported"); require(toAssetConfig.isSupported, "To asset is not supported"); // Load swap config into memory to avoid separate SLOADs SwapConfig memory config = swapConfig; // Scope a new block to remove toAssetBalBefore from the scope of swapCollateral. // This avoids a stack too deep error. { uint256 toAssetBalBefore = IERC20(_toAsset).balanceOf( address(this) ); // Transfer from assets to the swapper contract IERC20(_fromAsset).safeTransfer(config.swapper, _fromAssetAmount); // Call to the Swapper contract to do the actual swap // The -1 is required for stETH which sometimes transfers 1 wei less than what was specified. // slither-disable-next-line unused-return ISwapper(config.swapper).swap( _fromAsset, _toAsset, _fromAssetAmount - 1, _minToAssetAmount, _data ); // Compute the change in asset balance held by the Vault toAssetAmount = IERC20(_toAsset).balanceOf(address(this)) - toAssetBalBefore; } // Check the to assets returned is above slippage amount specified by the strategist require( toAssetAmount >= _minToAssetAmount, "Strategist slippage limit" ); // Scope a new block to remove minOracleToAssetAmount from the scope of swapCollateral. // This avoids a stack too deep error. { // Check the slippage against the Oracle in case the strategist made a mistake or has become malicious. // to asset amount = from asset amount * from asset price / to asset price uint256 minOracleToAssetAmount = (_fromAssetAmount * (1e4 - fromAssetConfig.allowedOracleSlippageBps) * IOracle(priceProvider).price(_fromAsset)) / (IOracle(priceProvider).price(_toAsset) * (1e4 + toAssetConfig.allowedOracleSlippageBps)); // Scale both sides up to 18 decimals to compare require( toAssetAmount.scaleBy(18, toAssetConfig.decimals) >= minOracleToAssetAmount.scaleBy( 18, fromAssetConfig.decimals ), "Oracle slippage limit exceeded" ); } // Check the vault's total value hasn't gone below the OToken total supply // by more than the allowed percentage. require( IVault(address(this)).totalValue() >= (oUSD.totalSupply() * ((1e4 - config.allowedUndervalueBps))) / 1e4, "Allowed value < supply" ); emit Swapped(_fromAsset, _toAsset, _fromAssetAmount, toAssetAmount); } /*************************************** Swap Config ****************************************/ /** * @notice Set the contract the performs swaps of collateral assets. * @param _swapperAddr Address of the Swapper contract that implements the ISwapper interface. */ function setSwapper(address _swapperAddr) external onlyGovernor { swapConfig.swapper = _swapperAddr; emit SwapperChanged(_swapperAddr); } /// @notice Contract that swaps the vault's collateral assets function swapper() external view returns (address swapper_) { swapper_ = swapConfig.swapper; } /** * @notice Set max allowed percentage the vault total value can drop below the OToken total supply in basis points * when executing collateral swaps. * @param _basis Percentage in basis points. eg 100 == 1% */ function setSwapAllowedUndervalue(uint16 _basis) external onlyGovernor { require(_basis < 10001, "Invalid basis points"); swapConfig.allowedUndervalueBps = _basis; emit SwapAllowedUndervalueChanged(_basis); } /** * @notice Max allowed percentage the vault total value can drop below the OToken total supply in basis points * when executing a collateral swap. * For example 100 == 1% * @return value Percentage in basis points. */ function allowedSwapUndervalue() external view returns (uint256 value) { value = swapConfig.allowedUndervalueBps; } /** * @notice Set the allowed slippage from the Oracle price for collateral asset swaps. * @param _asset Address of the asset token. * @param _allowedOracleSlippageBps allowed slippage from Oracle in basis points. eg 20 = 0.2%. Max 10%. */ function setOracleSlippage(address _asset, uint16 _allowedOracleSlippageBps) external onlyGovernor { require(assets[_asset].isSupported, "Asset not supported"); require(_allowedOracleSlippageBps < 1000, "Slippage too high"); assets[_asset].allowedOracleSlippageBps = _allowedOracleSlippageBps; emit SwapSlippageChanged(_asset, _allowedOracleSlippageBps); } /*************************************** Asset Config ****************************************/ /** * @notice Add a supported asset to the contract, i.e. one that can be * to mint OTokens. * @param _asset Address of asset */ function supportAsset(address _asset, uint8 _unitConversion) external virtual onlyGovernor { require(!assets[_asset].isSupported, "Asset already supported"); assets[_asset] = Asset({ isSupported: true, unitConversion: UnitConversion(_unitConversion), decimals: 0, // will be overridden in _cacheDecimals allowedOracleSlippageBps: 0 // 0% by default }); _cacheDecimals(_asset); allAssets.push(_asset); // Verify that our oracle supports the asset // slither-disable-next-line unused-return IOracle(priceProvider).price(_asset); emit AssetSupported(_asset); } /** * @notice Remove a supported asset from the Vault * @param _asset Address of asset */ function removeAsset(address _asset) external onlyGovernor { require(assets[_asset].isSupported, "Asset not supported"); require( IVault(address(this)).checkBalance(_asset) <= 1e13, "Vault still holds asset" ); uint256 assetsCount = allAssets.length; uint256 assetIndex = assetsCount; // initialize at invaid index for (uint256 i = 0; i < assetsCount; ++i) { if (allAssets[i] == _asset) { assetIndex = i; break; } } // Note: If asset is not found in `allAssets`, the following line // will revert with an out-of-bound error. However, there's no // reason why an asset would have `Asset.isSupported = true` but // not exist in `allAssets`. // Update allAssets array allAssets[assetIndex] = allAssets[assetsCount - 1]; allAssets.pop(); // Reset default strategy assetDefaultStrategies[_asset] = address(0); emit AssetDefaultStrategyUpdated(_asset, address(0)); // Remove asset from storage delete assets[_asset]; emit AssetRemoved(_asset); } /** * @notice Cache decimals on OracleRouter for a particular asset. This action * is required before that asset's price can be accessed. * @param _asset Address of asset token */ function cacheDecimals(address _asset) external onlyGovernor { _cacheDecimals(_asset); } /*************************************** Strategy Config ****************************************/ /** * @notice Add a strategy to the Vault. * @param _addr Address of the strategy to add */ function approveStrategy(address _addr) external onlyGovernor { require(!strategies[_addr].isSupported, "Strategy already approved"); strategies[_addr] = Strategy({ isSupported: true, _deprecated: 0 }); allStrategies.push(_addr); emit StrategyApproved(_addr); } /** * @notice Remove a strategy from the Vault. * @param _addr Address of the strategy to remove */ function removeStrategy(address _addr) external onlyGovernor { require(strategies[_addr].isSupported, "Strategy not approved"); uint256 assetCount = allAssets.length; for (uint256 i = 0; i < assetCount; ++i) { require( assetDefaultStrategies[allAssets[i]] != _addr, "Strategy is default for an asset" ); } // Initialize strategyIndex with out of bounds result so function will // revert if no valid index found uint256 stratCount = allStrategies.length; uint256 strategyIndex = stratCount; for (uint256 i = 0; i < stratCount; ++i) { if (allStrategies[i] == _addr) { strategyIndex = i; break; } } if (strategyIndex < stratCount) { allStrategies[strategyIndex] = allStrategies[stratCount - 1]; allStrategies.pop(); // Mark the strategy as not supported strategies[_addr].isSupported = false; // Withdraw all assets IStrategy strategy = IStrategy(_addr); strategy.withdrawAll(); emit StrategyRemoved(_addr); } } /*************************************** Strategies ****************************************/ /** * @notice Deposit multiple assets from the vault into the strategy. * @param _strategyToAddress Address of the Strategy to deposit assets into. * @param _assets Array of asset address that will be deposited into the strategy. * @param _amounts Array of amounts of each corresponding asset to deposit. */ function depositToStrategy( address _strategyToAddress, address[] calldata _assets, uint256[] calldata _amounts ) external onlyGovernorOrStrategist nonReentrant { _depositToStrategy(_strategyToAddress, _assets, _amounts); } function _depositToStrategy( address _strategyToAddress, address[] calldata _assets, uint256[] calldata _amounts ) internal virtual { require( strategies[_strategyToAddress].isSupported, "Invalid to Strategy" ); require(_assets.length == _amounts.length, "Parameter length mismatch"); uint256 assetCount = _assets.length; for (uint256 i = 0; i < assetCount; ++i) { address assetAddr = _assets[i]; require( IStrategy(_strategyToAddress).supportsAsset(assetAddr), "Asset unsupported" ); // Send required amount of funds to the strategy IERC20(assetAddr).safeTransfer(_strategyToAddress, _amounts[i]); } // Deposit all the funds that have been sent to the strategy IStrategy(_strategyToAddress).depositAll(); } /** * @notice Withdraw multiple assets from the strategy to the vault. * @param _strategyFromAddress Address of the Strategy to withdraw assets from. * @param _assets Array of asset address that will be withdrawn from the strategy. * @param _amounts Array of amounts of each corresponding asset to withdraw. */ function withdrawFromStrategy( address _strategyFromAddress, address[] calldata _assets, uint256[] calldata _amounts ) external onlyGovernorOrStrategist nonReentrant { _withdrawFromStrategy( address(this), _strategyFromAddress, _assets, _amounts ); } /** * @param _recipient can either be a strategy or the Vault */ function _withdrawFromStrategy( address _recipient, address _strategyFromAddress, address[] calldata _assets, uint256[] calldata _amounts ) internal virtual { require( strategies[_strategyFromAddress].isSupported, "Invalid from Strategy" ); require(_assets.length == _amounts.length, "Parameter length mismatch"); uint256 assetCount = _assets.length; for (uint256 i = 0; i < assetCount; ++i) { // Withdraw from Strategy to the recipient IStrategy(_strategyFromAddress).withdraw( _recipient, _assets[i], _amounts[i] ); } } /** * @notice Sets the maximum allowable difference between * total supply and backing assets' value. */ function setMaxSupplyDiff(uint256 _maxSupplyDiff) external onlyGovernor { maxSupplyDiff = _maxSupplyDiff; emit MaxSupplyDiffChanged(_maxSupplyDiff); } /** * @notice Sets the trusteeAddress that can receive a portion of yield. * Setting to the zero address disables this feature. */ function setTrusteeAddress(address _address) external onlyGovernor { trusteeAddress = _address; emit TrusteeAddressChanged(_address); } /** * @notice Sets the TrusteeFeeBps to the percentage of yield that should be * received in basis points. */ function setTrusteeFeeBps(uint256 _basis) external onlyGovernor { require(_basis <= 5000, "basis cannot exceed 50%"); trusteeFeeBps = _basis; emit TrusteeFeeBpsChanged(_basis); } /** * @notice Set OToken Metapool strategy * @param _ousdMetaStrategy Address of OToken metapool strategy */ function setOusdMetaStrategy(address _ousdMetaStrategy) external onlyGovernor { ousdMetaStrategy = _ousdMetaStrategy; emit OusdMetaStrategyUpdated(_ousdMetaStrategy); } /*************************************** Pause ****************************************/ /** * @notice Set the deposit paused flag to true to prevent rebasing. */ function pauseRebase() external onlyGovernorOrStrategist { rebasePaused = true; emit RebasePaused(); } /** * @notice Set the deposit paused flag to true to allow rebasing. */ function unpauseRebase() external onlyGovernorOrStrategist { rebasePaused = false; emit RebaseUnpaused(); } /** * @notice Set the deposit paused flag to true to prevent capital movement. */ function pauseCapital() external onlyGovernorOrStrategist { capitalPaused = true; emit CapitalPaused(); } /** * @notice Set the deposit paused flag to false to enable capital movement. */ function unpauseCapital() external onlyGovernorOrStrategist { capitalPaused = false; emit CapitalUnpaused(); } /*************************************** Utils ****************************************/ /** * @notice Transfer token to governor. Intended for recovering tokens stuck in * contract, i.e. mistaken sends. * @param _asset Address for the asset * @param _amount Amount of the asset to transfer */ function transferToken(address _asset, uint256 _amount) external onlyGovernor { require(!assets[_asset].isSupported, "Only unsupported assets"); IERC20(_asset).safeTransfer(governor(), _amount); } /*************************************** Strategies Admin ****************************************/ /** * @notice Withdraws all assets from the strategy and sends assets to the Vault. * @param _strategyAddr Strategy address. */ function withdrawAllFromStrategy(address _strategyAddr) external onlyGovernorOrStrategist { _withdrawAllFromStrategy(_strategyAddr); } function _withdrawAllFromStrategy(address _strategyAddr) internal virtual { require( strategies[_strategyAddr].isSupported, "Strategy is not supported" ); IStrategy strategy = IStrategy(_strategyAddr); strategy.withdrawAll(); } /** * @notice Withdraws all assets from all the strategies and sends assets to the Vault. */ function withdrawAllFromStrategies() external onlyGovernorOrStrategist { _withdrawAllFromStrategies(); } function _withdrawAllFromStrategies() internal virtual { uint256 stratCount = allStrategies.length; for (uint256 i = 0; i < stratCount; ++i) { IStrategy(allStrategies[i]).withdrawAll(); } } /*************************************** Utils ****************************************/ function _cacheDecimals(address token) internal { Asset storage tokenAsset = assets[token]; if (tokenAsset.decimals != 0) { return; } uint8 decimals = IBasicToken(token).decimals(); require(decimals >= 6 && decimals <= 18, "Unexpected precision"); tokenAsset.decimals = decimals; } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; /** * @title OToken VaultStorage contract * @notice The VaultStorage contract defines the storage for the Vault contracts * @author Origin Protocol Inc */ import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol"; import { SafeERC20 } from "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol"; import { Address } from "@openzeppelin/contracts/utils/Address.sol"; import { IStrategy } from "../interfaces/IStrategy.sol"; import { Governable } from "../governance/Governable.sol"; import { OUSD } from "../token/OUSD.sol"; import { Initializable } from "../utils/Initializable.sol"; import "../utils/Helpers.sol"; contract VaultStorage is Initializable, Governable { using SafeERC20 for IERC20; event AssetSupported(address _asset); event AssetRemoved(address _asset); event AssetDefaultStrategyUpdated(address _asset, address _strategy); event AssetAllocated(address _asset, address _strategy, uint256 _amount); event StrategyApproved(address _addr); event StrategyRemoved(address _addr); event Mint(address _addr, uint256 _value); event Redeem(address _addr, uint256 _value); event CapitalPaused(); event CapitalUnpaused(); event RebasePaused(); event RebaseUnpaused(); event VaultBufferUpdated(uint256 _vaultBuffer); event OusdMetaStrategyUpdated(address _ousdMetaStrategy); event RedeemFeeUpdated(uint256 _redeemFeeBps); event PriceProviderUpdated(address _priceProvider); event AllocateThresholdUpdated(uint256 _threshold); event RebaseThresholdUpdated(uint256 _threshold); event StrategistUpdated(address _address); event MaxSupplyDiffChanged(uint256 maxSupplyDiff); event YieldDistribution(address _to, uint256 _yield, uint256 _fee); event TrusteeFeeBpsChanged(uint256 _basis); event TrusteeAddressChanged(address _address); event NetOusdMintForStrategyThresholdChanged(uint256 _threshold); event SwapperChanged(address _address); event SwapAllowedUndervalueChanged(uint256 _basis); event SwapSlippageChanged(address _asset, uint256 _basis); event Swapped( address indexed _fromAsset, address indexed _toAsset, uint256 _fromAssetAmount, uint256 _toAssetAmount ); event StrategyAddedToMintWhitelist(address indexed strategy); event StrategyRemovedFromMintWhitelist(address indexed strategy); event DripperChanged(address indexed _dripper); event WithdrawalRequested( address indexed _withdrawer, uint256 indexed _requestId, uint256 _amount, uint256 _queued ); event WithdrawalClaimed( address indexed _withdrawer, uint256 indexed _requestId, uint256 _amount ); event WithdrawalClaimable(uint256 _claimable, uint256 _newClaimable); event WithdrawalClaimDelayUpdated(uint256 _newDelay); // Assets supported by the Vault, i.e. Stablecoins enum UnitConversion { DECIMALS, GETEXCHANGERATE } // Changed to fit into a single storage slot so the decimals needs to be recached struct Asset { // Note: OETHVaultCore doesn't use `isSupported` when minting, // redeeming or checking balance of assets. bool isSupported; UnitConversion unitConversion; uint8 decimals; // Max allowed slippage from the Oracle price when swapping collateral assets in basis points. // For example 40 == 0.4% slippage uint16 allowedOracleSlippageBps; } /// @dev mapping of supported vault assets to their configuration // slither-disable-next-line uninitialized-state mapping(address => Asset) internal assets; /// @dev list of all assets supported by the vault. // slither-disable-next-line uninitialized-state address[] internal allAssets; // Strategies approved for use by the Vault struct Strategy { bool isSupported; uint256 _deprecated; // Deprecated storage slot } /// @dev mapping of strategy contracts to their configuration // slither-disable-next-line uninitialized-state mapping(address => Strategy) internal strategies; /// @dev list of all vault strategies address[] internal allStrategies; /// @notice Address of the Oracle price provider contract // slither-disable-next-line uninitialized-state address public priceProvider; /// @notice pause rebasing if true bool public rebasePaused = false; /// @notice pause operations that change the OToken supply. /// eg mint, redeem, allocate, mint/burn for strategy bool public capitalPaused = true; /// @notice Redemption fee in basis points. eg 50 = 0.5% uint256 public redeemFeeBps; /// @notice Percentage of assets to keep in Vault to handle (most) withdrawals. 100% = 1e18. uint256 public vaultBuffer; /// @notice OToken mints over this amount automatically allocate funds. 18 decimals. uint256 public autoAllocateThreshold; /// @notice OToken mints over this amount automatically rebase. 18 decimals. uint256 public rebaseThreshold; /// @dev Address of the OToken token. eg OUSD or OETH. // slither-disable-next-line uninitialized-state OUSD internal oUSD; /// @dev Storage slot for the address of the VaultAdmin contract that is delegated to // keccak256("OUSD.vault.governor.admin.impl"); bytes32 constant adminImplPosition = 0xa2bd3d3cf188a41358c8b401076eb59066b09dec5775650c0de4c55187d17bd9; /// @dev Address of the contract responsible for post rebase syncs with AMMs address private _deprecated_rebaseHooksAddr = address(0); /// @dev Deprecated: Address of Uniswap // slither-disable-next-line constable-states address private _deprecated_uniswapAddr = address(0); /// @notice Address of the Strategist address public strategistAddr = address(0); /// @notice Mapping of asset address to the Strategy that they should automatically // be allocated to // slither-disable-next-line uninitialized-state mapping(address => address) public assetDefaultStrategies; /// @notice Max difference between total supply and total value of assets. 18 decimals. // slither-disable-next-line uninitialized-state uint256 public maxSupplyDiff; /// @notice Trustee contract that can collect a percentage of yield address public trusteeAddress; /// @notice Amount of yield collected in basis points. eg 2000 = 20% uint256 public trusteeFeeBps; /// @dev Deprecated: Tokens that should be swapped for stablecoins address[] private _deprecated_swapTokens; uint256 constant MINT_MINIMUM_UNIT_PRICE = 0.998e18; /// @notice Metapool strategy that is allowed to mint/burn OTokens without changing collateral // slither-disable-start constable-states // slither-disable-next-line uninitialized-state address public ousdMetaStrategy; /// @notice How much OTokens are currently minted by the strategy // slither-disable-next-line uninitialized-state int256 public netOusdMintedForStrategy; /// @notice How much net total OTokens are allowed to be minted by all strategies // slither-disable-next-line uninitialized-state uint256 public netOusdMintForStrategyThreshold; // slither-disable-end constable-states uint256 constant MIN_UNIT_PRICE_DRIFT = 0.7e18; uint256 constant MAX_UNIT_PRICE_DRIFT = 1.3e18; /// @notice Collateral swap configuration. /// @dev is packed into a single storage slot to save gas. struct SwapConfig { // Contract that swaps the vault's collateral assets address swapper; // Max allowed percentage the total value can drop below the total supply in basis points. // For example 100 == 1% uint16 allowedUndervalueBps; } SwapConfig internal swapConfig = SwapConfig(address(0), 0); // List of strategies that can mint oTokens directly // Used in OETHBaseVaultCore // slither-disable-next-line uninitialized-state mapping(address => bool) public isMintWhitelistedStrategy; /// @notice Address of the Dripper contract that streams harvested rewards to the Vault /// @dev The vault is proxied so needs to be set with setDripper against the proxy contract. // slither-disable-start constable-states // slither-disable-next-line uninitialized-state address public dripper; // slither-disable-end constable-states /// Withdrawal Queue Storage ///// struct WithdrawalQueueMetadata { // cumulative total of all withdrawal requests included the ones that have already been claimed uint128 queued; // cumulative total of all the requests that can be claimed including the ones that have already been claimed uint128 claimable; // total of all the requests that have been claimed uint128 claimed; // index of the next withdrawal request starting at 0 uint128 nextWithdrawalIndex; } /// @notice Global metadata for the withdrawal queue including: /// queued - cumulative total of all withdrawal requests included the ones that have already been claimed /// claimable - cumulative total of all the requests that can be claimed including the ones already claimed /// claimed - total of all the requests that have been claimed /// nextWithdrawalIndex - index of the next withdrawal request starting at 0 // slither-disable-next-line uninitialized-state WithdrawalQueueMetadata public withdrawalQueueMetadata; struct WithdrawalRequest { address withdrawer; bool claimed; uint40 timestamp; // timestamp of the withdrawal request // Amount of oTokens to redeem. eg OETH uint128 amount; // cumulative total of all withdrawal requests including this one. // this request can be claimed when this queued amount is less than or equal to the queue's claimable amount. uint128 queued; } /// @notice Mapping of withdrawal request indices to the user withdrawal request data mapping(uint256 => WithdrawalRequest) public withdrawalRequests; /// @notice Sets a minimum delay that is required to elapse between /// requesting async withdrawals and claiming the request. /// When set to 0 async withdrawals are disabled. // slither-disable-start constable-states // slither-disable-next-line uninitialized-state uint256 public withdrawalClaimDelay; // slither-disable-end constable-states // For future use uint256[44] private __gap; /** * @notice set the implementation for the admin, this needs to be in a base class else we cannot set it * @param newImpl address of the implementation */ function setAdminImpl(address newImpl) external onlyGovernor { require( Address.isContract(newImpl), "new implementation is not a contract" ); bytes32 position = adminImplPosition; // solhint-disable-next-line no-inline-assembly assembly { sstore(position, newImpl) } } }
{ "optimizer": { "enabled": true, "runs": 200 }, "evmVersion": "paris", "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "devdoc", "userdoc", "metadata", "abi" ] } }, "metadata": { "useLiteralContent": true }, "libraries": {} }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"inputs":[{"internalType":"address","name":"_wS","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"_threshold","type":"uint256"}],"name":"AllocateThresholdUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"_asset","type":"address"},{"indexed":false,"internalType":"address","name":"_strategy","type":"address"},{"indexed":false,"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"AssetAllocated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"_asset","type":"address"},{"indexed":false,"internalType":"address","name":"_strategy","type":"address"}],"name":"AssetDefaultStrategyUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"_asset","type":"address"}],"name":"AssetRemoved","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"_asset","type":"address"}],"name":"AssetSupported","type":"event"},{"anonymous":false,"inputs":[],"name":"CapitalPaused","type":"event"},{"anonymous":false,"inputs":[],"name":"CapitalUnpaused","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"_dripper","type":"address"}],"name":"DripperChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousGovernor","type":"address"},{"indexed":true,"internalType":"address","name":"newGovernor","type":"address"}],"name":"GovernorshipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"maxSupplyDiff","type":"uint256"}],"name":"MaxSupplyDiffChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"_addr","type":"address"},{"indexed":false,"internalType":"uint256","name":"_value","type":"uint256"}],"name":"Mint","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"_threshold","type":"uint256"}],"name":"NetOusdMintForStrategyThresholdChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"_ousdMetaStrategy","type":"address"}],"name":"OusdMetaStrategyUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousGovernor","type":"address"},{"indexed":true,"internalType":"address","name":"newGovernor","type":"address"}],"name":"PendingGovernorshipTransfer","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"_priceProvider","type":"address"}],"name":"PriceProviderUpdated","type":"event"},{"anonymous":false,"inputs":[],"name":"RebasePaused","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"_threshold","type":"uint256"}],"name":"RebaseThresholdUpdated","type":"event"},{"anonymous":false,"inputs":[],"name":"RebaseUnpaused","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"_addr","type":"address"},{"indexed":false,"internalType":"uint256","name":"_value","type":"uint256"}],"name":"Redeem","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"_redeemFeeBps","type":"uint256"}],"name":"RedeemFeeUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"_address","type":"address"}],"name":"StrategistUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"strategy","type":"address"}],"name":"StrategyAddedToMintWhitelist","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"_addr","type":"address"}],"name":"StrategyApproved","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"_addr","type":"address"}],"name":"StrategyRemoved","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"strategy","type":"address"}],"name":"StrategyRemovedFromMintWhitelist","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"_basis","type":"uint256"}],"name":"SwapAllowedUndervalueChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"_asset","type":"address"},{"indexed":false,"internalType":"uint256","name":"_basis","type":"uint256"}],"name":"SwapSlippageChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"_fromAsset","type":"address"},{"indexed":true,"internalType":"address","name":"_toAsset","type":"address"},{"indexed":false,"internalType":"uint256","name":"_fromAssetAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_toAssetAmount","type":"uint256"}],"name":"Swapped","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"_address","type":"address"}],"name":"SwapperChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"_address","type":"address"}],"name":"TrusteeAddressChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"_basis","type":"uint256"}],"name":"TrusteeFeeBpsChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"_vaultBuffer","type":"uint256"}],"name":"VaultBufferUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"_newDelay","type":"uint256"}],"name":"WithdrawalClaimDelayUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"_claimable","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_newClaimable","type":"uint256"}],"name":"WithdrawalClaimable","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"_withdrawer","type":"address"},{"indexed":true,"internalType":"uint256","name":"_requestId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"WithdrawalClaimed","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"_withdrawer","type":"address"},{"indexed":true,"internalType":"uint256","name":"_requestId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_amount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_queued","type":"uint256"}],"name":"WithdrawalRequested","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"_to","type":"address"},{"indexed":false,"internalType":"uint256","name":"_yield","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_fee","type":"uint256"}],"name":"YieldDistribution","type":"event"},{"inputs":[],"name":"allowedSwapUndervalue","outputs":[{"internalType":"uint256","name":"value","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_addr","type":"address"}],"name":"approveStrategy","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"assetDefaultStrategies","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"autoAllocateThreshold","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_asset","type":"address"}],"name":"cacheDecimals","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"capitalPaused","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"claimGovernance","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_strategyToAddress","type":"address"},{"internalType":"address[]","name":"_assets","type":"address[]"},{"internalType":"uint256[]","name":"_amounts","type":"uint256[]"}],"name":"depositToStrategy","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"dripper","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"governor","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"isGovernor","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"isMintWhitelistedStrategy","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxSupplyDiff","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"netOusdMintForStrategyThreshold","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"netOusdMintedForStrategy","outputs":[{"internalType":"int256","name":"","type":"int256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ousdMetaStrategy","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"pauseCapital","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"pauseRebase","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"priceProvider","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"rebasePaused","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"rebaseThreshold","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"redeemFeeBps","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_asset","type":"address"}],"name":"removeAsset","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_addr","type":"address"}],"name":"removeStrategy","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newImpl","type":"address"}],"name":"setAdminImpl","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_asset","type":"address"},{"internalType":"address","name":"_strategy","type":"address"}],"name":"setAssetDefaultStrategy","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_threshold","type":"uint256"}],"name":"setAutoAllocateThreshold","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_dripper","type":"address"}],"name":"setDripper","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_maxSupplyDiff","type":"uint256"}],"name":"setMaxSupplyDiff","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_threshold","type":"uint256"}],"name":"setNetOusdMintForStrategyThreshold","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_asset","type":"address"},{"internalType":"uint16","name":"_allowedOracleSlippageBps","type":"uint16"}],"name":"setOracleSlippage","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_ousdMetaStrategy","type":"address"}],"name":"setOusdMetaStrategy","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_priceProvider","type":"address"}],"name":"setPriceProvider","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_threshold","type":"uint256"}],"name":"setRebaseThreshold","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_redeemFeeBps","type":"uint256"}],"name":"setRedeemFeeBps","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_address","type":"address"}],"name":"setStrategistAddr","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint16","name":"_basis","type":"uint16"}],"name":"setSwapAllowedUndervalue","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_swapperAddr","type":"address"}],"name":"setSwapper","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_address","type":"address"}],"name":"setTrusteeAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_basis","type":"uint256"}],"name":"setTrusteeFeeBps","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_vaultBuffer","type":"uint256"}],"name":"setVaultBuffer","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_delay","type":"uint256"}],"name":"setWithdrawalClaimDelay","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"strategistAddr","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_asset","type":"address"},{"internalType":"uint8","name":"_unitConversion","type":"uint8"}],"name":"supportAsset","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_fromAsset","type":"address"},{"internalType":"address","name":"_toAsset","type":"address"},{"internalType":"uint256","name":"_fromAssetAmount","type":"uint256"},{"internalType":"uint256","name":"_minToAssetAmount","type":"uint256"},{"internalType":"bytes","name":"_data","type":"bytes"}],"name":"swapCollateral","outputs":[{"internalType":"uint256","name":"toAssetAmount","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"swapper","outputs":[{"internalType":"address","name":"swapper_","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_newGovernor","type":"address"}],"name":"transferGovernance","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_asset","type":"address"},{"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"transferToken","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"trusteeAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"trusteeFeeBps","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"unpauseCapital","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"unpauseRebase","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"vaultBuffer","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"weth","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"withdrawAllFromStrategies","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_strategyAddr","type":"address"}],"name":"withdrawAllFromStrategy","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_strategyFromAddress","type":"address"},{"internalType":"address[]","name":"_assets","type":"address[]"},{"internalType":"uint256[]","name":"_amounts","type":"uint256[]"}],"name":"withdrawFromStrategy","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"withdrawalClaimDelay","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"withdrawalQueueMetadata","outputs":[{"internalType":"uint128","name":"queued","type":"uint128"},{"internalType":"uint128","name":"claimable","type":"uint128"},{"internalType":"uint128","name":"claimed","type":"uint128"},{"internalType":"uint128","name":"nextWithdrawalIndex","type":"uint128"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"withdrawalRequests","outputs":[{"internalType":"address","name":"withdrawer","type":"address"},{"internalType":"bool","name":"claimed","type":"bool"},{"internalType":"uint40","name":"timestamp","type":"uint40"},{"internalType":"uint128","name":"amount","type":"uint128"},{"internalType":"uint128","name":"queued","type":"uint128"}],"stateMutability":"view","type":"function"}]
Deployed Bytecode
0x608060405234801561001057600080fd5b50600436106103a45760003560e01c8063603ea03b116101e9578063ae69f3cb1161010f578063c9919112116100ad578063e6cc54321161007c578063e6cc543214610876578063e829cc161461088a578063eb03654b1461089d578063fc0cfeee146108b057600080fd5b8063c99191121461083f578063d38bfff414610847578063d58e3b3a1461085a578063e45cc9f01461086d57600080fd5b8063b890ebf6116100e9578063b890ebf614610809578063bc90106b1461081c578063c5f008411461082f578063c7af33521461083757600080fd5b8063ae69f3cb146107d0578063b2c9336d146107e3578063b888879e146107f657600080fd5b8063840c4c7a1161018757806394828ffd1161015657806394828ffd146107835780639c82f2a41461078b5780639fa1826e1461079e578063a403e4d5146107a757600080fd5b8063840c4c7a146106ac5780638e510b52146106bf5780638ec489a2146106c8578063937b2581146106db57600080fd5b80636c7561e8116101c35780636c7561e81461066a578063773540b31461067d5780637a2202f3146106905780637b9a70961461069957600080fd5b8063603ea03b14610631578063636e6c4014610644578063663e64ce1461065757600080fd5b8063372aa224116102ce5780634a5e42b11161026c57806353ca9f241161023b57806353ca9f24146105ef578063570d8e1d14610603578063597c8910146106165780635d36b1901461062957600080fd5b80634a5e42b1146105ad5780634bed3bc0146105c057806350ba711c146105d357806352d38e5d146105e657600080fd5b80633fc8cef3116102a85780633fc8cef3146105375780634530820a1461055e57806345e4213b1461059157806349c1d54d1461059a57600080fd5b8063372aa224146105095780633b8ae3971461051c5780633dbc911f1461052f57600080fd5b80631cfbe7bc116103465780632da845a8116103155780632da845a8146104715780632e9958ab14610484578063362bd1a31461049757806336b6d944146104f657600080fd5b80631cfbe7bc1461043b5780631edfe3da1461044e578063207134b0146104575780632b3297f91461046057600080fd5b80630c340a24116103825780630c340a24146103e25780631072cbea14610402578063175188e81461041557806318ce56bd1461042857600080fd5b806309f49bf5146103a957806309f6442c146103b35780630acbda75146103cf575b600080fd5b6103b16108c3565b005b6103bc60385481565b6040519081526020015b60405180910390f35b6103b16103dd366004613024565b61093c565b6103ea6109ee565b6040516001600160a01b0390911681526020016103c6565b6103b1610410366004613059565b610a0b565b6103b1610423366004613083565b610ab8565b6045546103ea906001600160a01b031681565b6103b1610449366004613024565b610da7565b6103bc60395481565b6103bc60435481565b6048546001600160a01b03166103ea565b6103b161047f366004613083565b610e69565b6103b1610492366004613083565b610edb565b604b54604c546104c3916001600160801b0380821692600160801b928390048216928183169291041684565b604080516001600160801b03958616815293851660208501529184169183019190915290911660608201526080016103c6565b6103b1610504366004613083565b610f49565b6103b1610517366004613083565b610f79565b6103b161052a366004613083565b610feb565b6103b1611128565b6103ea7f000000000000000000000000039e2fb66102314ce7b64ce5ce3e5183bc94ad3881565b61058161056c366004613083565b60496020526000908152604090205460ff1681565b60405190151581526020016103c6565b6103bc604e5481565b6042546103ea906001600160a01b031681565b6103b16105bb366004613083565b61119e565b604854600160a01b900461ffff166103bc565b6103bc6105e136600461309e565b611493565b6103bc603b5481565b60375461058190600160a01b900460ff1681565b603f546103ea906001600160a01b031681565b6103b1610624366004613083565b611536565b6103b1611577565b604a546103ea906001600160a01b031681565b6103b1610652366004613024565b61161d565b6103b1610665366004613024565b61167b565b6103b1610678366004613156565b6116d4565b6103b161068b366004613083565b6118d2565b6103bc60475481565b6103b16106a736600461319f565b611944565b6103b16106ba36600461321e565b611a7a565b6103bc60415481565b6103b16106d6366004613024565b611b13565b61073c6106e9366004613024565b604d60205260009081526040902080546001909101546001600160a01b03821691600160a01b810460ff1691600160a81b90910464ffffffffff16906001600160801b0380821691600160801b90041685565b604080516001600160a01b039096168652931515602086015264ffffffffff909216928401929092526001600160801b03918216606084015216608082015260a0016103c6565b6103b1611bc8565b6103b1610799366004613083565b611c38565b6103bc603a5481565b6103ea6107b5366004613083565b6040602081905260009182529020546001600160a01b031681565b6103b16107de36600461321e565b611caa565b6103b16107f1366004613024565b611d38565b6037546103ea906001600160a01b031681565b6103b1610817366004613024565b611d91565b6103b161082a3660046132a4565b611dea565b6103b161201d565b610581612093565b6103b16120c4565b6103b1610855366004613083565b612104565b6103b1610868366004613083565b6121a8565b6103bc60465481565b60375461058190600160a81b900460ff1681565b6103b16108983660046132ce565b61221a565b6103b16108ab366004613024565b6122da565b6103b16108be366004613083565b61238f565b603f546001600160a01b03163314806108df57506108df612093565b6109045760405162461bcd60e51b81526004016108fb906132e9565b60405180910390fd5b6037805460ff60a01b191690556040517fbc044409505c95b6b851433df96e1beae715c909d8e7c1d6d7ab783300d4e3b990600090a1565b610944612093565b6109605760405162461bcd60e51b81526004016108fb90613331565b6113888111156109b25760405162461bcd60e51b815260206004820152601760248201527f62617369732063616e6e6f74206578636565642035302500000000000000000060448201526064016108fb565b60438190556040518181527f56287a45051933ea374811b3d5d165033047be5572cac676f7c28b8be4f746c7906020015b60405180910390a150565b6000610a066000805160206134ec8339815191525490565b905090565b610a13612093565b610a2f5760405162461bcd60e51b81526004016108fb90613331565b6001600160a01b03821660009081526033602052604090205460ff1615610a985760405162461bcd60e51b815260206004820152601760248201527f4f6e6c7920756e737570706f727465642061737365747300000000000000000060448201526064016108fb565b610ab4610aa36109ee565b6001600160a01b0384169083612431565b5050565b610ac0612093565b610adc5760405162461bcd60e51b81526004016108fb90613331565b6001600160a01b03811660009081526035602052604090205460ff16610b3c5760405162461bcd60e51b815260206004820152601560248201527414dd1c985d1959de481b9bdd08185c1c1c9bdd9959605a1b60448201526064016108fb565b60345460005b81811015610beb57826001600160a01b03166040600060348481548110610b6b57610b6b613368565b60009182526020808320909101546001600160a01b0390811684529083019390935260409091019020541603610be35760405162461bcd60e51b815260206004820181905260248201527f53747261746567792069732064656661756c7420666f7220616e20617373657460448201526064016108fb565b600101610b42565b506036548060005b82811015610c4257846001600160a01b031660368281548110610c1857610c18613368565b6000918252602090912001546001600160a01b031603610c3a57809150610c42565b600101610bf3565b5081811015610da1576036610c58600184613394565b81548110610c6857610c68613368565b600091825260209091200154603680546001600160a01b039092169183908110610c9457610c94613368565b9060005260206000200160006101000a8154816001600160a01b0302191690836001600160a01b031602179055506036805480610cd357610cd36133ad565b60008281526020808220830160001990810180546001600160a01b03191690559092019092556001600160a01b03861680835260359091526040808320805460ff19169055805163429c145b60e11b81529051879363853828b6926004808201939182900301818387803b158015610d4a57600080fd5b505af1158015610d5e573d6000803e3d6000fd5b50506040516001600160a01b03881681527f09a1db4b80c32706328728508c941a6b954f31eb5affd32f236c1fd405f8fea49250602001905060405180910390a1505b50505050565b610daf612093565b610dcb5760405162461bcd60e51b81526004016108fb90613331565b801580610de857506102588110158015610de857506213c6808111155b610e345760405162461bcd60e51b815260206004820152601a60248201527f496e76616c696420636c61696d2064656c617920706572696f6400000000000060448201526064016108fb565b604e8190556040518181527fc59f5e32049abab44ddea11021f5abb89422a2f550837afcf25df9fc8d0db6b0906020016109e3565b610e71612093565b610e8d5760405162461bcd60e51b81526004016108fb90613331565b604280546001600160a01b0319166001600160a01b0383169081179091556040519081527f1e4af5ac389e8cde1bdaa6830881b6c987c62a45cfb3b33d27d805cde3b57750906020016109e3565b610ee3612093565b610eff5760405162461bcd60e51b81526004016108fb90613331565b604a80546001600160a01b0319166001600160a01b0383169081179091556040517faf2910d9759321733de15af1827a49830692912adeb2b3646334861f2cd2eed490600090a250565b610f51612093565b610f6d5760405162461bcd60e51b81526004016108fb90613331565b610f7681612488565b50565b610f81612093565b610f9d5760405162461bcd60e51b81526004016108fb90613331565b603780546001600160a01b0319166001600160a01b0383169081179091556040519081527fb266add5f3044b17d27db796af992cecbe413921b4e8aaaee03c719e16b9806a906020016109e3565b610ff3612093565b61100f5760405162461bcd60e51b81526004016108fb90613331565b6001600160a01b03811660009081526035602052604090205460ff16156110785760405162461bcd60e51b815260206004820152601960248201527f537472617465677920616c726561647920617070726f7665640000000000000060448201526064016108fb565b6040805180820182526001808252600060208084018281526001600160a01b038716808452603583528684209551865460ff19169015151786559051948401949094556036805493840181559091527f4a11f94e20a93c79f6ec743a1954ec4fc2c08429ae2122118bf234b2185c81b890910180546001600160a01b0319168317905591519081527f960dd94cbb79169f09a4e445d58b895df2d9bffa5b31055d0932d801724a20d191016109e3565b603f546001600160a01b03163314806111445750611144612093565b6111605760405162461bcd60e51b81526004016108fb906132e9565b6037805460ff60a81b1916600160a81b1790556040517f71f0e5b62f846a22e0b4d159e516e62fa9c2b8eb570be15f83e67d98a2ee51e090600090a1565b6111a6612093565b6111c25760405162461bcd60e51b81526004016108fb90613331565b6001600160a01b03811660009081526033602052604090205460ff166112205760405162461bcd60e51b8152602060048201526013602482015272105cdcd95d081b9bdd081cdd5c1c1bdc9d1959606a1b60448201526064016108fb565b604051632fa8a91360e11b81526001600160a01b03821660048201526509184e72a000903090635f51522690602401602060405180830381865afa15801561126c573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061129091906133c3565b11156112de5760405162461bcd60e51b815260206004820152601760248201527f5661756c74207374696c6c20686f6c647320617373657400000000000000000060448201526064016108fb565b6034548060005b8281101561133457836001600160a01b03166034828154811061130a5761130a613368565b6000918252602090912001546001600160a01b03160361132c57809150611334565b6001016112e5565b506034611342600184613394565b8154811061135257611352613368565b600091825260209091200154603480546001600160a01b03909216918390811061137e5761137e613368565b9060005260206000200160006101000a8154816001600160a01b0302191690836001600160a01b0316021790555060348054806113bd576113bd6133ad565b60008281526020808220600019908401810180546001600160a01b031990811690915593019093556001600160a01b038616808252604080855280832080549094169093558251908152928301527fba58ce12801c949fa65f41c46ed108671c219baf945fa48d21026cea99ff252a910160405180910390a16001600160a01b038316600081815260336020908152604091829020805464ffffffffff1916905590519182527f37803e2125c48ee96c38ddf04e826daf335b0e1603579040fd275aba6d06b6fc910160405180910390a1505050565b7f53bf423e48ed90e97d02ab0ebab13b2a235a6bfbe9c321847d5c175333ac453580546000919060011981016114db5760405162461bcd60e51b81526004016108fb906133dc565b60028255603f546001600160a01b03163314806114fb57506114fb612093565b6115175760405162461bcd60e51b81526004016108fb906132e9565b61152589898989898961258f565b600190925550979650505050505050565b603f546001600160a01b03163314806115525750611552612093565b61156e5760405162461bcd60e51b81526004016108fb906132e9565b610f76816125da565b7f44c4d30b2eaad5130ad70c3ba6972730566f3e6359ab83e800d905c61b1c51db546001600160a01b0316336001600160a01b0316146116125760405162461bcd60e51b815260206004820152603060248201527f4f6e6c79207468652070656e64696e6720476f7665726e6f722063616e20636f60448201526f6d706c6574652074686520636c61696d60801b60648201526084016108fb565b61161b33612639565b565b611625612093565b6116415760405162461bcd60e51b81526004016108fb90613331565b600060465560478190556040518181527fc29d6fedbc6bdf267a08166c2b976fbd72aca5d6a769528616f8b9224c8f197f906020016109e3565b611683612093565b61169f5760405162461bcd60e51b81526004016108fb90613331565b60418190556040518181527f95201f9c21f26877223b1ff4073936a6484c35495649e60e55730497aeb60d93906020016109e3565b6116dc612093565b6116f85760405162461bcd60e51b81526004016108fb90613331565b6001600160a01b03821660009081526033602052604090205460ff16156117615760405162461bcd60e51b815260206004820152601760248201527f417373657420616c726561647920737570706f7274656400000000000000000060448201526064016108fb565b60405180608001604052806001151581526020018260ff16600181111561178a5761178a613404565b600181111561179b5761179b613404565b81526000602080830182905260409283018290526001600160a01b0386168252603381529190208251815490151560ff19821681178355928401519192839161ff001990911661ffff19909116176101008360018111156117fe576117fe613404565b02179055506040820151815460609093015161ffff1663010000000264ffff0000001960ff90921662010000029190911664ffffff0000199093169290921791909117905561184c82612488565b603480546001810182556000919091527f46bddb1178e94d7f2892ff5f366840eb658911794f2c3a44c450aa2c505186c10180546001600160a01b0319166001600160a01b0384169081179091556040519081527f4f1ac48525e50059cc1cc6e0e1940ece0dd653a4db4841538d6aef036be2fb7b906020015b60405180910390a15050565b6118da612093565b6118f65760405162461bcd60e51b81526004016108fb90613331565b603f80546001600160a01b0319166001600160a01b0383169081179091556040519081527f869e0abd13cc3a975de7b93be3df1cb2255c802b1cead85963cc79d99f131bee906020016109e3565b61194c612093565b6119685760405162461bcd60e51b81526004016108fb90613331565b6001600160a01b03821660009081526033602052604090205460ff166119c65760405162461bcd60e51b8152602060048201526013602482015272105cdcd95d081b9bdd081cdd5c1c1bdc9d1959606a1b60448201526064016108fb565b6103e88161ffff1610611a0f5760405162461bcd60e51b81526020600482015260116024820152700a6d8d2e0e0c2ceca40e8dede40d0d2ced607b1b60448201526064016108fb565b6001600160a01b038216600081815260336020908152604091829020805464ffff0000001916630100000061ffff8716908102919091179091558251938452908301527f8d22e9d2cbe8bb65a3c4412bd8970743864512a1a0e004e8d00fb96277b78b9491016118c6565b603f546001600160a01b0316331480611a965750611a96612093565b611ab25760405162461bcd60e51b81526004016108fb906132e9565b7f53bf423e48ed90e97d02ab0ebab13b2a235a6bfbe9c321847d5c175333ac453580546001198101611af65760405162461bcd60e51b81526004016108fb906133dc565b60028255611b0787878787876126fa565b50600190555050505050565b603f546001600160a01b0316331480611b2f5750611b2f612093565b611b4b5760405162461bcd60e51b81526004016108fb906132e9565b670de0b6b3a7640000811115611b935760405162461bcd60e51b815260206004820152600d60248201526c496e76616c69642076616c756560981b60448201526064016108fb565b60398190556040518181527f41ecb23a0e7865b25f38c268b7c3012220d822929e9edff07326e89d5bb822b5906020016109e3565b603f546001600160a01b0316331480611be45750611be4612093565b611c005760405162461bcd60e51b81526004016108fb906132e9565b6037805460ff60a81b191690556040517f891ebab18da80ebeeea06b1b1cede098329c4c008906a98370c2ac7a80b571cb90600090a1565b611c40612093565b611c5c5760405162461bcd60e51b81526004016108fb90613331565b604880546001600160a01b0319166001600160a01b0383169081179091556040519081527f7d7719313229e558c5a3893cad2eb86a86a049156d1d9ebd5c63a8eedefd1c03906020016109e3565b603f546001600160a01b0316331480611cc65750611cc6612093565b611ce25760405162461bcd60e51b81526004016108fb906132e9565b7f53bf423e48ed90e97d02ab0ebab13b2a235a6bfbe9c321847d5c175333ac453580546001198101611d265760405162461bcd60e51b81526004016108fb906133dc565b60028255611b07308888888888612932565b611d40612093565b611d5c5760405162461bcd60e51b81526004016108fb90613331565b603a8190556040518181527f2ec5fb5a3d2703edc461252d92ccd2799c3c74f01d97212b20388207fa17ae45906020016109e3565b611d99612093565b611db55760405162461bcd60e51b81526004016108fb90613331565b603b8190556040518181527f39367850377ac04920a9a670f2180e7a94d83b15ad302e59875ec58fd10bd37d906020016109e3565b603f546001600160a01b0316331480611e065750611e06612093565b611e225760405162461bcd60e51b81526004016108fb906132e9565b604080516001600160a01b038085168252831660208201527fba58ce12801c949fa65f41c46ed108671c219baf945fa48d21026cea99ff252a910160405180910390a16001600160a01b03811615611fef576001600160a01b03811660009081526035602052604090205460ff16611ed45760405162461bcd60e51b815260206004820152601560248201527414dd1c985d1959de481b9bdd08185c1c1c9bdd9959605a1b60448201526064016108fb565b6001600160a01b038216600090815260336020526040902054819060ff16611f375760405162461bcd60e51b8152602060048201526016602482015275105cdcd95d081a5cc81b9bdd081cdd5c1c1bdc9d195960521b60448201526064016108fb565b60405163551c457b60e11b81526001600160a01b03848116600483015282169063aa388af690602401602060405180830381865afa158015611f7d573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611fa1919061341a565b611fed5760405162461bcd60e51b815260206004820152601f60248201527f4173736574206e6f7420737570706f727465642062792053747261746567790060448201526064016108fb565b505b6001600160a01b03918216600090815260406020819052902080546001600160a01b03191691909216179055565b603f546001600160a01b03163314806120395750612039612093565b6120555760405162461bcd60e51b81526004016108fb906132e9565b6037805460ff60a01b1916600160a01b1790556040517f8cff26a5985614b3d30629cc4ab83824bf115aec971b718d8f2f99562032e97290600090a1565b60006120ab6000805160206134ec8339815191525490565b6001600160a01b0316336001600160a01b031614905090565b603f546001600160a01b03163314806120e057506120e0612093565b6120fc5760405162461bcd60e51b81526004016108fb906132e9565b61161b61299b565b61210c612093565b6121285760405162461bcd60e51b81526004016108fb90613331565b612150817f44c4d30b2eaad5130ad70c3ba6972730566f3e6359ab83e800d905c61b1c51db55565b806001600160a01b03166121706000805160206134ec8339815191525490565b6001600160a01b03167fa39cc5eb22d0f34d8beaefee8a3f17cc229c1a1d1ef87a5ad47313487b1c4f0d60405160405180910390a350565b6121b0612093565b6121cc5760405162461bcd60e51b81526004016108fb90613331565b604580546001600160a01b0319166001600160a01b0383169081179091556040519081527fa12850fb726e0b2b7b3c9a9342031e1268a8148d0eb06b4bea8613204ffcd2b8906020016109e3565b612222612093565b61223e5760405162461bcd60e51b81526004016108fb90613331565b6127118161ffff161061228a5760405162461bcd60e51b8152602060048201526014602482015273496e76616c696420626173697320706f696e747360601b60448201526064016108fb565b6048805461ffff60a01b1916600160a01b61ffff8416908102919091179091556040519081527ff12c00256bee2b6facb111a88a9b1cff86e79132939b44f1353212d6f7469557906020016109e3565b6122e2612093565b6122fe5760405162461bcd60e51b81526004016108fb90613331565b6103e881111561235a5760405162461bcd60e51b815260206004820152602160248201527f52656465656d206665652073686f756c64206e6f74206265206f7665722031306044820152602560f81b60648201526084016108fb565b60388190556040518181527fd6c7508d6658ccee36b7b7d7fd72e5cbaeefb40c64eff24e9ae7470e846304ee906020016109e3565b612397612093565b6123b35760405162461bcd60e51b81526004016108fb90613331565b803b61240d5760405162461bcd60e51b8152602060048201526024808201527f6e657720696d706c656d656e746174696f6e206973206e6f74206120636f6e746044820152631c9858dd60e21b60648201526084016108fb565b7fa2bd3d3cf188a41358c8b401076eb59066b09dec5775650c0de4c55187d17bd955565b604080516001600160a01b038416602482015260448082018490528251808303909101815260649091019091526020810180516001600160e01b031663a9059cbb60e01b1790526124839084906129f2565b505050565b6001600160a01b0381166000908152603360205260409020805462010000900460ff16156124b4575050565b6000826001600160a01b031663313ce5676040518163ffffffff1660e01b8152600401602060405180830381865afa1580156124f4573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612518919061343c565b905060068160ff1610158015612532575060128160ff1611155b6125755760405162461bcd60e51b81526020600482015260146024820152732ab732bc3832b1ba32b210383932b1b4b9b4b7b760611b60448201526064016108fb565b815460ff909116620100000262ff00001990911617905550565b60405162461bcd60e51b815260206004820152601d60248201527f436f6c6c61746572616c2073776170206e6f7420737570706f7274656400000060448201526000906064016108fb565b6125e381612ac4565b306001600160a01b031663b9b17f9f6040518163ffffffff1660e01b8152600401600060405180830381600087803b15801561261e57600080fd5b505af1158015612632573d6000803e3d6000fd5b5050505050565b6001600160a01b03811661268f5760405162461bcd60e51b815260206004820152601a60248201527f4e657720476f7665726e6f72206973206164647265737328302900000000000060448201526064016108fb565b806001600160a01b03166126af6000805160206134ec8339815191525490565b6001600160a01b03167fc7c0c772add429241571afb3805861fb3cfa2af374534088b76cdb4325a87e9a60405160405180910390a3610f76816000805160206134ec83398151915255565b6001600160a01b03851660009081526035602052604090205460ff166127585760405162461bcd60e51b8152602060048201526013602482015272496e76616c696420746f20537472617465677960681b60448201526064016108fb565b6001831480156127685750600181145b80156127cc57507f000000000000000000000000039e2fb66102314ce7b64ce5ce3e5183bc94ad386001600160a01b0316848460008181106127ac576127ac613368565b90506020020160208101906127c19190613083565b6001600160a01b0316145b6128115760405162461bcd60e51b815260206004820152601660248201527513db9b1e4815d15512081a5cc81cdd5c1c1bdc9d195960521b60448201526064016108fb565b612819612b88565b8282600081811061282c5761282c613368565b9050602002013511156128815760405162461bcd60e51b815260206004820152601960248201527f4e6f7420656e6f756768205745544820617661696c61626c650000000000000060448201526064016108fb565b6128d8858383600081811061289857612898613368565b905060200201357f000000000000000000000000039e2fb66102314ce7b64ce5ce3e5183bc94ad386001600160a01b03166124319092919063ffffffff16565b846001600160a01b031663de5f62686040518163ffffffff1660e01b8152600401600060405180830381600087803b15801561291357600080fd5b505af1158015612927573d6000803e3d6000fd5b505050505050505050565b612940868686868686612c96565b306001600160a01b031663b9b17f9f6040518163ffffffff1660e01b8152600401600060405180830381600087803b15801561297b57600080fd5b505af115801561298f573d6000803e3d6000fd5b50505050505050505050565b6129a3612e1e565b306001600160a01b031663b9b17f9f6040518163ffffffff1660e01b8152600401600060405180830381600087803b1580156129de57600080fd5b505af1158015610da1573d6000803e3d6000fd5b6000612a47826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564815250856001600160a01b0316612eaa9092919063ffffffff16565b8051909150156124835780806020019051810190612a65919061341a565b6124835760405162461bcd60e51b815260206004820152602a60248201527f5361666545524332303a204552433230206f7065726174696f6e20646964206e6044820152691bdd081cdd58d8d9595960b21b60648201526084016108fb565b6001600160a01b03811660009081526035602052604090205460ff16612b2c5760405162461bcd60e51b815260206004820152601960248201527f5374726174656779206973206e6f7420737570706f727465640000000000000060448201526064016108fb565b6000819050806001600160a01b031663853828b66040518163ffffffff1660e01b8152600401600060405180830381600087803b158015612b6c57600080fd5b505af1158015612b80573d6000803e3d6000fd5b505050505050565b60408051608081018252604b546001600160801b03808216808452600160801b9283900482166020850152604c54808316958501869052929092041660608301526000928391612bd89190613459565b6040516370a0823160e01b81523060048201526001600160801b039190911691506000906001600160a01b037f000000000000000000000000039e2fb66102314ce7b64ce5ce3e5183bc94ad3816906370a0823190602401602060405180830381865afa158015612c4d573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612c7191906133c3565b9050818111612c84576000935050505090565b612c8e8282613394565b935050505090565b6001600160a01b03851660009081526035602052604090205460ff16612cf65760405162461bcd60e51b8152602060048201526015602482015274496e76616c69642066726f6d20537472617465677960581b60448201526064016108fb565b828114612d455760405162461bcd60e51b815260206004820152601960248201527f506172616d65746572206c656e677468206d69736d617463680000000000000060448201526064016108fb565b8260005b81811015612e1457866001600160a01b031663d9caed1289888885818110612d7357612d73613368565b9050602002016020810190612d889190613083565b878786818110612d9a57612d9a613368565b6040516001600160e01b031960e088901b1681526001600160a01b03958616600482015294909316602485015250602090910201356044820152606401600060405180830381600087803b158015612df157600080fd5b505af1158015612e05573d6000803e3d6000fd5b50505050806001019050612d49565b5050505050505050565b60365460005b81811015610ab45760368181548110612e3f57612e3f613368565b60009182526020822001546040805163429c145b60e11b815290516001600160a01b039092169263853828b69260048084019382900301818387803b158015612e8757600080fd5b505af1158015612e9b573d6000803e3d6000fd5b50505050806001019050612e24565b6060612eb98484600085612ec3565b90505b9392505050565b606082471015612f245760405162461bcd60e51b815260206004820152602660248201527f416464726573733a20696e73756666696369656e742062616c616e636520666f6044820152651c8818d85b1b60d21b60648201526084016108fb565b843b612f725760405162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e747261637400000060448201526064016108fb565b600080866001600160a01b03168587604051612f8e919061349c565b60006040518083038185875af1925050503d8060008114612fcb576040519150601f19603f3d011682016040523d82523d6000602084013e612fd0565b606091505b5091509150612fe0828286612feb565b979650505050505050565b60608315612ffa575081612ebc565b82511561300a5782518084602001fd5b8160405162461bcd60e51b81526004016108fb91906134b8565b60006020828403121561303657600080fd5b5035919050565b80356001600160a01b038116811461305457600080fd5b919050565b6000806040838503121561306c57600080fd5b6130758361303d565b946020939093013593505050565b60006020828403121561309557600080fd5b612ebc8261303d565b60008060008060008060a087890312156130b757600080fd5b6130c08761303d565b95506130ce6020880161303d565b94506040870135935060608701359250608087013567ffffffffffffffff8111156130f857600080fd5b8701601f8101891361310957600080fd5b803567ffffffffffffffff81111561312057600080fd5b89602082840101111561313257600080fd5b60208201935080925050509295509295509295565b60ff81168114610f7657600080fd5b6000806040838503121561316957600080fd5b6131728361303d565b9150602083013561318281613147565b809150509250929050565b803561ffff8116811461305457600080fd5b600080604083850312156131b257600080fd5b6131bb8361303d565b91506131c96020840161318d565b90509250929050565b60008083601f8401126131e457600080fd5b50813567ffffffffffffffff8111156131fc57600080fd5b6020830191508360208260051b850101111561321757600080fd5b9250929050565b60008060008060006060868803121561323657600080fd5b61323f8661303d565b9450602086013567ffffffffffffffff81111561325b57600080fd5b613267888289016131d2565b909550935050604086013567ffffffffffffffff81111561328757600080fd5b613293888289016131d2565b969995985093965092949392505050565b600080604083850312156132b757600080fd5b6132c08361303d565b91506131c96020840161303d565b6000602082840312156132e057600080fd5b612ebc8261318d565b60208082526028908201527f43616c6c6572206973206e6f74207468652053747261746567697374206f722060408201526723b7bb32b93737b960c11b606082015260800190565b6020808252601a908201527f43616c6c6572206973206e6f742074686520476f7665726e6f72000000000000604082015260600190565b634e487b7160e01b600052603260045260246000fd5b634e487b7160e01b600052601160045260246000fd5b818103818111156133a7576133a761337e565b92915050565b634e487b7160e01b600052603160045260246000fd5b6000602082840312156133d557600080fd5b5051919050565b6020808252600e908201526d1499595b9d1c985b9d0818d85b1b60921b604082015260600190565b634e487b7160e01b600052602160045260246000fd5b60006020828403121561342c57600080fd5b81518015158114612ebc57600080fd5b60006020828403121561344e57600080fd5b8151612ebc81613147565b6001600160801b0382811682821603908111156133a7576133a761337e565b60005b8381101561349357818101518382015260200161347b565b50506000910152565b600082516134ae818460208701613478565b9190910192915050565b60208152600082518060208401526134d7816040850160208701613478565b601f01601f1916919091016040019291505056fe7bea13895fa79d2831e0a9e28edede30099005a50d652d8957cf8a607ee6ca4aa2646970667358221220458f392f9f91f3db00c08ee7c627941bb2a22094a2dd5e62320d67970312d0c264736f6c634300081c0033
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 30 Chains
Chain | Token | Portfolio % | Price | Amount | Value |
---|
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.