Source Code
Overview
S Balance
S Value
$0.00View more zero value Internal Transactions in Advanced View mode
Advanced mode:
Cross-Chain Transactions
Loading...
Loading
This contract may be a proxy contract. Click on More Options and select Is this a proxy? to confirm and enable the "Read as Proxy" & "Write as Proxy" tabs.
Contract Name:
Protocol
Compiler Version
v0.8.28+commit.7893614a
Optimization Enabled:
Yes with 50 runs
Other Settings:
shanghai EvmVersion
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
// OpenZeppelin
import { AccessControlUpgradeable } from "@openzeppelin/contracts-upgradeable/access/AccessControlUpgradeable.sol";
import { ReentrancyGuardUpgradeable } from "@openzeppelin/contracts-upgradeable/security/ReentrancyGuardUpgradeable.sol";
// libraries
import { ProtocolConfigState } from "./libs/ProtocolConfigState.sol";
import { FeeManagementLibrary } from "./libs/FeeManagementLibrary.sol";
import { CacheLibrary } from "./libs/CacheLibrary.sol";
import { HookLibrary } from "./libs/HookLibrary.sol";
import { ProtocolUtils } from "./libs/ProtocolUtils.sol";
import { OperationContextLibrary } from "./libs/OperationContextLibrary.sol";
import { TimeManagementLibrary } from "./libs/TimeManagementLibrary.sol";
import { ValidationLibrary } from "./libs/ValidationLibrary.sol";
import { CustomRevert } from "./libs/CustomRevert.sol";
// types
import { GroupId, GroupIdLibrary } from "./types/GroupId.sol";
import { GroupStateHelper, GroupSettings } from "./types/GroupStateHelper.sol";
import { GroupState, CollateralInfo, OperationTypes } from "./types/CommonTypes.sol";
import { GroupKey } from "./types/GroupKey.sol";
import { Currency, CurrencyLibrary } from "./types/Currency.sol";
// declarations
import { DProtocol } from "./declarations/DProtocol.sol";
import { DOperationContext } from "./declarations/DOperationContext.sol";
// interfaces
import { IProtocol } from "./interfaces/IProtocol.sol";
import { ITreasuryMinimum as ITreasury } from "./interfaces/ITreasuryMinimum.sol";
import { IHooks } from "./interfaces/IHooks.sol";
contract Protocol is AccessControlUpgradeable, ReentrancyGuardUpgradeable, IProtocol {
using CurrencyLibrary for Currency;
using GroupIdLibrary for GroupKey;
using OperationContextLibrary for DOperationContext.OperationContext;
using CacheLibrary for CacheLibrary.Storage;
using FeeManagementLibrary for FeeManagementLibrary.FeeStorage;
using TimeManagementLibrary for TimeManagementLibrary.TimeStorage;
using HookLibrary for IHooks;
using CustomRevert for bytes4;
using ValidationLibrary for *;
/// @notice Cache storage for group states
CacheLibrary.Storage private cacheStorage;
/// @notice Fee storage for owed fees
FeeManagementLibrary.FeeStorage private feeStorage;
/// @notice Time storage for operation time tracking
TimeManagementLibrary.TimeStorage private timeStorage;
/// @notice Configuration storage for protocol-wide flags
ProtocolConfigState.ConfigStorage private configStorage;
/// @notice Admin address
address public adminAddress;
/// @custom:oz-upgrades-unsafe-allow constructor
constructor() initializer {}
/**
* @notice Allow direct ETH transfers
*/
receive() external payable {}
/**
* @notice Fallback function to prevent direct calls
*/
fallback() external payable {
IProtocol.NotPermitted.selector.revertWith();
}
/**
* @notice Initializes the Protocol contract.
* @param _admin The address of the admin.
*/
function initialize(address _admin) external initializer {
if (_admin == address(0)) {
IProtocol.ZeroAddress.selector.revertWith();
}
__AccessControl_init();
__ReentrancyGuard_init();
_grantRole(DEFAULT_ADMIN_ROLE, _admin);
adminAddress = _admin;
}
/**
* @notice Forces an update of the cache for a specific group.
* @param groupId The identifier of the group.
*/
function forceUpdateGroupCache(address tokenRegistry, GroupId groupId) external nonReentrant onlyRole(DEFAULT_ADMIN_ROLE) {
cacheStorage.forceUpdate(tokenRegistry, groupId);
address groupTreasury = cacheStorage.getGroupTreasury(groupId);
if (groupTreasury != address(0)) {
try ITreasury(groupTreasury).forceUpdateGroupCache(tokenRegistry, groupId) {
/// @dev Treasury cache update succeeded
} catch {
/// @dev Bubble up the revert reason if Treasury fails
CustomRevert.bubbleUpAndRevertWith(IProtocol.TreasuryGroupUpdateFailed.selector, address(groupTreasury));
}
} else {
IProtocol.TreasuryGroupUpdateFailed.selector.revertWith(groupId);
}
}
/**
* @notice Allows the hook contract to settle owed fees.
* @param groupId The key identifying the group.
* @param token The token to settle fees in.
* @dev Only the hook contract can call this function.
*/
function settleOwedFees(GroupId groupId, address token) external nonReentrant {
address hookContract = cacheStorage.getGroupHookContract(groupId);
// Validate caller is the hook contract
if (_msgSender() != hookContract) {
IProtocol.NotPermitted.selector.revertWith();
}
feeStorage.settleOwedFees(groupId, hookContract, token);
}
/**
* @notice Mints tokens based on the provided parameters.
* @param groupKey The key identifying the group.
* @param params The mint parameters.
* @return result The result of the mint operation.
*/
function mintToken(
GroupKey calldata groupKey,
DProtocol.MintParams calldata params
) external payable override nonReentrant returns (DProtocol.MintResult memory result) {
GroupId groupId = groupKey.toId();
GroupState memory groupState = cacheStorage.getGroupState(groupId);
(bool isSupported, CollateralInfo memory collateralInfo) = ValidationLibrary.isPaymentTokenSupported(
groupState.acceptableCollaterals,
params.baseIn,
Currency.wrap(params.paymentToken),
true,
msg.value
);
if (!isSupported) {
IProtocol.UnsupportedCollateralType.selector.revertWith();
}
uint256 baseIn = params.baseIn;
// If user sets baseIn == max, and payment token is non-native,
// then we override baseIn with their entire balance.
if (params.baseIn == type(uint256).max && !Currency.wrap(params.paymentToken).isNative()) {
baseIn = Currency.wrap(params.paymentToken).balanceOf(_msgSender());
}
// If baseIn > 0, proceed with the minting logic
if (baseIn > 0) {
DOperationContext.Context memory opParams = DOperationContext.Context({
groupId: groupId,
groupState: groupState,
/// @dev amount0 is the payment token amount
/// @dev amount1 reserved for further use for the hooks
/// @dev context updates the amount1 to the minted token amount after the conversion is done
amounts: DOperationContext.ContextAmounts({ amount0: baseIn, amount1: 0 }),
operationType: params.operationType,
paymentToken: collateralInfo,
slippage: params.slippage,
recipient: _msgSender(),
hookData: params.hookData
});
DOperationContext.OperationContext memory context = OperationContextLibrary.createContext(opParams);
ValidationLibrary.validateOperation(configStorage, groupId, context);
result = _processMint(groupId, context);
return result;
}
// If baseIn == 0, returning an empty MintResult
return DProtocol.MintResult({ ytMinted: 0, vtMinted: 0 });
}
/**
* @notice Redeems tokens based on the provided parameters.
* @param groupKey The key identifying the group.
* @param params The redeem parameters.
* @return baseOut The amount of base tokens received.
*/
function redeemToken(
GroupKey calldata groupKey,
DProtocol.RedeemParams calldata params
) external override nonReentrant returns (uint256 baseOut) {
GroupId groupId = groupKey.toId();
GroupState memory groupState = cacheStorage.getGroupState(groupId);
(bool isSupported, CollateralInfo memory collateralInfo) = ValidationLibrary.isPaymentTokenSupported(
groupState.acceptableCollaterals,
params.baseOut,
Currency.wrap(params.desiredCollateral),
false,
0
);
if (!isSupported) IProtocol.UnsupportedCollateralType.selector.revertWith();
baseOut = params.baseOut;
if (params.baseOut == type(uint256).max) {
if (params.operationType == uint8(OperationTypes.OP_TYPE_REDEEM_VT)) {
baseOut = groupState.core.xToken.balanceOf(_msgSender());
} else {
baseOut = groupState.extended.rebalancePool.balanceOf(_msgSender());
}
}
if (baseOut > 0) {
// Creating operation context without hookData
DOperationContext.Context memory opParams = DOperationContext.Context({
groupId: groupId,
groupState: groupState,
/// @dev amount0 is the redeeming token amount
/// @dev amount1 reserved for further use for the hooks
/// @dev context updates the amount1 to the desired token amount after the conversion is done
amounts: DOperationContext.ContextAmounts({ amount0: baseOut, amount1: 0 }),
operationType: params.operationType,
paymentToken: collateralInfo,
slippage: params.slippage,
recipient: _msgSender(),
hookData: params.hookData
});
DOperationContext.OperationContext memory context = OperationContextLibrary.createContext(opParams);
ValidationLibrary.validateOperation(configStorage, groupId, context);
baseOut = _processRedeem(groupId, context);
return baseOut;
}
}
/**
* @notice Processes the mint operation.
* @param groupId The group ID.
* @param context The operation context.
* @return result The result of the mint operation.
*/
function _processMint(
GroupId groupId,
DOperationContext.OperationContext memory context
) private returns (DProtocol.MintResult memory result) {
address recipient = context.getRecipient();
address hookAddress = context.getHookContract();
// Handle any necessary before-mint hooks using HookLibrary
if (hookAddress != address(0)) {
IHooks hook = IHooks(hookAddress);
HookLibrary.beforeMint(hook, groupId, context);
}
// Prepare the token amount for the treasury & distribute fees
(Currency preparedToken, uint256 amountWithoutFee, ProtocolUtils.FeeSplit memory feeSplit) = feeStorage.prepareTokenForTreasury(
timeStorage,
context,
context.getAmount0(),
true
);
// Mint the tokens to the recipient
result = ProtocolUtils.mintTokens(context, amountWithoutFee, context.getRecipient());
// Distribute fees based on the split
feeStorage.distributeFees(context, context.getYieldBearingToken(), feeSplit);
emit MintToken(groupId, preparedToken.toAddress(), recipient, _msgSender(), result.ytMinted, result.vtMinted);
// Handle any necessary after-mint hooks
if (hookAddress != address(0)) {
IHooks hook = IHooks(hookAddress);
HookLibrary.afterMint(hook, groupId, context);
}
return result;
}
/**
* @notice Processes the redeem operation.
* @param groupId The group ID.
* @param context The operation context.
* @return baseOut The amount of base tokens received.
*/
function _processRedeem(GroupId groupId, DOperationContext.OperationContext memory context) private returns (uint256 baseOut) {
address from = context.getRecipient();
address hookAddress = context.getHookContract();
// Handle any necessary before-redeem hooks
if (hookAddress != address(0)) {
IHooks hook = IHooks(hookAddress);
HookLibrary.beforeRedeem(hook, groupId, context);
}
// Prepare tokens and distribute fees
(Currency preparedToken, uint256 amountWithoutFee, ProtocolUtils.FeeSplit memory feeSplit) = feeStorage.prepareTokenForTreasury(
timeStorage,
context,
context.getAmount0(),
false
);
// Distribute fees based on the split
feeStorage.distributeFees(context, preparedToken, feeSplit);
// Convert tokens to desired collateral and apply slippage check
baseOut = ProtocolUtils.convertToDesiredCollateralAndSend(context, amountWithoutFee, from);
emit RedeemToken(groupId, preparedToken.toAddress(), from, _msgSender(), baseOut);
// Handle any necessary after-redeem hooks
if (hookAddress != address(0)) {
IHooks hook = IHooks(hookAddress);
HookLibrary.afterRedeem(hook, groupId, context);
}
return baseOut;
}
/** Get user operation times for a group.
* @return lastVTMintTime The last VT mint time for the user.
* @return lastYTMintTime The last YT mint time for the user.
* @return lastVTRedeemTime The last VT redeem time for the user.
* @return lastYTRedeemTime The last YT redeem time for the user.
*/
function getUserOperationTime(
GroupId groupId,
address user
) external view returns (uint256 lastVTMintTime, uint256 lastYTMintTime, uint256 lastVTRedeemTime, uint256 lastYTRedeemTime) {
(lastVTMintTime, lastYTMintTime, lastVTRedeemTime, lastYTRedeemTime) = timeStorage.getUserOperationTime(groupId, user);
return (lastVTMintTime, lastYTMintTime, lastVTRedeemTime, lastYTRedeemTime);
}
/**
* @notice Returns the group mint flag.
* @param groupId The group identifier.
* @return True if minting is enabled for the group, false otherwise.
*/
function getGroupMintFlag(GroupId groupId) external view returns (bool) {
return ProtocolConfigState.getMintFlag(configStorage, groupId);
}
/**
* @notice Returns the group redeem flag.
* @param groupId The group identifier.
* @return True if redeeming is enabled for the group, false otherwise.
*/
function getGroupRedeemFlag(GroupId groupId) external view returns (bool) {
return ProtocolConfigState.getRedeemFlag(configStorage, groupId);
}
/**
* @notice Returns the group stability flag.
* @param groupId The group identifier.
* @return True if the group is in stability mode, false otherwise.
*/
function getGroupStabilityFlag(GroupId groupId) external view returns (bool) {
return ProtocolConfigState.getStabilityFlag(configStorage, groupId);
}
/**
* @notice Returns the global mint flag.
* @return True if minting is globally enabled, false otherwise.
*/
function getGlobalMintFlag() external view returns (bool) {
return ProtocolConfigState.getGlobalMintFlag(configStorage);
}
/**
* @notice Returns the global redeem flag.
* @return True if redeeming is globally enabled, false otherwise.
*/
function getGlobalRedeemFlag() external view returns (bool) {
return ProtocolConfigState.getGlobalRedeemFlag(configStorage);
}
/**
* @notice Returns the stability ratio for a group.
* @param groupId The group identifier.
* @return The stability ratio.
*/
function stabilityRatio(GroupId groupId) external view returns (uint96) {
GroupState memory groupState = cacheStorage.getGroupState(groupId);
return GroupStateHelper.getStabilityRatio(GroupSettings.wrap(groupState.groupSettings));
}
/**
* @notice Sets the global mint flag.
* @param value The new value for the global mint flag.
*/
function setGlobalMintFlag(bool value) external onlyRole(DEFAULT_ADMIN_ROLE) {
ProtocolConfigState.setGlobalMintFlag(configStorage, value);
}
/**
* @notice Sets the global redeem flag.
* @param value The new value for the global redeem flag.
*/
function setGlobalRedeemFlag(bool value) external onlyRole(DEFAULT_ADMIN_ROLE) {
ProtocolConfigState.setGlobalRedeemFlag(configStorage, value);
}
/**
* @notice Sets the mint flag for a specific group.
* @param groupId The group identifier.
* @param value The new value for the group's mint flag.
*/
function setGroupMintFlag(GroupId groupId, bool value) external onlyRole(DEFAULT_ADMIN_ROLE) {
ProtocolConfigState.setMintFlag(configStorage, groupId, value);
}
/**
* @notice Sets the redeem flag for a specific group.
* @param groupId The group identifier.
* @param value The new value for the group's redeem flag.
*/
function setGroupRedeemFlag(GroupId groupId, bool value) external onlyRole(DEFAULT_ADMIN_ROLE) {
ProtocolConfigState.setRedeemFlag(configStorage, groupId, value);
}
/**
* @notice Sets the stability flag for a specific group.
* @param groupId The group identifier.
* @param value The new value for the group's stability flag.
*/
function setGroupStabilityFlag(GroupId groupId, bool value) external onlyRole(DEFAULT_ADMIN_ROLE) {
ProtocolConfigState.setStabilityFlag(configStorage, groupId, value);
}
/**
* @notice Gets the amount of owed fees for a hook contract.
* @param groupId The group ID.
* @param hookContract The address of the hook contract.
* @param token The token to settle fees in.
* @return The amount of owed fees.
*/
function getOwedFees(GroupId groupId, address hookContract, address token) external view returns (uint256) {
return feeStorage.getOwedFees(groupId, hookContract, token);
}
/**
* @notice Allows the admin to recover ERC20 tokens mistakenly sent to the contract.
* @param tokenAddress The address of the token to recover.
* @param tokenAmount The amount of tokens to recover.
* @dev Only the admin can call this function. And normally protocol not keeps erc20 tokens apart from hook owed fees.
*/
function recoverERC20(address tokenAddress, uint256 tokenAmount) external nonReentrant onlyRole(DEFAULT_ADMIN_ROLE) {
if (tokenAddress == address(0)) {
IProtocol.ZeroAddress.selector.revertWith();
}
Currency token = Currency.wrap(tokenAddress);
uint256 currentBalance = token.balanceOf(address(this));
if (tokenAmount > currentBalance) {
IProtocol.InsufficientBalance.selector.revertWith(tokenAddress);
}
/// @dev Transfer the amount to the admin- whoever calls this function
token.safeTransfer(adminAddress, tokenAmount);
emit ERC20Recovered(token, tokenAmount);
}
/**
* @notice Allows the admin to recover ETH mistakenly sent to the contract.
* @param amount The amount of ETH to recover.
* @dev Only the admin can call this function.
* @dev Normally protocol not keeps ETH.
* @notice Slither false-positive. The function is only accessible by the admin.
*/
//slither-disable-next-line low-level-calls
function recoversEther(uint256 amount) external nonReentrant onlyRole(DEFAULT_ADMIN_ROLE) {
if (amount > address(this).balance) {
IProtocol.InsufficientBalance.selector.revertWith(address(0));
}
emit NativeTokenWithdrawn(amount);
/// @dev Transfer the amount to the admin- whoever calls this function
(bool success, ) = payable(adminAddress).call{ value: amount }("");
if (!success) revert IProtocol.NativeTokenWithdrawnFailed(amount);
}
/**
* @notice Prevents renouncing roles for security
* @dev Overrides the default renounceRole function to prevent accidental role removal
*/
function renounceRole(bytes32, address) public virtual override {
revert("Roles can't be renounced");
}
/**
* @notice Checks if the contract supports a specific interface.
* @param interfaceId The interface identifier.
* @return True if the interface is supported, false otherwise.
*/
function supportsInterface(bytes4 interfaceId) public view override(AccessControlUpgradeable) returns (bool) {
return interfaceId == type(IProtocol).interfaceId || super.supportsInterface(interfaceId);
}
// slither-disable-next-line unused-state
uint256[50] private __gap; // Storage gap for upgradeability
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.9.0) (access/AccessControl.sol)
pragma solidity ^0.8.0;
import "./IAccessControlUpgradeable.sol";
import "../utils/ContextUpgradeable.sol";
import "../utils/StringsUpgradeable.sol";
import "../utils/introspection/ERC165Upgradeable.sol";
import {Initializable} from "../proxy/utils/Initializable.sol";
/**
* @dev Contract module that allows children to implement role-based access
* control mechanisms. This is a lightweight version that doesn't allow enumerating role
* members except through off-chain means by accessing the contract event logs. Some
* applications may benefit from on-chain enumerability, for those cases see
* {AccessControlEnumerable}.
*
* Roles are referred to by their `bytes32` identifier. These should be exposed
* in the external API and be unique. The best way to achieve this is by
* using `public constant` hash digests:
*
* ```solidity
* bytes32 public constant MY_ROLE = keccak256("MY_ROLE");
* ```
*
* Roles can be used to represent a set of permissions. To restrict access to a
* function call, use {hasRole}:
*
* ```solidity
* function foo() public {
* require(hasRole(MY_ROLE, msg.sender));
* ...
* }
* ```
*
* Roles can be granted and revoked dynamically via the {grantRole} and
* {revokeRole} functions. Each role has an associated admin role, and only
* accounts that have a role's admin role can call {grantRole} and {revokeRole}.
*
* By default, the admin role for all roles is `DEFAULT_ADMIN_ROLE`, which means
* that only accounts with this role will be able to grant or revoke other
* roles. More complex role relationships can be created by using
* {_setRoleAdmin}.
*
* WARNING: The `DEFAULT_ADMIN_ROLE` is also its own admin: it has permission to
* grant and revoke this role. Extra precautions should be taken to secure
* accounts that have been granted it. We recommend using {AccessControlDefaultAdminRules}
* to enforce additional security measures for this role.
*/
abstract contract AccessControlUpgradeable is Initializable, ContextUpgradeable, IAccessControlUpgradeable, ERC165Upgradeable {
struct RoleData {
mapping(address => bool) members;
bytes32 adminRole;
}
mapping(bytes32 => RoleData) private _roles;
bytes32 public constant DEFAULT_ADMIN_ROLE = 0x00;
/**
* @dev Modifier that checks that an account has a specific role. Reverts
* with a standardized message including the required role.
*
* The format of the revert reason is given by the following regular expression:
*
* /^AccessControl: account (0x[0-9a-f]{40}) is missing role (0x[0-9a-f]{64})$/
*
* _Available since v4.1._
*/
modifier onlyRole(bytes32 role) {
_checkRole(role);
_;
}
function __AccessControl_init() internal onlyInitializing {
}
function __AccessControl_init_unchained() internal onlyInitializing {
}
/**
* @dev See {IERC165-supportsInterface}.
*/
function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
return interfaceId == type(IAccessControlUpgradeable).interfaceId || super.supportsInterface(interfaceId);
}
/**
* @dev Returns `true` if `account` has been granted `role`.
*/
function hasRole(bytes32 role, address account) public view virtual override returns (bool) {
return _roles[role].members[account];
}
/**
* @dev Revert with a standard message if `_msgSender()` is missing `role`.
* Overriding this function changes the behavior of the {onlyRole} modifier.
*
* Format of the revert message is described in {_checkRole}.
*
* _Available since v4.6._
*/
function _checkRole(bytes32 role) internal view virtual {
_checkRole(role, _msgSender());
}
/**
* @dev Revert with a standard message if `account` is missing `role`.
*
* The format of the revert reason is given by the following regular expression:
*
* /^AccessControl: account (0x[0-9a-f]{40}) is missing role (0x[0-9a-f]{64})$/
*/
function _checkRole(bytes32 role, address account) internal view virtual {
if (!hasRole(role, account)) {
revert(
string(
abi.encodePacked(
"AccessControl: account ",
StringsUpgradeable.toHexString(account),
" is missing role ",
StringsUpgradeable.toHexString(uint256(role), 32)
)
)
);
}
}
/**
* @dev Returns the admin role that controls `role`. See {grantRole} and
* {revokeRole}.
*
* To change a role's admin, use {_setRoleAdmin}.
*/
function getRoleAdmin(bytes32 role) public view virtual override returns (bytes32) {
return _roles[role].adminRole;
}
/**
* @dev Grants `role` to `account`.
*
* If `account` had not been already granted `role`, emits a {RoleGranted}
* event.
*
* Requirements:
*
* - the caller must have ``role``'s admin role.
*
* May emit a {RoleGranted} event.
*/
function grantRole(bytes32 role, address account) public virtual override onlyRole(getRoleAdmin(role)) {
_grantRole(role, account);
}
/**
* @dev Revokes `role` from `account`.
*
* If `account` had been granted `role`, emits a {RoleRevoked} event.
*
* Requirements:
*
* - the caller must have ``role``'s admin role.
*
* May emit a {RoleRevoked} event.
*/
function revokeRole(bytes32 role, address account) public virtual override onlyRole(getRoleAdmin(role)) {
_revokeRole(role, account);
}
/**
* @dev Revokes `role` from the calling account.
*
* Roles are often managed via {grantRole} and {revokeRole}: this function's
* purpose is to provide a mechanism for accounts to lose their privileges
* if they are compromised (such as when a trusted device is misplaced).
*
* If the calling account had been revoked `role`, emits a {RoleRevoked}
* event.
*
* Requirements:
*
* - the caller must be `account`.
*
* May emit a {RoleRevoked} event.
*/
function renounceRole(bytes32 role, address account) public virtual override {
require(account == _msgSender(), "AccessControl: can only renounce roles for self");
_revokeRole(role, account);
}
/**
* @dev Grants `role` to `account`.
*
* If `account` had not been already granted `role`, emits a {RoleGranted}
* event. Note that unlike {grantRole}, this function doesn't perform any
* checks on the calling account.
*
* May emit a {RoleGranted} event.
*
* [WARNING]
* ====
* This function should only be called from the constructor when setting
* up the initial roles for the system.
*
* Using this function in any other way is effectively circumventing the admin
* system imposed by {AccessControl}.
* ====
*
* NOTE: This function is deprecated in favor of {_grantRole}.
*/
function _setupRole(bytes32 role, address account) internal virtual {
_grantRole(role, account);
}
/**
* @dev Sets `adminRole` as ``role``'s admin role.
*
* Emits a {RoleAdminChanged} event.
*/
function _setRoleAdmin(bytes32 role, bytes32 adminRole) internal virtual {
bytes32 previousAdminRole = getRoleAdmin(role);
_roles[role].adminRole = adminRole;
emit RoleAdminChanged(role, previousAdminRole, adminRole);
}
/**
* @dev Grants `role` to `account`.
*
* Internal function without access restriction.
*
* May emit a {RoleGranted} event.
*/
function _grantRole(bytes32 role, address account) internal virtual {
if (!hasRole(role, account)) {
_roles[role].members[account] = true;
emit RoleGranted(role, account, _msgSender());
}
}
/**
* @dev Revokes `role` from `account`.
*
* Internal function without access restriction.
*
* May emit a {RoleRevoked} event.
*/
function _revokeRole(bytes32 role, address account) internal virtual {
if (hasRole(role, account)) {
_roles[role].members[account] = false;
emit RoleRevoked(role, account, _msgSender());
}
}
/**
* @dev This empty reserved space is put in place to allow future versions to add new
* variables without shifting down storage in the inheritance chain.
* See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
*/
uint256[49] private __gap;
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.9.0) (security/ReentrancyGuard.sol)
pragma solidity ^0.8.0;
import {Initializable} from "../proxy/utils/Initializable.sol";
/**
* @dev Contract module that helps prevent reentrant calls to a function.
*
* Inheriting from `ReentrancyGuard` will make the {nonReentrant} modifier
* available, which can be applied to functions to make sure there are no nested
* (reentrant) calls to them.
*
* Note that because there is a single `nonReentrant` guard, functions marked as
* `nonReentrant` may not call one another. This can be worked around by making
* those functions `private`, and then adding `external` `nonReentrant` entry
* points to them.
*
* TIP: If you would like to learn more about reentrancy and alternative ways
* to protect against it, check out our blog post
* https://blog.openzeppelin.com/reentrancy-after-istanbul/[Reentrancy After Istanbul].
*/
abstract contract ReentrancyGuardUpgradeable is Initializable {
// Booleans are more expensive than uint256 or any type that takes up a full
// word because each write operation emits an extra SLOAD to first read the
// slot's contents, replace the bits taken up by the boolean, and then write
// back. This is the compiler's defense against contract upgrades and
// pointer aliasing, and it cannot be disabled.
// The values being non-zero value makes deployment a bit more expensive,
// but in exchange the refund on every call to nonReentrant will be lower in
// amount. Since refunds are capped to a percentage of the total
// transaction's gas, it is best to keep them low in cases like this one, to
// increase the likelihood of the full refund coming into effect.
uint256 private constant _NOT_ENTERED = 1;
uint256 private constant _ENTERED = 2;
uint256 private _status;
function __ReentrancyGuard_init() internal onlyInitializing {
__ReentrancyGuard_init_unchained();
}
function __ReentrancyGuard_init_unchained() internal onlyInitializing {
_status = _NOT_ENTERED;
}
/**
* @dev Prevents a contract from calling itself, directly or indirectly.
* Calling a `nonReentrant` function from another `nonReentrant`
* function is not supported. It is possible to prevent this from happening
* by making the `nonReentrant` function external, and making it call a
* `private` function that does the actual work.
*/
modifier nonReentrant() {
_nonReentrantBefore();
_;
_nonReentrantAfter();
}
function _nonReentrantBefore() private {
// On the first call to nonReentrant, _status will be _NOT_ENTERED
require(_status != _ENTERED, "ReentrancyGuard: reentrant call");
// Any calls to nonReentrant after this point will fail
_status = _ENTERED;
}
function _nonReentrantAfter() private {
// By storing the original value once again, a refund is triggered (see
// https://eips.ethereum.org/EIPS/eip-2200)
_status = _NOT_ENTERED;
}
/**
* @dev Returns true if the reentrancy guard is currently set to "entered", which indicates there is a
* `nonReentrant` function in the call stack.
*/
function _reentrancyGuardEntered() internal view returns (bool) {
return _status == _ENTERED;
}
/**
* @dev This empty reserved space is put in place to allow future versions to add new
* variables without shifting down storage in the inheritance chain.
* See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
*/
uint256[49] private __gap;
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { WordCodec } from "./WordCodec.sol";
import { GroupId } from "../types/GroupId.sol";
import { IProtocol } from "../interfaces/IProtocol.sol";
import { CustomRevert } from "./CustomRevert.sol";
/// @title Protocol Config State Library
/// @notice Provides functions to manage global and group-specific configuration flags.
library ProtocolConfigState {
using WordCodec for bytes32;
using CustomRevert for bytes4;
uint256 internal constant GLOBAL_MINT_FLAG_OFFSET = 0; // Offset for the global mint flag (boolean)
uint256 internal constant GLOBAL_REDEEM_FLAG_OFFSET = 1; // Offset for the global redeem flag (boolean)
uint256 internal constant GLOBAL_STABILITY_FLAG_OFFSET = 2; // Offset for the global stability flag (boolean)
// Offsets for per-group configuration flags (bits 0 to 2)
uint256 internal constant GROUP_MINT_FLAG_OFFSET = 0; // Offset for the group mint flag (boolean)
uint256 internal constant GROUP_REDEEM_FLAG_OFFSET = 1; // Offset for the group redeem flag (boolean)
uint256 internal constant GROUP_STABILITY_FLAG_OFFSET = 2; // Offset for the group stability flag (boolean)
error ZeroGroupKey();
// Bits 3 to 255 are reserved for future use
/// @notice Configuration storage structure.
struct ConfigStorage {
/// @notice Encoded protocol-wide configuration data.
bytes32 protocolConfigData;
/// @notice Mapping of group IDs to their encoded configuration data.
mapping(bytes32 => bytes32) groupConfigData;
}
/// @notice Retrieves the global mint flag.
/// @param self The configuration storage.
/// @return True if global minting is enabled, false otherwise.
function getGlobalMintFlag(ConfigStorage storage self) internal view returns (bool) {
return self.protocolConfigData.decodeBool(GLOBAL_MINT_FLAG_OFFSET);
}
/// @notice Sets the global mint flag.
/// @param self The configuration storage.
/// @param value The new value for the global mint flag.
function setGlobalMintFlag(ConfigStorage storage self, bool value) internal {
self.protocolConfigData = self.protocolConfigData.insertBool(value, GLOBAL_MINT_FLAG_OFFSET);
}
/// @notice Retrieves the global redeem flag.
/// @param self The configuration storage.
/// @return True if global redeeming is enabled, false otherwise.
function getGlobalRedeemFlag(ConfigStorage storage self) internal view returns (bool) {
return self.protocolConfigData.decodeBool(GLOBAL_REDEEM_FLAG_OFFSET);
}
/// @notice Sets the global redeem flag.
/// @param self The configuration storage.
/// @param value The new value for the global redeem flag.
function setGlobalRedeemFlag(ConfigStorage storage self, bool value) internal {
self.protocolConfigData = self.protocolConfigData.insertBool(value, GLOBAL_REDEEM_FLAG_OFFSET);
}
/// @notice Retrieves the global stability flag.
/// @param self The configuration storage.
/// @return True if global stability is enabled, false otherwise.
function getGlobalStabilityFlag(ConfigStorage storage self) internal view returns (bool) {
return self.protocolConfigData.decodeBool(GLOBAL_STABILITY_FLAG_OFFSET);
}
/// @notice Sets the global stability flag.
/// @param self The configuration storage.
/// @param value The new value for the global stability flag.
function setGlobalStabilityFlag(ConfigStorage storage self, bool value) internal {
self.protocolConfigData = self.protocolConfigData.insertBool(value, GLOBAL_STABILITY_FLAG_OFFSET);
}
/// @notice Retrieves the mint flag for a specific group.
/// @param self The configuration storage.
/// @param groupId The group ID.
/// @return True if minting is enabled for the group, false otherwise.
function getMintFlag(ConfigStorage storage self, GroupId groupId) internal view returns (bool) {
bytes32 groupKey = _validateAndGetGroupKey(groupId);
return self.groupConfigData[groupKey].decodeBool(GROUP_MINT_FLAG_OFFSET);
}
/// @notice Sets the mint flag for a specific group.
/// @param self The configuration storage.
/// @param groupId The group ID.
/// @param value The new value for the group's mint flag.
function setMintFlag(ConfigStorage storage self, GroupId groupId, bool value) internal {
bytes32 groupKey = _validateAndGetGroupKey(groupId);
self.groupConfigData[groupKey] = self.groupConfigData[groupKey].insertBool(value, GROUP_MINT_FLAG_OFFSET);
}
/// @notice Retrieves the redeem flag for a specific group.
/// @param self The configuration storage.
/// @param groupId The group ID.
/// @return True if redeeming is enabled for the group, false otherwise.
function getRedeemFlag(ConfigStorage storage self, GroupId groupId) internal view returns (bool) {
bytes32 groupKey = _validateAndGetGroupKey(groupId);
return self.groupConfigData[groupKey].decodeBool(GROUP_REDEEM_FLAG_OFFSET);
}
/// @notice Sets the redeem flag for a specific group.
/// @param self The configuration storage.
/// @param groupId The group ID.
/// @param value The new value for the group's redeem flag.
function setRedeemFlag(ConfigStorage storage self, GroupId groupId, bool value) internal {
bytes32 groupKey = _validateAndGetGroupKey(groupId);
self.groupConfigData[groupKey] = self.groupConfigData[groupKey].insertBool(value, GROUP_REDEEM_FLAG_OFFSET);
}
/// @notice Retrieves the stability flag for a specific group.
/// @param self The configuration storage.
/// @param groupId The group ID.
/// @return True if stability is enabled for the group, false otherwise.
function getStabilityFlag(ConfigStorage storage self, GroupId groupId) internal view returns (bool) {
bytes32 groupKey = _validateAndGetGroupKey(groupId);
return self.groupConfigData[groupKey].decodeBool(GROUP_STABILITY_FLAG_OFFSET);
}
/// @notice Sets the stability flag for a specific group.
/// @param self The configuration storage.
/// @param groupId The group ID.
/// @param value The new value for the group's stability flag.
function setStabilityFlag(ConfigStorage storage self, GroupId groupId, bool value) internal {
bytes32 groupKey = _validateAndGetGroupKey(groupId);
self.groupConfigData[groupKey] = self.groupConfigData[groupKey].insertBool(value, GROUP_STABILITY_FLAG_OFFSET);
}
/// @notice Validates the group ID and returns the corresponding group key.
/// @param groupId The group ID to validate.
/// @return The group key derived from the group ID.
function _validateAndGetGroupKey(GroupId groupId) private pure returns (bytes32) {
bytes32 groupKey = GroupId.unwrap(groupId);
if (groupKey == bytes32(0)) {
ZeroGroupKey.selector.revertWith();
}
return groupKey;
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { Currency, CurrencyLibrary } from "../types/Currency.sol";
import { GroupId } from "../types/GroupId.sol";
import { DOperationContext } from "../declarations/DOperationContext.sol";
import { OperationContextLibrary } from "./OperationContextLibrary.sol";
import { ProtocolUtils } from "./ProtocolUtils.sol";
import { CustomRevert } from "./CustomRevert.sol";
import { IProtocol } from "../interfaces/IProtocol.sol";
import { Address, AddressLibrary } from "../types/Address.sol";
import { TimeManagementLibrary } from "./TimeManagementLibrary.sol";
/**
* @title FeeManagementLibrary
* @notice Library for managing protocol fees and treasury-related operations
*/
library FeeManagementLibrary {
using CurrencyLibrary for Currency;
using OperationContextLibrary for DOperationContext.OperationContext;
using TimeManagementLibrary for TimeManagementLibrary.TimeStorage;
using AddressLibrary for Address;
using CustomRevert for bytes4;
/// @dev The storage struct for fee management
struct FeeStorage {
mapping(GroupId => mapping(address => mapping(Currency => uint256))) owedFees;
}
// Events
event FeeDelegated(GroupId indexed groupId, address indexed hookContract, uint256 directFeeAmount, uint256 hookFeeAmount);
event FeeCollected(GroupId indexed groupId, address indexed feeCollector, address indexed token, uint256 amount);
event FeesSettled(GroupId indexed groupId, address indexed hookContract, address indexed token, uint256 amount);
// Errors
error ZeroAmount(GroupId groupId);
error ZeroAddress(GroupId groupId);
error NoFeesOwed(GroupId groupId);
error InsufficientBalance(GroupId groupId, address token);
/**
* @notice Distributes fees between the fee collector and hook contract
* @param self The fee storage
* @param context The operation context
* @param feeToken The token to use for fees
* @param feeSplit The fee split information
*/
function distributeFees(
FeeStorage storage self,
DOperationContext.OperationContext memory context,
Currency feeToken,
ProtocolUtils.FeeSplit memory feeSplit
) internal {
if (feeSplit.directFeeAmount > 0 || feeSplit.protocolFeeAmount > 0) {
address hookContract = context.getHookContract();
// Handle hook share
if (feeSplit.hookFeeAmount > 0 && hookContract != address(0)) {
// Accumulate owed fees to hook
self.owedFees[context.groupId][hookContract][feeToken] += feeSplit.hookFeeAmount;
emit FeeDelegated(context.groupId, hookContract, feeSplit.directFeeAmount, feeSplit.hookFeeAmount);
}
}
// Handle direct fee emit
if (feeSplit.directFeeAmount > 0) {
emit FeeCollected(context.groupId, context.getFeeCollector().toAddress(), feeToken.toAddress(), feeSplit.directFeeAmount);
}
}
/**
* @notice Prepares tokens for treasury operations
* @param /self The fee storage
* @param context The operation context
* @param baseIn The amount of base tokens
* @param isMinting Whether this is a minting operation
* @return preparedToken The token prepared for treasury
* @return amountWithoutFee The amount after fees
* @return feeSplit The fee split details
*/
function prepareTokenForTreasury(
FeeStorage storage /*self*/,
TimeManagementLibrary.TimeStorage storage timeStorage,
DOperationContext.OperationContext memory context,
uint256 baseIn,
bool isMinting
) internal returns (Currency preparedToken, uint256 amountWithoutFee, ProtocolUtils.FeeSplit memory feeSplit) {
if (baseIn == 0) {
ZeroAmount.selector.revertWith(context.groupId);
}
// Update operation time
timeStorage.updateOperationTime(context.groupId, context.getOperationType(), context.getRecipient(), isMinting);
// Check for early exit fee if redeeming
if (!isMinting) {
(bool applyEarlyFee, uint24 additionalFeeBps) = timeStorage.checkEarlyExitFee(context, context.getRecipient());
if (applyEarlyFee) {
uint24 currentFee = context.getContextFee();
uint24 newFee = currentFee + additionalFeeBps;
context.setContextFee(newFee);
}
}
// Prepare token for treasury
if (isMinting) {
Currency paymentToken = context.getPaymentToken();
if (paymentToken.isNative()) baseIn = msg.value;
return ProtocolUtils.prepareTokenMintForTreasury(context, baseIn);
} else {
return ProtocolUtils.redeemTokens(context, baseIn, context.getRecipient());
}
}
/**
* @notice Allows hook contracts to settle owed fees
* @param self The fee storage
* @param groupId The group ID
* @param hookContract The hook contract address
* @param token The token to settle fees in
*/
function settleOwedFees(FeeStorage storage self, GroupId groupId, address hookContract, address token) internal {
if (token == address(0)) {
ZeroAddress.selector.revertWith(groupId);
}
Currency tokenCurrency = Currency.wrap(token);
uint256 owedAmount = self.owedFees[groupId][hookContract][tokenCurrency];
if (owedAmount == 0) {
NoFeesOwed.selector.revertWith(groupId);
}
self.owedFees[groupId][hookContract][tokenCurrency] = 0;
// Check if protocol has enough balance
if (tokenCurrency.isNative()) IProtocol.NotPermitted.selector.revertWith();
if (tokenCurrency.balanceOf(address(this)) < owedAmount) {
InsufficientBalance.selector.revertWith(groupId, token);
}
tokenCurrency.safeTransfer(hookContract, owedAmount);
emit FeesSettled(groupId, hookContract, token, owedAmount);
}
/**
* @notice Gets the amount of owed fees for a hook contract
* @param self The fee storage
* @param groupId The group ID
* @param hookContract The hook contract address
* @param token The token to check fees in
* @return The amount of owed fees
*/
function getOwedFees(FeeStorage storage self, GroupId groupId, address hookContract, address token) internal view returns (uint256) {
return self.owedFees[groupId][hookContract][Currency.wrap(token)];
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { ITokenRegistryMinimum as ITokenRegistry } from "../interfaces/ITokenRegistryMinimum.sol";
import { DTokenRegistry } from "../declarations/DTokenRegistry.sol";
import { GroupId } from "../types/GroupId.sol";
import { IProtocol } from "../interfaces/IProtocol.sol";
import { Address, AddressLibrary } from "../types/Address.sol";
import { Currency, CurrencyLibrary } from "../types/Currency.sol";
import { CustomRevert } from "./CustomRevert.sol";
import { GroupStateHelper, GroupSettings } from "../types/GroupStateHelper.sol";
import { GroupState, CollateralInfo } from "../types/CommonTypes.sol";
library CacheLibrary {
using CurrencyLibrary for Currency;
using AddressLibrary for Address;
using CustomRevert for bytes4;
struct CachedGroupState {
GroupState data;
uint256 lastUpdateBlock;
bool exists;
}
struct Storage {
mapping(GroupId => CachedGroupState) cachedStates;
}
event CacheUpdated(GroupId indexed groupId);
error ConfigNotReady(GroupId groupId);
error InvalidTokenRegistry(address tokenRegistry);
error InvalidGroupConfiguration(GroupId groupId);
error EmptyCollateralListOnUpdate(GroupId groupId);
/**
* @notice Retrieves the cached group state, updating the cache if necessary.
* @param self The storage containing cached group states.
* @param groupId The ID of the group.
* @return The group state.
*/
function getGroupState(Storage storage self, GroupId groupId) internal view returns (GroupState memory) {
CachedGroupState storage cachedState = self.cachedStates[groupId];
if (!cachedState.exists) {
ConfigNotReady.selector.revertWith(groupId);
}
return cachedState.data;
}
/**
* @notice Updates the cache for a specific group ID.
* @param self The storage containing cached group states.
* @param tokenRegistry The token registry to fetch group state from.
* @param groupId The ID of the group.
*/
function updateCache(Storage storage self, ITokenRegistry tokenRegistry, GroupId groupId) internal {
CachedGroupState storage cachedState = self.cachedStates[groupId];
GroupState memory groupState = tokenRegistry.getGroup(groupId);
if (groupState.core.aToken.isZero()) {
InvalidGroupConfiguration.selector.revertWith(groupId);
}
uint256 len = groupState.acceptableCollaterals.length;
if (len > 0) {
delete cachedState.data.acceptableCollaterals;
for (uint256 i = 0; i < len; ) {
cachedState.data.acceptableCollaterals.push(groupState.acceptableCollaterals[i]);
unchecked {
++i;
}
}
} else {
EmptyCollateralListOnUpdate.selector.revertWith(groupId);
}
/// @dev Validate core tokens with simple zero checks.
/// @dev Token registry should have already validated these but this is another layer of fundamental validation.
_simpleValidateGroupCoreTokens(groupState.core, groupId);
_simpleValidateGroupExtendedTokens(groupState.extended, groupId);
cachedState.data.core = groupState.core;
cachedState.data.extended = groupState.extended;
cachedState.data.feesPacked = groupState.feesPacked;
cachedState.data.groupSettings = groupState.groupSettings;
cachedState.data.hookContract = groupState.hookContract;
cachedState.lastUpdateBlock = block.number;
cachedState.exists = true;
emit CacheUpdated(groupId);
}
/**
* @notice Validates the core tokens of a group.
* @param core The core tokens of the group.
* @param groupId The ID of the group.
* @dev This is a simple validation that checks for zero addresses and cartesian checks.
* @dev Token registry should have already validated these but this is another layer of fundamental validation.
*/
function _simpleValidateGroupCoreTokens(DTokenRegistry.GroupCore memory core, GroupId groupId) private pure {
if (core.aToken.isZero() || core.xToken.isZero() || core.baseToken.isZero() || core.yieldBearingToken.isZero()) {
InvalidGroupConfiguration.selector.revertWith(groupId);
}
/// @dev cartesian check
if (
core.aToken.equals(core.xToken) ||
core.aToken.equals(core.baseToken) ||
core.aToken.equals(core.yieldBearingToken) ||
core.xToken.equals(core.baseToken) ||
core.xToken.equals(core.yieldBearingToken) ||
core.baseToken.equals(core.yieldBearingToken)
) {
InvalidGroupConfiguration.selector.revertWith(groupId);
}
}
/**
* @notice Validates the extended tokens of a group.
* @param extended The extended tokens of the group.
* @param groupId The ID of the group.
* @dev This is a simple validation that checks for zero addresses.
* @dev Token registry should have already validated these but this is another layer of fundamental validation.
*/
function _simpleValidateGroupExtendedTokens(DTokenRegistry.GroupExtended memory extended, GroupId groupId) private pure {
if (
extended.priceOracle.isZero() ||
extended.rateProvider.isZero() ||
extended.swapRouter.isZero() ||
extended.treasury.isZero() ||
extended.feeCollector.isZero() ||
extended.strategy.isZero() ||
extended.rebalancePool.isZero()
) InvalidGroupConfiguration.selector.revertWith(groupId);
}
/**
* @notice Forces the cache to update for a specific group ID.
* @param self The storage containing cached group states.
* @param tokenRegistry The token registry to fetch group state from.
* @param groupId The ID of the group.
*/
function forceUpdate(Storage storage self, address tokenRegistry, GroupId groupId) internal {
if (tokenRegistry == address(0)) InvalidTokenRegistry.selector.revertWith(groupId);
updateCache(self, ITokenRegistry(tokenRegistry), groupId);
}
function getGroupTreasury(Storage storage self, GroupId groupId) internal view returns (address) {
if (!self.cachedStates[groupId].exists) {
ConfigNotReady.selector.revertWith(groupId);
}
return self.cachedStates[groupId].data.extended.treasury.toAddress();
}
function getGroupHookContract(Storage storage self, GroupId groupId) internal view returns (address) {
if (!self.cachedStates[groupId].exists) {
ConfigNotReady.selector.revertWith(groupId);
}
return self.cachedStates[groupId].data.hookContract.toAddress();
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { CustomRevert } from "./CustomRevert.sol";
import { GroupId } from "../types/GroupId.sol";
import { IHooks } from "../interfaces/IHooks.sol";
import { DOperationContext } from "../declarations/DOperationContext.sol";
import { OperationContextLibrary } from "./OperationContextLibrary.sol";
import { GroupFeeLibrary } from "./GroupFeeLibrary.sol";
/**
* @title HookLibrary
* @notice Provides utilities to manage hooks related to minting, redeeming.
* It handles the invocation of hook functions and manages `hookData` only when necessary.
* @dev Hooks are for internal purpose so its generally safer
*/
library HookLibrary {
using GroupFeeLibrary for uint24;
using OperationContextLibrary for DOperationContext.OperationContext;
using CustomRevert for bytes4;
uint256 internal constant MAX_HOOK_DATA_LENGTH = 1024; // 1 KB limit for hookData
uint256 internal constant EMPTY_BYTES_LENGTH = 32; // bytes length when empty
uint256 internal constant MIN_HOOK_RESPONSE_LENGTH = 32 + 32; // bytes4 selector + empty bytes updatedHookData
uint256 internal constant MAX_HOOK_RESPONSE_LENGTH = MIN_HOOK_RESPONSE_LENGTH + MAX_HOOK_DATA_LENGTH; // bytes4 selector + bytes hookData + uint24 lpFeeOverride
/// @dev Thrown when the hook response is invalid.
error InvalidHookResponse(GroupId groupId);
/// @dev Thrown when the provided hook contract address is invalid.
/// @param hookContract The address of the invalid hook contract.
error InvalidHookContract(address hookContract);
/// @dev Thrown when a hook call fails, providing the hook address and revert reason.
/// @param hookContract The address of the hook contract that failed.
/// @param revertReason The raw revert reason returned by the hook contract.
error Wrap__FailedHookCall(address hookContract, bytes revertReason);
/// @dev Emitted when a hook call fails.
/// @param hookContract The address of the hook contract.
/// @param hookType The type of hook attempted.
/// @param sender The address initiating the hook.
/// @param groupId The identifier of the group.
/// @param reason The reason for the hook failure.
event HookFailed(GroupId indexed groupId, address indexed hookContract, address indexed sender, string hookType, string reason);
/**
* @dev Defines the structure of permissions for various hook operations.
*/
struct Permissions {
bool beforeMint;
bool afterMint;
bool beforeRedeem;
bool afterRedeem;
}
// Permission Flags
uint16 internal constant BEFORE_MINT_FLAG = 1 << 0;
uint16 internal constant AFTER_MINT_FLAG = 1 << 1;
uint16 internal constant BEFORE_REDEEM_FLAG = 1 << 2;
uint16 internal constant AFTER_REDEEM_FLAG = 1 << 3;
// Mask of all possible flags
uint16 internal constant ALL_HOOK_MASK = BEFORE_MINT_FLAG | AFTER_MINT_FLAG | BEFORE_REDEEM_FLAG | AFTER_REDEEM_FLAG;
// =========================
// ==== Permission Functions =====
// =========================
/**
* @dev Packs the permissions into a uint16.
* @param flags The Permissions struct containing individual permission flags.
* @return permissions The packed permissions as a uint16.
*/
function packPermissions(Permissions memory flags) internal pure returns (uint16 permissions) {
if (flags.beforeMint) permissions |= BEFORE_MINT_FLAG;
if (flags.afterMint) permissions |= AFTER_MINT_FLAG;
if (flags.beforeRedeem) permissions |= BEFORE_REDEEM_FLAG;
if (flags.afterRedeem) permissions |= AFTER_REDEEM_FLAG;
}
// =========================
// ==== Hook Contract Validation =====
// =========================
/**
* @dev Validates the permissions of a hook contract during initialization.
* @param self The IHooks contract.
* @param expectedPermissions The expected uint16 packed permissions
*/
function validateHookPermissions(IHooks self, uint16 expectedPermissions) internal view {
uint16 actualPermissions = getHookContractPermissions(self);
if (actualPermissions != expectedPermissions) {
InvalidHookContract.selector.revertWith(address(self));
}
}
/**
* @dev Checks if a hook contract is valid during initialization.
* @param self The IHooks contract.
* @param fee The fee associated with the hook.
* @return True if the hook contract is valid.
*/
function isValidHookContract(IHooks self, uint24 fee) internal view returns (bool) {
if (address(self) == address(0)) {
// If there is no hook contract set, then fee cannot be dynamic
return !fee.isDynamicFee();
} else {
// If a hook contract is set, it must have at least 1 flag set, or have a dynamic fee
uint16 permissions = getHookContractPermissions(self);
return (permissions & ALL_HOOK_MASK != 0) || fee.isDynamicFee();
}
}
/**
* @dev Retrieves the permissions from the hook contract.
* @param self The IHooks contract.
* @return The permissions as a uint16.
*/
function getHookContractPermissions(IHooks self) internal view returns (uint16) {
(bool success, bytes memory data) = address(self).staticcall(abi.encodeWithSelector(IHooks.getHookPermissions.selector));
if (!success || data.length == 0) {
InvalidHookContract.selector.bubbleUpAndRevertWith(address(self));
}
uint16 permissions = abi.decode(data, (uint16));
return permissions;
}
/**
* @dev Checks if the given permissions include the specified flag.
* @param actualPermissions The actual permissions represented as a uint256.
* @param flag The permission flag to check.
* @return True if the flag is set in the actualPermissions.
*/
function hasPermission(uint256 actualPermissions, uint16 flag) internal pure returns (bool) {
return uint16(actualPermissions) & flag != 0;
}
// =========================
// ==== Hook Invocation =====
// =========================
/**
* @dev Calls the hook contract with the provided data.
* @param self The IHooks contract.
* @param data The calldata to be sent to the hook contract.
* @return result The result returned by the hook contract.
*/
function callHook(IHooks self, bytes memory data) internal returns (bytes memory result) {
bool success;
assembly ("memory-safe") {
success := call(gas(), self, 0, add(data, 32), mload(data), 0, 0)
let size := returndatasize()
result := mload(0x40) // Load the free memory pointer
mstore(0x40, add(result, and(add(size, 63), not(31)))) // Update free memory pointer
mstore(result, size) // Set the length of result
returndatacopy(add(result, 32), 0, size) // Copy return data
}
if (!success) {
if (!success) Wrap__FailedHookCall.selector.bubbleUpAndRevertWith(address(self));
}
}
// =========================
// ==== Hook Operations =====
// =========================
/**
* @dev Modifier to prevent self-calls to the hook contract.
* @param self The IHooks contract.
*/
modifier noSelfCall(IHooks self) {
if (msg.sender != address(self)) {
_;
}
}
/**
* @dev Invokes the beforeMint hook if the permission is set.
* @param self The IHooks contract.
* @param groupId The GroupId for the operation.
* @param context The operation context.
*/
function beforeMint(IHooks self, GroupId groupId, DOperationContext.OperationContext memory context) internal noSelfCall(self) {
if (!hasPermission(context.getHookPermissions(), BEFORE_MINT_FLAG)) {
return;
}
bytes memory hookData = context.getHookData();
uint24 contextFee = context.getContextFee();
bool isDynamic = contextFee & GroupFeeLibrary.DYNAMIC_FEE_FLAG != 0;
bool isDelegate = contextFee & GroupFeeLibrary.DELEGATE_FEE_FLAG != 0;
bytes memory data = abi.encodeWithSelector(IHooks.beforeMint.selector, msg.sender, groupId, context, hookData);
bytes memory result = callHook(self, data);
if (result.length == 0) {
// Hook call failed or returned empty result, proceed without changing context
return;
}
// Expected result is (bytes4 selector, bytes updatedHookData, 24 lpFeeOverride)
/// @notice Each primitive type gets padded to 32 bytes + bytes
if (result.length < MIN_HOOK_RESPONSE_LENGTH + MIN_HOOK_RESPONSE_LENGTH || result.length > MAX_HOOK_RESPONSE_LENGTH) {
InvalidHookResponse.selector.revertWith(context.groupId);
}
(bytes4 returnedSelector, bytes memory updatedHookData, uint24 lpFeeOverrideUint) = abi.decode(result, (bytes4, bytes, uint24));
if (returnedSelector != IHooks.beforeMint.selector) {
InvalidHookResponse.selector.revertWith(context.groupId);
}
(bool isValid, uint24 lpFeeOverride) = _sanitizeAndProcessFeeOverride(lpFeeOverrideUint);
if (isValid && lpFeeOverride > 0 && (isDynamic || isDelegate)) {
// Update the fee in the context
context.setOverriddenFee(lpFeeOverride);
}
// Update hookData in context
setHookData(context, updatedHookData);
}
/**
* @dev Sanitizes and processes the fee override value.
* @param _lpFeeOverrideUint The fee override value to sanitize.
* @return isValid True if the fee override is valid, false otherwise.
* @return fee The sanitized fee override value.
*/
function _sanitizeAndProcessFeeOverride(uint24 _lpFeeOverrideUint) internal pure returns (bool, uint24) {
uint24 sanitizedFee = _lpFeeOverrideUint & GroupFeeLibrary.REMOVE_ALL_FLAGS_MASK;
// If ZERO_FEE_FLAG is set, REQUIRE fee amount to be 0
if (_lpFeeOverrideUint & GroupFeeLibrary.ZERO_FEE_FLAG != 0) {
// If hook claims zero fee but sends non-zero amount, reject it
if (sanitizedFee != 0) {
return (false, 0);
}
return (true, GroupFeeLibrary.ZERO_FEE_FLAG);
}
// For non-zero fee cases
return (sanitizedFee > 0, sanitizedFee);
}
/**
* @dev Invokes the afterMint hook if the permission is set.
* @param self The IHooks contract.
* @param groupId The GroupId for the operation.
* @param context The operation context.
*/
function afterMint(IHooks self, GroupId groupId, DOperationContext.OperationContext memory context) internal noSelfCall(self) {
if (!hasPermission(context.getHookPermissions(), AFTER_MINT_FLAG)) {
return;
}
bytes memory hookData = context.getHookData();
bytes memory data = abi.encodeWithSelector(IHooks.afterMint.selector, msg.sender, groupId, context, hookData);
bytes memory result = callHook(self, data);
if (result.length == 0) {
// Hook call failed or returned empty result, proceed without changing context
return;
}
// Expected result is (bytes4 selector, bytes updatedHookData)
/// @notice Each primitive type gets padded to 32 bytes + bytes
if (result.length < MIN_HOOK_RESPONSE_LENGTH || result.length > MAX_HOOK_RESPONSE_LENGTH) {
InvalidHookResponse.selector.revertWith(context.groupId);
}
(bytes4 returnedSelector, bytes memory updatedHookData) = abi.decode(result, (bytes4, bytes));
if (returnedSelector != IHooks.afterMint.selector) {
InvalidHookResponse.selector.revertWith(context.groupId);
}
// Update hookData in context
setHookData(context, updatedHookData);
}
/**
* @dev Invokes the beforeRedeem hook if the permission is set.
* @param self The IHooks contract.
* @param groupId The GroupId for the operation.
* @param context The operation context.
*/
function beforeRedeem(IHooks self, GroupId groupId, DOperationContext.OperationContext memory context) internal noSelfCall(self) {
if (!hasPermission(context.getHookPermissions(), BEFORE_REDEEM_FLAG)) {
return;
}
bytes memory hookData = context.getHookData();
uint24 contextFee = context.getContextFee();
bool isDynamic = contextFee & GroupFeeLibrary.DYNAMIC_FEE_FLAG != 0;
bool isDelegate = contextFee & GroupFeeLibrary.DELEGATE_FEE_FLAG != 0;
bytes memory data = abi.encodeWithSelector(IHooks.beforeRedeem.selector, msg.sender, groupId, context, hookData);
bytes memory result = callHook(self, data);
if (result.length == 0) {
// Hook call failed or returned empty result, proceed without changing context
return;
}
// Expected result is (bytes4 selector, bytes updatedHookData, uint24 lpFeeOverride)
/// @notice Each primitive type gets padded to 32 bytes
if (result.length < MIN_HOOK_RESPONSE_LENGTH + MIN_HOOK_RESPONSE_LENGTH || result.length > MAX_HOOK_RESPONSE_LENGTH) {
InvalidHookResponse.selector.revertWith(context.groupId);
}
(bytes4 returnedSelector, bytes memory updatedHookData, uint24 lpFeeOverrideUint) = abi.decode(result, (bytes4, bytes, uint24));
if (returnedSelector != IHooks.beforeRedeem.selector) {
InvalidHookResponse.selector.revertWith(context.groupId);
}
(bool isValid, uint24 lpFeeOverride) = _sanitizeAndProcessFeeOverride(lpFeeOverrideUint);
if (isValid && lpFeeOverride > 0 && lpFeeOverride != context.getContextFee() && (isDynamic || isDelegate)) {
// Update the fee in the context
context.setOverriddenFee(lpFeeOverride);
}
// Update hookData in context
setHookData(context, updatedHookData);
}
/**
* @dev Invokes the afterRedeem hook if the permission is set.
* @param self The IHooks contract.
* @param groupId The GroupId for the operation.
* @param context The operation context.
*/
function afterRedeem(IHooks self, GroupId groupId, DOperationContext.OperationContext memory context) internal noSelfCall(self) {
if (!hasPermission(context.getHookPermissions(), AFTER_REDEEM_FLAG)) {
return;
}
bytes memory hookData = context.getHookData();
bytes memory data = abi.encodeWithSelector(IHooks.afterRedeem.selector, msg.sender, groupId, context, hookData);
bytes memory result = callHook(self, data);
if (result.length == 0) {
// Hook call failed or returned empty result, proceed without changing context
return;
}
// Expected result is (bytes4 selector, bytes updatedHookData)
/// @notice Each primitive type gets padded to 32 bytes + bytes
if (result.length < MIN_HOOK_RESPONSE_LENGTH || result.length > MAX_HOOK_RESPONSE_LENGTH) {
InvalidHookResponse.selector.revertWith(context.groupId);
}
(bytes4 returnedSelector, bytes memory updatedHookData) = abi.decode(result, (bytes4, bytes));
if (returnedSelector != IHooks.afterRedeem.selector) {
InvalidHookResponse.selector.revertWith(context.groupId);
}
// Update hookData in context
setHookData(context, updatedHookData);
}
// =========================
// ==== Helper Functions =====
// =========================
/**
* @dev Sets the hookData in the operation context.
* @param context The operation context.
* @param hookData The hook data to set.
*/
function setHookData(DOperationContext.OperationContext memory context, bytes memory hookData) internal pure {
context.setHookData(hookData);
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { Currency } from "../types/Currency.sol";
import { Address, AddressLibrary } from "../types/Address.sol";
import { ITreasury } from "../interfaces/ITreasury.sol";
import { IWToken } from "../interfaces/IWToken.sol";
import { IDEXRouter } from "../interfaces/IDEXRouter.sol";
import { IRebalancePool } from "../interfaces/IRebalancePool.sol";
import { OperationContextLibrary } from "./OperationContextLibrary.sol";
import { StabilityFeeOperations } from "./StabilityFeeOperations.sol";
import { DOperationContext } from "../declarations/DOperationContext.sol";
import { DProtocol } from "../declarations/DProtocol.sol";
import { CurrencyLibrary } from "../types/Currency.sol";
import { GroupFeeLibrary } from "./GroupFeeLibrary.sol";
import { GroupId } from "../types/GroupId.sol";
import { CustomRevert } from "./CustomRevert.sol";
import { IPriceOracle } from "../interfaces/IPriceOracle.sol";
import { FullMath } from "./math/FullMath.sol";
import { OperationTypes } from "../types/CommonTypes.sol";
import 'forge-std/console.sol';
/**
* @title ProtocolUtils
* @notice Library containing utility functions for protocol operations with integrated stability and fee management.
* @dev The logic here:
* - baseIn represents the amount of payment tokens the user is willing to pay (not the requested baseToken amount).
* - We first convert this payment token amount (baseIn) to an equivalent "base token" amount we could potentially mint.
* - Then we apply treasury limits (boundedAmount).
* - If the treasury can mint less than what baseIn implies, we scale down payment token usage accordingly.
* - Then we calculate fees on the final "effective base token" amount.
* - For ERC20 payment: we only now do a `transferFrom` for the exact required payment amount after adjusting for treasury limits and fees.
* - For native token payment: we received `msg.value`. After final calculations and swaps, we refund the unused portion at the end as yieldBearingToken.
* - We handle swaps, wrapping/unwrapping, and fee distributions accordingly.
*/
library ProtocolUtils {
using CurrencyLibrary for Currency;
using OperationContextLibrary for DOperationContext.OperationContext;
using AddressLibrary for Address;
using CustomRevert for bytes4;
using StabilityFeeOperations for DOperationContext.OperationContext;
/**
* @dev Custom errors for better debugging and revert reasons.
*/
error ZeroNativeAmount(GroupId groupId);
error ZeroAmount(GroupId groupId);
error ZeroAddress(GroupId groupId);
error NotPermitted(GroupId groupId);
error InsufficientPreparedAmount(GroupId groupId);
error InsufficientAllowance(GroupId groupId);
error SwapOpFailed(GroupId groupId);
error BaseTokenCannotBeUsedAsCollateral(GroupId groupId);
error WrappedAmountIsZero(GroupId groupId);
error NativeTokenWrappingNotAllowed(GroupId groupId);
error UnwrappedAmountIsZero(GroupId groupId);
error InvalidSwapRouter(GroupId groupId);
error InvalidOperationType(GroupId groupId);
error InvalidRate(GroupId groupId);
error InvalidMinimumAmount(GroupId groupId, uint256 minimumAmount, uint256 amountReceived);
error InvalidSlippageTolerance(GroupId groupId, uint256 minOutAmount, uint256 amountReceived);
error TransferFailed(GroupId groupId, address recipient, uint256 amount);
error ExcessiveSwapAmount(GroupId groupId);
struct FeeSplit {
uint256 directFeeAmount; // immediate fee (in yield bearing token terms)
uint256 hookFeeAmount; // hook fee amount (if applicable)
uint256 protocolFeeAmount; // protocol fee amount (if applicable)
uint24 appliedFeeRate; // fee rate applied
}
uint256 private constant PRECISION = 1e18;
uint256 private constant HUNDRED = 100;
uint256 private constant BASIS_POINTS = 10_000;
uint24 private constant MAX_SLIPPAGE_TOLERANCE = 1_000; // 10% bps
uint256 private constant REFUND_SAFETY_MARGIN = HUNDRED; // 1% in bps for refunds
/**
* @dev Emitted when a refund happens (in yieldBearingToken).
*/
event Refund(address indexed account, uint256 indexed safeRefundAmount);
// New event declarations
/**
* @dev Emitted when fees are collected.
*/
event FeeCollected(address indexed feeCollector, uint256 protocolFeeAmount, uint256 hookFeeAmount);
/**
* @dev Emitted when tokens are swapped.
*/
event TokensSwapped(address indexed fromToken, address indexed toToken, uint256 amountIn, uint256 amountOut);
/**
* @dev Emitted when `aTokens` are auto-compounded into the `RebalancePool`.
* @param amountIn The amount of `aTokens` deposited.
* @param amountOut The amount of tokens minted from the `RebalancePool`.
* @param recipient The address receiving the tokens from the `RebalancePool`.
*/
event AutoCompounded(uint256 indexed amountIn, uint256 indexed amountOut, address indexed recipient);
/**
* @notice Prepares tokens for treasury operations when user provides `baseIn` as the payment token amount they are willing to spend.
* @dev Sequence of operations:
* 1. Convert payment token to base token equivalent and check bounds
* 2. Scale down payment if needed based on bounds
* 3. Calculate fees on effective base amount
* 4. Handle payment (transfer for ERC20 or verify msg.value)
* 5. Swap to yieldBearingToken
* 6. Take protocol fees from received yieldBearingToken
* 7. Wrap remaining yieldBearingToken to baseToken
* 8. Handle excess for native payments (as yieldBearingToken)
*
* @param context The operation context
* @param baseIn The amount of payment token user is willing to pay (msg.value if native)
* @return preparedToken The base token prepared for treasury
* @return finalBaseAmount The final amount after fees
* @return feeSplit The fee split details
*/
function prepareTokenMintForTreasury(
DOperationContext.OperationContext memory context,
uint256 baseIn
) internal returns (Currency preparedToken, uint256 finalBaseAmount, FeeSplit memory feeSplit) {
GroupId groupId = context.groupId;
Currency baseToken = context.getBaseToken();
Currency yieldBearingToken = context.getYieldBearingToken();
Currency paymentToken = context.getPaymentToken();
if (baseToken.isZero() || yieldBearingToken.isZero()) {
ZeroAddress.selector.revertWith(groupId);
}
// 1. Calculate conversions and bounds first
(uint256 paymentTokenPrice, uint256 yieldBearingTokenPrice) = _getPrices(context);
uint256 paymentTokenDecimals = uint256(context.getPaymentTokenDecimals());
uint256 adjustedBaseIn = (baseIn * PRECISION) / (10 ** paymentTokenDecimals);
uint256 expectedBaseFromPayment = (adjustedBaseIn * paymentTokenPrice) / yieldBearingTokenPrice;
// 2. Check treasury bounds
StabilityFeeOperations.StabilityState memory stabilityState = StabilityFeeOperations.getStabilityState(
context,
expectedBaseFromPayment
);
uint256 boundedAmount = stabilityState.boundedAmount;
if (boundedAmount == 0) ZeroAmount.selector.revertWith(groupId);
// 3. Scale payment if needed based on bounds
uint256 effectiveBaseAmount;
uint256 effectivePaymentAmount;
if (expectedBaseFromPayment > boundedAmount) {
uint256 ratio = (boundedAmount * PRECISION) / expectedBaseFromPayment;
effectiveBaseAmount = boundedAmount;
effectivePaymentAmount = (baseIn * ratio) / PRECISION;
} else {
effectiveBaseAmount = expectedBaseFromPayment;
effectivePaymentAmount = baseIn;
}
if (effectiveBaseAmount == 0) ZeroAmount.selector.revertWith(groupId);
// 4. Calculate fees based on effective base amount
(uint256 amountWithoutFee, FeeSplit memory fs) = _calculateAndSplitFees(context, effectiveBaseAmount);
if (amountWithoutFee == 0 && fs.directFeeAmount == 0) ZeroAmount.selector.revertWith(groupId);
uint256 totalBaseRequired = amountWithoutFee + fs.directFeeAmount;
uint256 requiredPaymentWithFee = _baseToPayment(totalBaseRequired, paymentTokenDecimals, paymentTokenPrice, yieldBearingTokenPrice);
// 5. Handle payment transfer/verification
uint256 excessNativeAmount = 0;
if (!paymentToken.isNative()) {
paymentToken.safeTransferFrom(context.getRecipient(), address(this), requiredPaymentWithFee);
} else {
if (msg.value < requiredPaymentWithFee) {
InsufficientPreparedAmount.selector.revertWith(groupId);
}
// Calculate excess native amount
excessNativeAmount = msg.value - requiredPaymentWithFee;
}
// 6. Swap payment to yieldBearingToken (sAVAX)
uint256 yieldBearingTokenAmount = _handleTokenSwap(
context,
requiredPaymentWithFee,
totalBaseRequired,
paymentTokenPrice,
yieldBearingTokenPrice,
paymentTokenDecimals
);
// New: Swap excess native amount to yieldBearingToken for refund
uint256 excessYieldBearingTokenAmount = 0;
if (excessNativeAmount > 1e2) {
excessYieldBearingTokenAmount = _handleExcessNativeTokenSwap(
context,
excessNativeAmount,
paymentTokenPrice,
yieldBearingTokenPrice,
paymentTokenDecimals
);
}
// Check if we got significantly more than needed
if (yieldBearingTokenAmount > (totalBaseRequired * (HUNDRED + 1)) / HUNDRED) {
// 1% threshold
ExcessiveSwapAmount.selector.revertWith(groupId);
}
// 7. Take protocol fees from received yieldBearingToken
if (fs.protocolFeeAmount > 0) {
yieldBearingToken.safeTransfer(context.getFeeCollector().toAddress(), fs.protocolFeeAmount);
}
// Emit FeeCollected event after fees are distributed
if (fs.protocolFeeAmount > 0 || fs.hookFeeAmount > 0) {
emit FeeCollected(context.getFeeCollector().toAddress(), fs.protocolFeeAmount, fs.hookFeeAmount);
}
// 8. Wrap remaining yieldBearingToken to baseToken
yieldBearingTokenAmount = yieldBearingTokenAmount * (10 ** paymentTokenDecimals) / PRECISION;
fs.directFeeAmount = fs.directFeeAmount * (10 ** paymentTokenDecimals) / PRECISION;
uint256 remainingToWrap = yieldBearingTokenAmount - fs.directFeeAmount;
uint256 finalAmount = _wrapTokens(context, remainingToWrap);
if (finalAmount == 0) revert InsufficientPreparedAmount(groupId);
// 9. Handle excess yieldBearingToken refund
uint256 remainingYieldBearing = yieldBearingTokenAmount - fs.protocolFeeAmount - remainingToWrap;
uint256 totalExcessYieldBearing = remainingYieldBearing + excessYieldBearingTokenAmount;
if (totalExcessYieldBearing > 1e2) {
uint256 safeExcess = (totalExcessYieldBearing * REFUND_SAFETY_MARGIN) / BASIS_POINTS;
yieldBearingToken.safeTransfer(context.getRecipient(), safeExcess);
emit Refund(context.getRecipient(), safeExcess);
}
preparedToken = baseToken;
finalBaseAmount = finalAmount;
feeSplit = fs;
return (preparedToken, finalBaseAmount, feeSplit);
}
/**
* @notice Calculates and splits the fees according to the configured fee rates.
* @dev Uses high-precision arithmetic to minimize rounding errors.
* @param context The operation context.
* @param boundedAmount The amount after applying treasury bounds.
* @return amountWithoutFee The amount after subtracting total fees.
* @return feeSplit The detailed fee split.
*/
function _calculateAndSplitFees(
DOperationContext.OperationContext memory context,
uint256 boundedAmount
) private pure returns (uint256 amountWithoutFee, FeeSplit memory feeSplit) {
uint24 contextFee = context.getContextFee();
uint24 overriddenFee = context.getOverriddenFee();
// First check for zero fee flag
if ((overriddenFee & GroupFeeLibrary.ZERO_FEE_FLAG) != 0) {
return (boundedAmount, FeeSplit({ directFeeAmount: 0, hookFeeAmount: 0, protocolFeeAmount: 0, appliedFeeRate: 0 }));
}
// Get clean fees without flags
uint24 cleanContextFee = contextFee & GroupFeeLibrary.REMOVE_ALL_FLAGS_MASK;
uint24 cleanOverriddenFee = overriddenFee & GroupFeeLibrary.REMOVE_ALL_FLAGS_MASK;
// Determine which fee to apply based on context flags
uint24 appliedFeeRate;
bool hasHookShare = (contextFee & GroupFeeLibrary.DELEGATE_FEE_FLAG) != 0;
bool isDynamic = (contextFee & GroupFeeLibrary.DYNAMIC_FEE_FLAG) != 0;
if (cleanOverriddenFee > 0 && (isDynamic || hasHookShare)) {
appliedFeeRate = cleanOverriddenFee;
} else {
appliedFeeRate = cleanContextFee;
}
// Calculate total fee amount
uint256 totalFeeAmount = FullMath.mulDiv(boundedAmount, appliedFeeRate, BASIS_POINTS);
// Split fees based on context flags
if (hasHookShare) {
uint24 protocolFeeRate = context.getProtocolFee();
uint256 protocolShare = protocolFeeRate > 0 ? FullMath.mulDiv(totalFeeAmount, protocolFeeRate, BASIS_POINTS) : 0;
feeSplit.protocolFeeAmount = protocolShare;
feeSplit.directFeeAmount = totalFeeAmount;
feeSplit.hookFeeAmount = totalFeeAmount - protocolShare;
} else {
feeSplit.directFeeAmount = totalFeeAmount;
feeSplit.protocolFeeAmount = totalFeeAmount;
feeSplit.hookFeeAmount = 0;
}
amountWithoutFee = boundedAmount - totalFeeAmount;
feeSplit.appliedFeeRate = appliedFeeRate;
return (amountWithoutFee, feeSplit);
}
/**
* @notice Mints VT or YT tokens after we have prepared the base tokens.
* @dev This function handles the final step of minting after all preparations are complete.
* @param context The operation context
* @param underlyingValue The underlying base token amount to use for minting
* @param recipient Recipient of the minted tokens
*/
function mintTokens(
DOperationContext.OperationContext memory context,
uint256 underlyingValue,
address recipient
) internal returns (DProtocol.MintResult memory result) {
ITreasury treasury = ITreasury(context.getTreasury().toAddress());
IRebalancePool rebalancePool = IRebalancePool(context.getRebalancePool().toAddress());
if (address(treasury) == address(0) || address(rebalancePool) == address(0)) {
revert ZeroAddress(context.groupId);
}
StabilityFeeOperations.StabilityState memory stabilityState = StabilityFeeOperations.getStabilityState(context, underlyingValue);
uint8 operationType = context.getOperationType();
Currency baseToken = context.getBaseToken();
uint256 currentAllowance = baseToken.allowance(address(this), address(treasury));
if (currentAllowance < stabilityState.boundedAmount) {
baseToken.safeIncreaseAllowance(address(treasury), stabilityState.boundedAmount - currentAllowance);
}
if (operationType == uint8(OperationTypes.OP_TYPE_MINT_VT)) {
result.vtMinted = _handleVTMinting(context, stabilityState.boundedAmount, recipient, treasury);
} else if (operationType == uint8(OperationTypes.OP_TYPE_MINT_YT)) {
result.ytMinted = _handleYTMinting(context, stabilityState.boundedAmount, recipient, treasury, rebalancePool);
} else {
InvalidOperationType.selector.revertWith(context.groupId);
}
context.setAmount1(result.vtMinted > 0 ? result.vtMinted : result.ytMinted);
return result;
}
/**
* @notice Handles the minting of VT tokens.
* @param context The operation context.
* @param amount The amount of base tokens to mint.
* @param recipient The recipient of the minted VT tokens.
* @param treasury The treasury contract.
* @return mintedAmount The amount of VT tokens minted.
*/
function _handleVTMinting(
DOperationContext.OperationContext memory context,
uint256 amount,
address recipient,
ITreasury treasury
) private returns (uint256 mintedAmount) {
mintedAmount = treasury.mintXToken(context.groupId, amount, recipient);
if (mintedAmount == 0) {
revert InvalidMinimumAmount(context.groupId, amount, mintedAmount);
}
return mintedAmount;
}
/**
* @notice Mints YT tokens by depositing `aTokens` into the `RebalancePool`.
* @param context The operation context.
* @param amount The amount of base tokens to mint.
* @param recipient The recipient of the minted YT tokens.
* @param treasury The treasury contract.
* @param rebalancePool The rebalance pool contract.
* @return mintedAmount The amount of YT tokens minted.
*/
function _handleYTMinting(
DOperationContext.OperationContext memory context,
uint256 amount,
address recipient,
ITreasury treasury,
IRebalancePool rebalancePool
) private returns (uint256 mintedAmount) {
uint256 aTokensMinted = treasury.mintAToken(context.groupId, amount, address(this));
if (aTokensMinted == 0) {
revert InvalidMinimumAmount(context.groupId, amount, aTokensMinted);
}
Currency aToken = context.getAToken();
uint256 currentAllowance = aToken.allowance(address(this), address(rebalancePool));
if (currentAllowance < aTokensMinted) {
aToken.safeIncreaseAllowance(address(rebalancePool), aTokensMinted - currentAllowance);
}
mintedAmount = rebalancePool.deposit(aTokensMinted, recipient);
// Emit AutoCompounded event after depositing into the RebalancePool
emit AutoCompounded(aTokensMinted, mintedAmount, recipient);
return mintedAmount;
}
/**
* @notice Redeems VT or YT tokens, applying stability checks, fees, and unwrapping.
* @dev The sequence here is:
* 1. Redeem tokens from treasury/pool
* 2. Unwrap to yieldBearingToken if needed
* 3. Calculate and take fees
*/
function redeemTokens(
DOperationContext.OperationContext memory context,
uint256 receivedAmount,
address from
) internal returns (Currency preparedToken, uint256 preparedAmount, FeeSplit memory feeSplit) {
Currency baseToken = context.getBaseToken();
if (baseToken.isZero()) {
ZeroAddress.selector.revertWith(context.groupId);
}
StabilityFeeOperations.StabilityState memory stabilityState = StabilityFeeOperations.getStabilityState(context, receivedAmount);
// First handle the redemption
uint256 redeemedAmount = _handleTokenRedemption(context, stabilityState.boundedAmount, from);
// Then unwrap the redeemed amount
uint256 unwrappedAmount = _unwrapTokens(context, redeemedAmount);
// Calculate fees on the unwrapped amount
(preparedAmount, feeSplit) = _calculateAndSplitFees(context, unwrappedAmount);
// Double check that we have enough balance after fees
uint256 actualBalance = context.getYieldBearingToken().balanceOf(address(this));
if (actualBalance < feeSplit.protocolFeeAmount + preparedAmount) {
InsufficientPreparedAmount.selector.revertWith(context.groupId);
}
// Take protocol fees
if (feeSplit.protocolFeeAmount > 0) {
context.getYieldBearingToken().safeTransfer(context.getFeeCollector().toAddress(), feeSplit.protocolFeeAmount);
}
preparedToken = context.getYieldBearingToken();
return (preparedToken, preparedAmount, feeSplit);
}
function _handleTokenRedemption(
DOperationContext.OperationContext memory context,
uint256 amount,
address from
) private returns (uint256 redeemedAmount) {
uint8 operationType = context.getOperationType();
ITreasury treasury = ITreasury(context.getTreasury().toAddress());
if (operationType == uint8(OperationTypes.OP_TYPE_REDEEM_VT)) {
redeemedAmount = _handleVTRedemption(context, amount, from, treasury);
} else if (operationType == uint8(OperationTypes.OP_TYPE_REDEEM_YT)) {
redeemedAmount = _handleYTRedemption(context, amount, from);
} else {
InvalidOperationType.selector.revertWith(context.groupId);
}
return redeemedAmount;
}
function _handleVTRedemption(
DOperationContext.OperationContext memory context,
uint256 amount,
address from,
ITreasury treasury
) private returns (uint256 redeemedAmount) {
Currency vtToken = context.getXToken();
if (vtToken.isZero()) revert ZeroAddress(context.groupId);
vtToken.safeTransferFrom(from, address(this), amount);
uint256 currentAllowance = vtToken.allowance(address(this), address(treasury));
if (currentAllowance < amount) {
vtToken.safeIncreaseAllowance(address(treasury), amount - currentAllowance);
}
redeemedAmount = treasury.redeem(context.groupId, 0, amount, address(this));
return redeemedAmount;
}
function _handleYTRedemption(
DOperationContext.OperationContext memory context,
uint256 amount,
address from
) private returns (uint256 redeemedAmount) {
IRebalancePool rebalancePool = IRebalancePool(context.getRebalancePool().toAddress());
if (address(rebalancePool) == address(0)) revert ZeroAddress(context.groupId);
uint256 aTokensRedeemed = rebalancePool.redeem(amount, address(this), from);
if (aTokensRedeemed == 0) {
InsufficientPreparedAmount.selector.revertWith(context.groupId);
}
ITreasury treasury = ITreasury(context.getTreasury().toAddress());
Currency aToken = context.getAToken();
uint256 currentAllowance = aToken.allowance(address(this), address(treasury));
if (currentAllowance < aTokensRedeemed) {
aToken.safeIncreaseAllowance(address(treasury), aTokensRedeemed - currentAllowance);
}
redeemedAmount = treasury.redeem(context.groupId, aTokensRedeemed, 0, address(this));
return redeemedAmount;
}
function _wrapTokens(DOperationContext.OperationContext memory context, uint256 amountToWrap) private returns (uint256 wrappedAmount) {
Currency baseToken = context.getBaseToken();
Currency yieldBearingToken = context.getYieldBearingToken();
IWToken wToken = IWToken(baseToken.toAddress());
uint256 currentAllowance = yieldBearingToken.allowance(address(this), address(wToken));
if (currentAllowance < amountToWrap) {
yieldBearingToken.safeIncreaseAllowance(address(wToken), amountToWrap - currentAllowance);
}
try wToken.wrap(amountToWrap) returns (uint256 actualWrappedAmount) {
wrappedAmount = actualWrappedAmount;
} catch {
SwapOpFailed.selector.revertWith(context.groupId);
}
if (wrappedAmount == 0) {
WrappedAmountIsZero.selector.revertWith(context.groupId);
}
return wrappedAmount;
}
function _unwrapTokens(
DOperationContext.OperationContext memory context,
uint256 amountToUnwrap
) private returns (uint256 unwrappedAmount) {
if (!context.isWrappingRequired()) {
return amountToUnwrap;
}
Currency baseToken = context.getBaseToken();
IWToken wToken = IWToken(baseToken.toAddress());
try wToken.unwrap(amountToUnwrap) returns (uint256 actualUnwrappedAmount) {
unwrappedAmount = actualUnwrappedAmount;
} catch {
SwapOpFailed.selector.revertWith(context.groupId);
}
if (unwrappedAmount == 0) {
revert UnwrappedAmountIsZero(context.groupId);
}
return unwrappedAmount;
}
/**
* @notice Convert base token amount (yieldBearing scale) to payment token amount using prices.
*/
function _baseToPayment(
uint256 baseAmount, // in yieldBearing scale (1e18)
uint256 paymentTokenDecimals,
uint256 paymentTokenPrice,
uint256 yieldBearingTokenPrice
) private pure returns (uint256) {
uint256 rawPayment = (baseAmount * yieldBearingTokenPrice) / paymentTokenPrice;
uint256 factor = 10 ** paymentTokenDecimals;
return (rawPayment * factor) / PRECISION;
}
/**
* @notice Retrieves token prices from oracle.
*/
function _getPrices(
DOperationContext.OperationContext memory context
) private view returns (uint256 paymentTokenPrice, uint256 yieldBearingTokenPrice) {
IPriceOracle priceOracle = IPriceOracle(context.getPriceOracle().toAddress());
Currency paymentToken = context.getPaymentToken();
Currency yieldBearingToken = context.getYieldBearingToken();
bool validPaymentToken;
bool validYieldBearingToken;
(validPaymentToken, paymentTokenPrice) = priceOracle.getPrice(paymentToken.toAddress());
(validYieldBearingToken, yieldBearingTokenPrice) = priceOracle.getPrice(yieldBearingToken.toAddress());
if (paymentTokenPrice == 0 || yieldBearingTokenPrice == 0 || !validPaymentToken || !validYieldBearingToken) {
InvalidRate.selector.revertWith(context.groupId);
}
}
/**
* @notice Performs token swap from paymentToken to yieldBearingToken.
* @param context The operation context
* @param paymentIn Payment token amount we have after calculations
* @param totalBaseRequired The total yieldBearingToken needed
* @return yieldBearingTokenAmount The yieldBearingToken obtained after swap
*/
function _handleTokenSwap(
DOperationContext.OperationContext memory context,
uint256 paymentIn,
uint256 totalBaseRequired,
uint256 paymentTokenPrice,
uint256 yieldBearingTokenPrice,
uint256 paymentTokenDecimals
) private returns (uint256 yieldBearingTokenAmount) {
Currency paymentToken = context.getPaymentToken();
Currency yieldBearingToken = context.getYieldBearingToken();
if (paymentToken.equals(yieldBearingToken)) {
// No swap needed
return totalBaseRequired;
}
// Slippage check
uint256 slippageRate = uint256(context.getSlippageRate());
if (slippageRate > MAX_SLIPPAGE_TOLERANCE) {
revert InvalidSlippageTolerance(context.groupId, MAX_SLIPPAGE_TOLERANCE, slippageRate);
}
// Calculate expected output and minimum acceptable amount
uint256 adjustedPaymentIn = (paymentIn * PRECISION) / (10 ** paymentTokenDecimals);
uint256 expectedOut = (adjustedPaymentIn * paymentTokenPrice) / yieldBearingTokenPrice;
uint256 minOut = (expectedOut * (BASIS_POINTS - slippageRate)) / BASIS_POINTS;
// Ensure minOut doesn't exceed what we actually need
if (minOut > totalBaseRequired) {
minOut = totalBaseRequired;
}
// Perform swap
yieldBearingTokenAmount = swapTokens(context, paymentToken, yieldBearingToken, paymentIn, minOut, paymentToken.isNative());
return yieldBearingTokenAmount;
}
function _handleExcessNativeTokenSwap(
DOperationContext.OperationContext memory context,
uint256 excessNativeAmount,
uint256 paymentTokenPrice,
uint256 yieldBearingTokenPrice,
uint256 paymentTokenDecimals
) private returns (uint256 yieldBearingTokenAmount) {
Currency paymentToken = context.getPaymentToken();
Currency yieldBearingToken = context.getYieldBearingToken();
// Calculate expected output and minimum acceptable amount for the excess
uint256 adjustedPaymentIn = (excessNativeAmount * PRECISION) / (10 ** paymentTokenDecimals);
uint256 expectedOut = (adjustedPaymentIn * paymentTokenPrice) / yieldBearingTokenPrice;
uint256 minOut = (expectedOut * (BASIS_POINTS - MAX_SLIPPAGE_TOLERANCE)) / BASIS_POINTS;
// Perform swap for excess native token to yieldBearingToken
yieldBearingTokenAmount = swapTokens(context, paymentToken, yieldBearingToken, excessNativeAmount, minOut, paymentToken.isNative());
return yieldBearingTokenAmount;
}
/**
* @notice Executes token swaps through DEX router with optimal path selection.
* @dev Handles native token wrapping/unwrapping automatically.
*/
function swapTokens(
DOperationContext.OperationContext memory context,
Currency fromToken,
Currency toToken,
uint256 amount,
uint256 minAmountOut,
bool isNative
) private returns (uint256 swappedAmount) {
address swapRouter = context.getSwapRouter().toAddress();
if (swapRouter == address(0)) InvalidSwapRouter.selector.revertWith(context.groupId);
IDEXRouter router = IDEXRouter(swapRouter);
uint256 deadline = block.timestamp + 5;
address wethAddress = context.getGroupWethAddress();
address[] memory path;
if (fromToken.isNative()) {
path = new address[](2);
path[0] = wethAddress;
path[1] = toToken.toAddress();
require(toToken.equals(context.getYieldBearingToken()), "Only native to yieldBearingToken swaps supported");
uint256[] memory amounts = router.swapExactETHForTokens{ value: amount }(minAmountOut, path, address(this), deadline);
swappedAmount = amounts[amounts.length - 1];
} else if (isNative) {
path = new address[](2);
path[0] = fromToken.toAddress();
path[1] = wethAddress;
_approveRouter(fromToken, swapRouter, amount);
uint256[] memory amounts = router.swapExactTokensForETH(amount, minAmountOut, path, address(this), deadline);
swappedAmount = amounts[amounts.length - 1];
} else {
if (fromToken.toAddress() == wethAddress || toToken.toAddress() == wethAddress) {
path = new address[](2);
path[0] = fromToken.toAddress();
path[1] = toToken.toAddress();
} else {
path = new address[](3);
path[0] = fromToken.toAddress();
path[1] = wethAddress;
path[2] = toToken.toAddress();
}
_approveRouter(fromToken, swapRouter, amount);
uint256[] memory amounts = router.swapExactTokensForTokens(amount, minAmountOut, path, address(this), deadline);
swappedAmount = amounts[amounts.length - 1];
}
if (swappedAmount < minAmountOut) {
revert InvalidMinimumAmount(context.groupId, minAmountOut, swappedAmount);
}
// Emit TokensSwapped event after a successful swap
emit TokensSwapped(fromToken.toAddress(), toToken.toAddress(), amount, swappedAmount);
return swappedAmount;
}
function _approveRouter(Currency tokenContract, address router, uint256 amount) private {
uint256 currentAllowance = tokenContract.allowance(address(this), router);
if (currentAllowance < amount) {
tokenContract.safeIncreaseAllowance(router, amount - currentAllowance);
}
}
/**
* @notice Converts and sends tokens to the desired collateral type if needed.
* @dev Important: Base token cannot be used as collateral.
*/
function convertToDesiredCollateralAndSend(
DOperationContext.OperationContext memory context,
uint256 baseAmountWithoutFee,
address recipient
) internal returns (uint256 swappedAmount) {
if (baseAmountWithoutFee == 0) {
UnwrappedAmountIsZero.selector.revertWith(context.groupId);
}
Currency baseToken = context.getBaseToken();
Currency yieldBearingToken = context.getYieldBearingToken();
Currency desiredCollateral = context.getPaymentToken();
if (desiredCollateral.equals(baseToken)) {
BaseTokenCannotBeUsedAsCollateral.selector.revertWith(context.groupId);
}
// If user wants yieldBearingToken back, just transfer it
if (desiredCollateral.equals(yieldBearingToken)) {
yieldBearingToken.safeTransfer(recipient, baseAmountWithoutFee);
return baseAmountWithoutFee;
}
bool isNative = desiredCollateral.isNative();
uint256 slippageRate = uint256(context.getSlippageRate());
if (slippageRate > MAX_SLIPPAGE_TOLERANCE) {
revert InvalidSlippageTolerance(context.groupId, MAX_SLIPPAGE_TOLERANCE, slippageRate);
}
// Get prices for conversion
IPriceOracle priceOracle = IPriceOracle(context.getPriceOracle().toAddress());
(bool isValidYieldBearingTokenPrice, uint256 yieldBearingTokenPrice) = priceOracle.getPrice(yieldBearingToken.toAddress());
(bool isValidCollateralPrice, uint256 desiredCollateralPrice) = priceOracle.getPrice(desiredCollateral.toAddress());
if (!isValidYieldBearingTokenPrice || !isValidCollateralPrice || yieldBearingTokenPrice == 0 || desiredCollateralPrice == 0) {
InvalidRate.selector.revertWith(context.groupId);
}
uint8 baseTokenDecimals = context.getBaseTokenDecimals();
uint256 desiredCollateralDecimals = uint256(context.getPaymentTokenDecimals());
// ---------------------------------------------------------------------
// 1) Normalize baseAmountWithoutFee from baseTokenDecimals up to 1e18
// ---------------------------------------------------------------------
// If baseTokenDecimals == 18, this is effectively just `baseAmountWithoutFee`.
// Otherwise we scale it so the math against yieldBearingTokenPrice and
// desiredCollateralPrice (both typically at 1e18) stays consistent.
// ---------------------------------------------------------------------
uint256 normalizedBaseAmount = (baseAmountWithoutFee * PRECISION) / (10 ** baseTokenDecimals);
// ---------------------------------------------------------------------
// 2) Calculate expectedOutAmount with prices at 1e18
// ---------------------------------------------------------------------
// Now normalizedBaseAmount is in 1e18. Multiplying by yieldBearingTokenPrice
// and dividing by desiredCollateralPrice keeps the math in 1e18 as well.
// ---------------------------------------------------------------------
uint256 expectedOutAmount = (normalizedBaseAmount * yieldBearingTokenPrice) / desiredCollateralPrice;
// ---------------------------------------------------------------------
// 3) Apply slippage to get minOutAmount still in 1e18
// ---------------------------------------------------------------------
uint256 minOutAmount = (expectedOutAmount * (BASIS_POINTS - slippageRate)) / BASIS_POINTS;
// ---------------------------------------------------------------------
// 4) Convert minOutAmount from 1e18 down to desiredCollateralDecimals
// ---------------------------------------------------------------------
uint256 minAmountOutInDesiredDecimals = (minOutAmount * (10 ** desiredCollateralDecimals)) / PRECISION;
swappedAmount = swapTokens(
context,
yieldBearingToken,
desiredCollateral,
baseAmountWithoutFee,
minAmountOutInDesiredDecimals,
isNative
);
if (swappedAmount == 0) {
TransferFailed.selector.revertWith(context.groupId, recipient, swappedAmount);
}
// If paying out a native token (e.g. ETH), we have to forward it
// and adjust swappedAmount in yieldBearingToken terms for accounting.
if (isNative) {
uint256 yieldBearingEquivalent = (((swappedAmount * PRECISION) / (10 ** desiredCollateralDecimals)) * desiredCollateralPrice) /
yieldBearingTokenPrice;
(bool success, ) = recipient.call{ value: swappedAmount }("");
require(success, "ETH transfer failed");
swappedAmount = yieldBearingEquivalent; // Return the yieldBearingToken equivalent
} else {
// For standard ERC20 tokens, just transfer the swapped tokens
desiredCollateral.safeTransfer(recipient, swappedAmount);
}
context.setAmount1(swappedAmount);
return swappedAmount;
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { DOperationContext } from "../declarations/DOperationContext.sol";
import { Slot0, Slot0Library } from "../types/Slot0.sol";
import { Currency, CurrencyLibrary } from "../types/Currency.sol";
import { GroupId } from "../types/GroupId.sol";
import { CustomRevert } from "./CustomRevert.sol";
// import { DTokenRegistry } from "../declarations/DTokenRegistry.sol";
import { Address, AddressLibrary } from "../types/Address.sol";
import { GroupState, FeePermissions, DefaultFeeParams, OperationTypes } from "../types/CommonTypes.sol";
import { GroupStateHelper, GroupSettings } from "../types/GroupStateHelper.sol";
import { GroupFeeLibrary } from "../libs/GroupFeeLibrary.sol";
import { IRateProvider } from "../interfaces/IRateProvider.sol";
import { ITreasury } from "../interfaces/ITreasury.sol";
/**
* @title OperationContextLibrary
* @notice Library for creating and managing operation contexts within the protocol.
*/
library OperationContextLibrary {
using Slot0Library for Slot0;
using GroupStateHelper for GroupState;
using GroupStateHelper for GroupSettings;
using AddressLibrary for Address;
using CurrencyLibrary for Currency;
using CustomRevert for bytes4;
// Custom Errors
error ZeroAddress(GroupId groupId);
error ZeroAmount(GroupId groupId);
error CycleOperationsAreNotPermitted(GroupId groupId);
error InvalidCollateralRatio(GroupId groupId);
error ErrorMintingPausedInStabilityMode(GroupId groupId);
error InvalidOperationType(GroupId groupId);
error ErrorGettingRate(GroupId groupId, uint256 rate);
error InvalidSlippageTolerance(GroupId groupId, uint256 minAmount1, uint256 amount0);
error ArbitraryCallsNotAllowed(GroupId groupId, address recipient, address sender);
// Constants
uint256 private constant PRECISION = 1e18;
uint24 private constant MAX_SLIPPAGE_TOLERANCE = 1_000; // 10% bps
uint24 private constant BASIS_POINTS = 10_000;
/**
* @notice Creates an operation context.
* @param params Struct containing parameters needed to create the context.
* @return context The created operation context.
*/
function createContext(
DOperationContext.Context memory params
) internal view returns (DOperationContext.OperationContext memory context) {
// Validate recipient and amounts
_validateParams(params);
// Initialize context without applying slippage checks
context = _initializeContext(params);
// Retrieve operation type
uint8 operationType = params.operationType;
// Get stability ratios
uint96 stabilityRatio = _getStabilityRatio(params);
uint96 stabilityConditionsTriggeringRate = _getstabilityConditionsTriggeringRate(params);
// Retrieve collateral ratio and validate
context.collateralRatio = _getCollateralRatio(params, stabilityRatio);
// Calculate context fee & override fee depending on the group's permissions
uint24 contextFee = _calculateContextFee(params, operationType, context.collateralRatio, stabilityConditionsTriggeringRate);
// Get initial protocol fee
uint24 initialProtocolFee = GroupStateHelper.getProtocolFee(params.groupState.feesPacked);
// Pack Slot0 and Slot1
context.params = _createParams(params, operationType, initialProtocolFee, contextFee, stabilityConditionsTriggeringRate);
}
// ------ Helper Functions ------ //
/**
* @notice Validates the parameters for creating an operation context.
* @param params Struct containing parameters needed to create the context.
*/
function _validateParams(DOperationContext.Context memory params) internal pure {
if (params.recipient == address(0)) {
ZeroAddress.selector.revertWith(params.groupId);
}
if (params.amounts.amount0 == 0 && params.amounts.amount1 == 0) {
ZeroAmount.selector.revertWith(params.groupId);
}
if (params.slippage > MAX_SLIPPAGE_TOLERANCE) {
revert InvalidSlippageTolerance(params.groupId, params.slippage, MAX_SLIPPAGE_TOLERANCE);
}
}
/**
* @notice Initializes the operation context without applying slippage checks.
* @param params Struct containing parameters needed to create the context.
* @return context The initialized operation context.
*/
function _initializeContext(
DOperationContext.Context memory params
) internal pure returns (DOperationContext.OperationContext memory context) {
context.groupId = params.groupId;
context.amount0 = params.amounts.amount0;
context.amount1 = params.amounts.amount1;
context.recipient = params.recipient;
context.paymentToken = params.paymentToken;
context.slippage = _validateSlippage(params.groupId, params.slippage);
context.groupState = DOperationContext.TightGroupState({
core: params.groupState.core,
extended: params.groupState.extended,
groupSettings: params.groupState.groupSettings,
feesPacked: params.groupState.feesPacked,
hookContract: params.groupState.hookContract
});
}
/**
* @notice Validates the slippage value.
* @param groupId The group ID.
* @param slippage The slippage value to validate.
* @return The validated slippage value.
*/
function _validateSlippage(GroupId groupId, uint24 slippage) internal pure returns (uint24) {
if (slippage > MAX_SLIPPAGE_TOLERANCE) {
revert InvalidSlippageTolerance(groupId, slippage, MAX_SLIPPAGE_TOLERANCE);
}
return slippage;
}
/**
* @notice Retrieves the collateral ratio and validates it.
* @param params Struct containing parameters needed to create the context.
* @param stabilityRatio The stability ratio.
* @return collateralRatio The validated collateral ratio.
*/
function _getCollateralRatio(
DOperationContext.Context memory params,
uint96 stabilityRatio
) internal view returns (uint256 collateralRatio) {
ITreasury treasury = ITreasury(params.groupState.extended.treasury.toAddress());
try treasury.collateralRatio(params.groupId) returns (uint256 _collateralRatio) {
collateralRatio = _collateralRatio;
if (collateralRatio <= uint256(stabilityRatio)) {
ErrorMintingPausedInStabilityMode.selector.revertWith(params.groupId);
}
} catch {
InvalidCollateralRatio.selector.revertWith(params.groupId);
}
}
/**
* @notice Calculates the context fee.
* @param params Struct containing parameters needed to create the context.
* @param operationType The operation type.
* @param collateralRatio The collateral ratio.
* @param stabilityConditionsTriggeringRate The stability conditions triggering rate.
* @return contextFee The calculated context fee.
*/
function _calculateContextFee(
DOperationContext.Context memory params,
uint8 operationType,
uint256 collateralRatio,
uint96 stabilityConditionsTriggeringRate
) internal pure returns (uint24 contextFee) {
bool isStabilityMode = collateralRatio < uint256(stabilityConditionsTriggeringRate);
if (isStabilityMode) {
contextFee = _getStabilityFee(params.groupId, params.groupState.feesPacked, operationType);
} else {
contextFee = _getRegularFee(params.groupId, params.groupState.feesPacked, operationType);
}
contextFee = _applyFeePermissions(contextFee, params.groupState.groupSettings);
return contextFee;
}
/**
* @notice Retrieves the stability fee.
* @param groupId The group ID.
* @param feesPacked The packed fees.
* @param operationType The operation type.
* @return The stability fee.
*/
function _getStabilityFee(GroupId groupId, bytes32 feesPacked, uint8 operationType) internal pure returns (uint24) {
if (operationType == uint8(OperationTypes.OP_TYPE_MINT_VT)) {
return GroupStateHelper.getStabilityMintFeeVT(feesPacked);
} else if (operationType == uint8(OperationTypes.OP_TYPE_MINT_YT)) {
return GroupStateHelper.getStabilityMintFeeYT(feesPacked);
} else if (operationType == uint8(OperationTypes.OP_TYPE_REDEEM_VT)) {
return GroupStateHelper.getStabilityRedeemFeeVT(feesPacked);
} else if (operationType == uint8(OperationTypes.OP_TYPE_REDEEM_YT)) {
return GroupStateHelper.getStabilityRedeemFeeYT(feesPacked);
} else {
InvalidOperationType.selector.revertWith(groupId);
}
}
/**
* @notice Retrieves the regular fee.
* @param groupId The group ID.
* @param feesPacked The packed fees.
* @param operationType The operation type.
* @return The regular fee.
*/
function _getRegularFee(GroupId groupId, bytes32 feesPacked, uint8 operationType) internal pure returns (uint24) {
if (operationType == uint8(OperationTypes.OP_TYPE_MINT_VT)) {
return GroupStateHelper.getMintFeeVT(feesPacked);
} else if (operationType == uint8(OperationTypes.OP_TYPE_MINT_YT)) {
return GroupStateHelper.getMintFeeYT(feesPacked);
} else if (operationType == uint8(OperationTypes.OP_TYPE_REDEEM_VT)) {
return GroupStateHelper.getRedeemFeeVT(feesPacked);
} else if (operationType == uint8(OperationTypes.OP_TYPE_REDEEM_YT)) {
return GroupStateHelper.getRedeemFeeYT(feesPacked);
} else {
InvalidOperationType.selector.revertWith(groupId);
}
}
/**
* @notice Applies fee permissions to the context fee.
* @param fee The context fee.
* @param groupSettings The group settings.
* @return The context fee with applied permissions.
*/
function _applyFeePermissions(uint24 fee, bytes32 groupSettings) internal pure returns (uint24) {
FeePermissions memory feePermissions = GroupStateHelper.getFeePermissions(GroupSettings.wrap(groupSettings));
if (feePermissions.isDynamic && !feePermissions.allowDelegation) {
fee |= GroupFeeLibrary.DYNAMIC_FEE_FLAG;
}
if (feePermissions.allowDelegation) {
fee |= GroupFeeLibrary.DELEGATE_FEE_FLAG;
}
return fee;
}
/**
* @notice Creates the parameters for the operation context.
* @param params Struct containing parameters needed to create the context.
* @param operationType The operation type.
* @param initialProtocolFee The initial protocol fee.
* @param contextFee The context fee.
* @param stabilityConditionsTriggeringRate The stability conditions triggering rate.
* @return operationParams The created parameters.
*/
function _createParams(
DOperationContext.Context memory params,
uint8 operationType,
uint24 initialProtocolFee,
uint24 contextFee,
uint96 stabilityConditionsTriggeringRate
) internal pure returns (DOperationContext.Params memory operationParams) {
Slot0 slot0 = Slot0Library.pack(
operationType,
initialProtocolFee,
contextFee,
0, // overriddenFee
stabilityConditionsTriggeringRate
);
operationParams = DOperationContext.Params({ slot0: slot0, slot1: params.hookData });
}
/**
* @notice Retrieves the stability ratio.
* @param params Struct containing parameters needed to create the context.
* @return The stability ratio.
*/
function _getStabilityRatio(DOperationContext.Context memory params) internal pure returns (uint96) {
return GroupStateHelper.getStabilityRatio(GroupSettings.wrap(params.groupState.groupSettings));
}
/**
* @notice Retrieves the stability conditions triggering rate.
* @param params Struct containing parameters needed to create the context.
* @return The stability conditions triggering rate.
*/
function _getstabilityConditionsTriggeringRate(DOperationContext.Context memory params) internal pure returns (uint96) {
return GroupStateHelper.getStabilityTriggeringRatio(GroupSettings.wrap(params.groupState.groupSettings));
}
// ------ Getter Functions ------ //
/**
* @notice Retrieves the operation type from the operation context.
* @param self The operation context.
* @return The operation type.
*/
function getOperationType(DOperationContext.OperationContext memory self) internal pure returns (uint8) {
return self.params.slot0.operationType();
}
/**
* @notice Retrieves the first amount from the operation context.
* @param self The operation context.
* @return The first amount.
*/
function getAmount0(DOperationContext.OperationContext memory self) internal pure returns (uint256) {
return self.amount0;
}
/**
* @notice Retrieves the slippage rate from the operation context.
* @param self The operation context.
* @return The slippage rate.
*/
function getSlippageRate(DOperationContext.OperationContext memory self) internal pure returns (uint24) {
return self.slippage;
}
/**
* @notice Retrieves the context fee from the operation context.
* @param self The operation context.
* @return The context fee.
*/
function getContextFee(DOperationContext.OperationContext memory self) internal pure returns (uint24) {
return self.params.slot0.contextFee();
}
/**
* @notice Retrieves the protocol fee from the operation context.
* @param self The operation context.
* @return The protocol fee.
*/
function getProtocolFee(DOperationContext.OperationContext memory self) internal pure returns (uint24) {
return self.params.slot0.contextProtocolFee();
}
/**
* @notice Retrieves the overridden fee from the operation context.
* @param self The operation context.
* @return The overridden fee.
*/
function getOverriddenFee(DOperationContext.OperationContext memory self) internal pure returns (uint24) {
return self.params.slot0.overriddenFee();
}
/**
* @notice Retrieves the hook data from the operation context.
* @param self The operation context.
* @return The hook data.
*/
function getHookData(DOperationContext.OperationContext memory self) internal pure returns (bytes memory) {
return self.params.slot1;
}
/**
* @notice Retrieves the recipient from the operation context.
* @param self The operation context.
* @return The recipient.
*/
function getRecipient(DOperationContext.OperationContext memory self) internal view returns (address) {
address recipient = self.recipient;
if (recipient != msg.sender) revert ArbitraryCallsNotAllowed(self.groupId, recipient, msg.sender);
return self.recipient;
}
/**
* @notice Retrieves the payment token from the operation context.
* @param self The operation context.
* @return The payment token.
*/
function getPaymentToken(DOperationContext.OperationContext memory self) internal pure returns (Currency) {
return self.paymentToken.token;
}
/**
* @notice Retrieves the payment token decimals from the operation context.
* @param self The operation context.
* @return The payment token decimals.
*/
function getPaymentTokenDecimals(DOperationContext.OperationContext memory self) internal pure returns (uint8) {
return self.paymentToken.decimals;
}
/**
* @notice Retrieves the base token decimals from the operation context.
* @param self The operation context.
* @return The base token decimals.
*/
function getBaseTokenDecimals(DOperationContext.OperationContext memory self) internal pure returns (uint8) {
return GroupStateHelper.getBaseTokenDecimals(GroupSettings.wrap(self.groupState.groupSettings));
}
/**
* @notice Retrieves the hook contract from the operation context.
* @param self The operation context.
* @return The hook contract.
*/
function getHookContract(DOperationContext.OperationContext memory self) internal pure returns (address) {
return self.groupState.hookContract.toAddress();
}
function getGroupWethAddress(DOperationContext.OperationContext memory self) internal pure returns (address) {
return self.groupState.core.wethToken.toAddress();
}
/**
* @notice Retrieves the stability ratio from the operation context.
* @param self The operation context.
* @return The stability ratio.
*/
function getStabilityRatio(DOperationContext.OperationContext memory self) internal pure returns (uint96) {
return GroupStateHelper.getStabilityRatio(GroupSettings.wrap(self.groupState.groupSettings));
}
// ------ Setter Functions ------ //
/**
* @notice Sets the overridden fee in the operation context.
* @param self The operation context.
* @param overriddenFee The overridden fee.
*/
function setOverriddenFee(DOperationContext.OperationContext memory self, uint24 overriddenFee) internal pure {
FeePermissions memory feePermissions = GroupStateHelper.getFeePermissions(GroupSettings.wrap(self.groupState.groupSettings));
DefaultFeeParams memory defaultFees = GroupStateHelper.getDefaultFeeParams(self.groupState.feesPacked);
uint24 validatedFee = GroupFeeLibrary.processGroupFee(defaultFees, feePermissions, overriddenFee);
self.params.slot0 = self.params.slot0.setOverriddenFee(validatedFee);
}
/**
* @notice Sets the hook data in the operation context.
* @param self The operation context.
* @param hookData The hook data.
*/
function setHookData(DOperationContext.OperationContext memory self, bytes memory hookData) internal pure {
/// @dev Only set hookData if hookContract exists
if (!self.groupState.hookContract.isZero()) {
self.params.slot1 = hookData;
}
}
/**
* @notice Sets the context fee in the operation context.
* @param self The operation context.
* @param contextFee The context fee.
*/
function setContextFee(DOperationContext.OperationContext memory self, uint24 contextFee) internal pure {
self.params.slot0 = self.params.slot0.setContextFee(contextFee);
}
// ------ Utility Functions ------ //
/**
* @notice Validates the recipient in the operation context.
* @param self The operation context.
*/
function validateRecipient(DOperationContext.OperationContext memory self) internal pure {
if (self.recipient == address(0)) {
ZeroAddress.selector.revertWith(self.groupId);
}
/// @dev guarding for hooks to cycle minting/redeeming
if (self.recipient == self.groupState.hookContract.toAddress()) {
CycleOperationsAreNotPermitted.selector.revertWith(self.groupId);
}
}
/**
* @notice Retrieves the fee model from the operation context.
* @param self The operation context.
* @return The fee model.
*/
function getFeeModel(DOperationContext.OperationContext memory self) internal pure returns (uint8) {
return GroupStateHelper.getFeeModel(GroupSettings.wrap(self.groupState.groupSettings));
}
/**
* @notice Retrieves the context collateral ratio from the operation context.
* @param self The operation context.
* @return The context collateral ratio.
*/
function getContextCollateralRatio(DOperationContext.OperationContext memory self) internal pure returns (uint256) {
return self.collateralRatio;
}
/**
* @notice Checks if wrapping is required in the operation context.
* @param self The operation context.
* @return True if wrapping is required, false otherwise.
*/
function isWrappingRequired(DOperationContext.OperationContext memory self) internal pure returns (bool) {
return GroupStateHelper.isWrappingRequired(GroupSettings.wrap(self.groupState.groupSettings));
}
/**
* @notice Retrieves the hook permissions from the operation context.
* @param self The operation context.
* @return The hook permissions.
*/
function getHookPermissions(DOperationContext.OperationContext memory self) internal pure returns (uint256) {
return GroupStateHelper.getHookPermissions(GroupSettings.wrap(self.groupState.groupSettings));
}
// ------ Interaction with Slot0 and Slot1 ------ //
/**
* @notice Sets the second amount in the operation context.
* @param self The operation context.
* @param amount1 The second amount.
*/
function setAmount1(DOperationContext.OperationContext memory self, uint256 amount1) internal pure {
self.amount1 = amount1;
}
/**
* @notice Retrieves the AToken from the operation context.
* @param self The operation context.
* @return The AToken.
*/
function getAToken(DOperationContext.OperationContext memory self) internal pure returns (Currency) {
return self.groupState.core.aToken;
}
/**
* @notice Retrieves the XToken from the operation context.
* @param self The operation context.
* @return The XToken.
*/
function getXToken(DOperationContext.OperationContext memory self) internal pure returns (Currency) {
return self.groupState.core.xToken;
}
/**
* @notice Retrieves the base token from the operation context.
* @param self The operation context.
* @return The base token.
*/
function getBaseToken(DOperationContext.OperationContext memory self) internal pure returns (Currency) {
return self.groupState.core.baseToken;
}
/**
* @notice Retrieves the yield bearing token from the operation context.
* @param self The operation context.
* @return The yield bearing token.
*/
function getYieldBearingToken(DOperationContext.OperationContext memory self) internal pure returns (Currency) {
return self.groupState.core.yieldBearingToken;
}
/**
* @notice Retrieves the swap router from the operation context.
* @param self The operation context.
* @return The swap router.
*/
function getSwapRouter(DOperationContext.OperationContext memory self) internal pure returns (Address) {
return self.groupState.extended.swapRouter;
}
/**
* @notice Retrieves the price oracle from the operation context.
* @param self The operation context.
* @return The price oracle.
*/
function getPriceOracle(DOperationContext.OperationContext memory self) internal pure returns (Address) {
return self.groupState.extended.priceOracle;
}
/**
* @notice Retrieves the treasury from the operation context.
* @param self The operation context.
* @return The treasury.
*/
function getTreasury(DOperationContext.OperationContext memory self) internal pure returns (Address) {
return self.groupState.extended.treasury;
}
/**
* @notice Retrieves the fee collector from the operation context.
* @param self The operation context.
* @return The fee collector.
*/
function getFeeCollector(DOperationContext.OperationContext memory self) internal pure returns (Address) {
return self.groupState.extended.feeCollector;
}
/**
* @notice Retrieves the rebalance pool from the operation context.
* @param self The operation context.
* @return The rebalance pool.
*/
function getRebalancePool(DOperationContext.OperationContext memory self) internal pure returns (Currency) {
return self.groupState.extended.rebalancePool;
}
function isMintOperation(DOperationContext.OperationContext memory self) internal pure returns (bool) {
uint8 operationType = self.params.slot0.operationType();
return operationType == uint8(OperationTypes.OP_TYPE_MINT_VT) || operationType == uint8(OperationTypes.OP_TYPE_MINT_YT);
}
function isRedeemOperation(DOperationContext.OperationContext memory self) internal pure returns (bool) {
uint8 operationType = self.params.slot0.operationType();
return operationType == uint8(OperationTypes.OP_TYPE_REDEEM_VT) || operationType == uint8(OperationTypes.OP_TYPE_REDEEM_YT);
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { DOperationContext } from "../declarations/DOperationContext.sol";
import { OperationContextLibrary } from "./OperationContextLibrary.sol";
import { GroupId } from "../types/GroupId.sol";
import { IProtocol } from "../interfaces/IProtocol.sol";
import { CustomRevert } from "./CustomRevert.sol";
import { OperationTypes } from "../types/CommonTypes.sol";
/**
* @title TimeManagementLibrary
* @notice Library for managing time-based operations and restrictions
*/
library TimeManagementLibrary {
using OperationContextLibrary for DOperationContext.OperationContext;
using CustomRevert for bytes4;
/// @notice Early exit and time lock constants
uint256 private constant EARLY_EXIT_WINDOW = 0;
uint24 private constant EARLY_EXIT_FEE_BPS = 25; // 0.25%
struct TimeStorage {
/// @notice Track last operation times per user per group
mapping(GroupId => mapping(address => UserOperationTime)) userOperationTimes;
// /// @notice Track which fee models require time locks
// mapping(GroupId => bool) hasRedeemLock;
}
/// @notice Track last operation times per user per group
struct UserOperationTime {
uint256 lastVTMintTime; // Last VT mint time
uint256 lastYTMintTime; // Last YT mint time
uint256 lastVTRedeemTime; // Last VT redeem time
uint256 lastYTRedeemTime; // Last YT redeem time
}
/**
* @notice Checks if an early exit fee should be applied
* @param self The time storage
* @param context The operation context
* @param user The user address
* @return feeApplied Whether early exit fee should be applied
* @return additionalFeeBps The additional fee in basis points
*/
function checkEarlyExitFee(
TimeStorage storage self,
DOperationContext.OperationContext memory context,
address user
) internal view returns (bool feeApplied, uint24 additionalFeeBps) {
GroupId groupId = context.groupId;
UserOperationTime storage userTime = self.userOperationTimes[groupId][user];
uint8 operationType = context.getOperationType();
if (context.isRedeemOperation()) {
uint256 relevantMintTime = operationType == uint8(OperationTypes.OP_TYPE_REDEEM_VT) ? userTime.lastVTMintTime : userTime.lastYTMintTime;
if (relevantMintTime != 0 && block.timestamp - relevantMintTime < EARLY_EXIT_WINDOW) {
return (true, EARLY_EXIT_FEE_BPS);
}
}
return (false, 0);
}
/**
* @notice Updates the operation timestamp for a user
* @param self The time storage
* @param groupId The group ID
* @param operationType The operation type
* @param user The user address
* @param isMinting Whether the operation is minting
*/
function updateOperationTime(TimeStorage storage self, GroupId groupId, uint8 operationType, address user, bool isMinting) internal {
if (isMinting) {
if (operationType == uint8(OperationTypes.OP_TYPE_MINT_VT)) {
self.userOperationTimes[groupId][user].lastVTMintTime = block.timestamp;
} else if (operationType == uint8(OperationTypes.OP_TYPE_MINT_YT)) {
self.userOperationTimes[groupId][user].lastYTMintTime = block.timestamp;
}
} else {
if (operationType == uint8(OperationTypes.OP_TYPE_REDEEM_VT)) {
self.userOperationTimes[groupId][user].lastVTRedeemTime = block.timestamp;
} else if (operationType == uint8(OperationTypes.OP_TYPE_REDEEM_YT)) {
self.userOperationTimes[groupId][user].lastYTRedeemTime = block.timestamp;
}
}
}
// /**
// * @notice Validates time-based restrictions for a user
// * @param self The time storage
// * @param context The operation context
// * @param user The user address
// */
// function validateTimeRestrictions(
// TimeStorage storage self,
// DOperationContext.OperationContext memory context,
// address user
// ) internal view {
// GroupId groupId = context.groupId;
// if (!self.hasRedeemLock[groupId]) {
// return;
// }
// UserOperationTime storage userTime = self.userOperationTimes[groupId][user];
// uint8 operationType = context.getOperationType();
// uint256 relevantMintTime = operationType == uint8(OperationTypes.OP_TYPE_REDEEM_VT) ? userTime.lastVTMintTime : userTime.lastYTMintTime;
// if (relevantMintTime != 0 && block.timestamp - relevantMintTime < EARLY_EXIT_WINDOW) {
// IProtocol.RedeemLocked.selector.revertWith(groupId);
// }
// }
/**
* @notice Gets the operation times for a user
* @param self The time storage
* @param groupId The group ID
* @param user The user address
* @return lastVTMintTime The last VT mint time
* @return lastYTMintTime The last YT mint time
* @return lastVTRedeemTime The last VT redeem time
* @return lastYTRedeemTime The last YT redeem time
*/
function getUserOperationTime(
TimeStorage storage self,
GroupId groupId,
address user
) internal view returns (uint256 lastVTMintTime, uint256 lastYTMintTime, uint256 lastVTRedeemTime, uint256 lastYTRedeemTime) {
UserOperationTime storage userTime = self.userOperationTimes[groupId][user];
return (userTime.lastVTMintTime, userTime.lastYTMintTime, userTime.lastVTRedeemTime, userTime.lastYTRedeemTime);
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { IProtocol } from "../interfaces/IProtocol.sol";
import { CustomRevert } from "./CustomRevert.sol";
import { ProtocolConfigState } from "./ProtocolConfigState.sol";
import { TimeManagementLibrary } from "./TimeManagementLibrary.sol";
import { DOperationContext } from "../declarations/DOperationContext.sol";
import { OperationContextLibrary } from "./OperationContextLibrary.sol";
import { GroupState } from "../types/CommonTypes.sol";
import { GroupId } from "../types/GroupId.sol";
import { Currency, CurrencyLibrary } from "../types/Currency.sol";
import { CollateralInfo } from "../types/CommonTypes.sol";
import { OperationTypes } from "../types/CommonTypes.sol";
library ValidationLibrary {
using CurrencyLibrary for Currency;
using OperationContextLibrary for DOperationContext.OperationContext;
using TimeManagementLibrary for TimeManagementLibrary.TimeStorage;
using CustomRevert for bytes4;
error ErrorMintPaused(GroupId groupId);
error ErrorRedeemPaused(GroupId groupId);
error InvalidOperationType(GroupId groupId);
error InsufficientAllowance(GroupId groupId);
error InsufficientBalance(GroupId groupId);
error InvalidPaymentAmount();
error InvalidMinMaxCollateralAmount();
/**
* @notice Validates the operation based on the context.
* @param configStorage The configuration storage.
* param timeStorage The time storage.
* @param groupId The group ID.
* @param context The operation context.
*/
function validateOperation(
ProtocolConfigState.ConfigStorage storage configStorage,
// TimeManagementLibrary.TimeStorage storage timeStorage,
GroupId groupId,
DOperationContext.OperationContext memory context
) public view {
if (context.isMintOperation()) {
if (!ProtocolConfigState.getGlobalMintFlag(configStorage) || !ProtocolConfigState.getMintFlag(configStorage, groupId)) {
ErrorMintPaused.selector.revertWith(groupId);
}
} else if (context.isRedeemOperation()) {
if (!ProtocolConfigState.getGlobalRedeemFlag(configStorage) || !ProtocolConfigState.getRedeemFlag(configStorage, groupId)) {
ErrorRedeemPaused.selector.revertWith(groupId);
}
context.validateRecipient();
validateUserBalance(context);
// timeStorage.validateTimeRestrictions(context, msgSender);
} else {
InvalidOperationType.selector.revertWith(groupId);
}
}
/**
* @notice Validates the user balance for redeem operations.
* @param context The operation context.
*/
function validateUserBalance(DOperationContext.OperationContext memory context) public view {
uint8 operationType = context.getOperationType();
uint256 tokenIn = context.getAmount0();
address msgSender = context.getRecipient();
if (operationType == uint8(OperationTypes.OP_TYPE_REDEEM_VT)) {
if (context.getXToken().balanceOf(msgSender) < tokenIn) InsufficientBalance.selector.revertWith(context.groupId);
if (context.getXToken().allowance(msgSender, address(this)) < tokenIn) InsufficientAllowance.selector.revertWith(context.groupId);
} else if (operationType == uint8(OperationTypes.OP_TYPE_REDEEM_YT)) {
if (context.getRebalancePool().balanceOf(msgSender) < tokenIn) InsufficientBalance.selector.revertWith(context.groupId);
if (context.getRebalancePool().allowance(msgSender, address(this)) < tokenIn) InsufficientAllowance.selector.revertWith(context.groupId);
} else {
InvalidOperationType.selector.revertWith(context.groupId);
}
}
/**
* @notice Checks if the payment token is supported by the group.
* @param acceptableCollaterals The acceptable collaterals for the group.
* @param baseIn The base amount for the operation.
* @param paymentToken The payment token to check.
* @param isMintOperation True if the operation is a mint operation, false otherwise.
* @param msgValue The amount of ETH sent with the transaction.
* @return True if the payment token is supported, false otherwise.
*/
function isPaymentTokenSupported(
CollateralInfo[] memory acceptableCollaterals,
uint256 baseIn,
Currency paymentToken,
bool isMintOperation,
uint256 msgValue
) public pure returns (bool, CollateralInfo memory) {
CollateralInfo memory collateralInfo;
if (paymentToken.isNative() && (isMintOperation && msgValue != baseIn)) {
InvalidPaymentAmount.selector.revertWith();
}
for (uint256 i = 0; i < acceptableCollaterals.length; ) {
if (acceptableCollaterals[i].token.equals(paymentToken)) {
if (isMintOperation) {
if (baseIn <= acceptableCollaterals[i].minAmount || baseIn >= acceptableCollaterals[i].maxAmount) {
InvalidMinMaxCollateralAmount.selector.revertWith();
}
}
collateralInfo = acceptableCollaterals[i];
return (true, collateralInfo);
}
unchecked {
++i;
}
}
return (false, collateralInfo);
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { GroupId } from "../types/GroupId.sol";
// solhint-disable
/// @title Library for reverting with custom errors efficiently
/// @notice Contains functions for reverting with custom errors with different argument types efficiently
/// @dev To use this library, declare `using CustomRevert for bytes4;` and replace `revert CustomError()` with
/// `CustomError.selector.revertWith()`
/// @dev The functions may tamper with the free memory pointer but it is fine since the call context is exited immediately
library CustomRevert {
/// @dev Reverts with the selector of a custom error in the scratch space
function revertWith(bytes4 selector) internal pure {
assembly("memory-safe") {
mstore(0, selector)
revert(0, 0x04)
}
}
/// @dev Reverts with a custom error with an address argument in the scratch space
function revertWith(bytes4 selector, address addr) internal pure {
assembly("memory-safe") {
mstore(0, selector)
mstore(0x04, and(addr, 0xffffffffffffffffffffffffffffffffffffffff))
revert(0, 0x24)
}
}
/// @dev Reverts with a custom error with an int24 argument in the scratch space
function revertWith(bytes4 selector, int24 value) internal pure {
assembly("memory-safe") {
mstore(0, selector)
mstore(0x04, signextend(2, value))
revert(0, 0x24)
}
}
/// @dev Reverts with a custom error with a uint160 argument in the scratch space
function revertWith(bytes4 selector, uint160 value) internal pure {
assembly("memory-safe") {
mstore(0, selector)
mstore(0x04, and(value, 0xffffffffffffffffffffffffffffffffffffffff))
revert(0, 0x24)
}
}
/// @dev Reverts with a custom error with two int24 arguments
function revertWith(bytes4 selector, int24 value1, int24 value2) internal pure {
assembly("memory-safe") {
let fmp := mload(0x40)
mstore(fmp, selector)
mstore(add(fmp, 0x04), signextend(2, value1))
mstore(add(fmp, 0x24), signextend(2, value2))
revert(fmp, 0x44)
}
}
/// @dev Reverts with a custom error with two uint160 arguments
function revertWith(bytes4 selector, uint160 value1, uint160 value2) internal pure {
assembly("memory-safe") {
let fmp := mload(0x40)
mstore(fmp, selector)
mstore(add(fmp, 0x04), and(value1, 0xffffffffffffffffffffffffffffffffffffffff))
mstore(add(fmp, 0x24), and(value2, 0xffffffffffffffffffffffffffffffffffffffff))
revert(fmp, 0x44)
}
}
/// @dev Reverts with a custom error with two address arguments
function revertWith(bytes4 selector, address value1, address value2) internal pure {
assembly("memory-safe") {
mstore(0, selector)
mstore(0x04, and(value1, 0xffffffffffffffffffffffffffffffffffffffff))
mstore(0x24, and(value2, 0xffffffffffffffffffffffffffffffffffffffff))
revert(0, 0x44)
}
}
/// @dev Reverts with a custom error with a bytes32 argument in the scratch space
function revertWith(bytes4 selector, bytes32 value) internal pure {
assembly("memory-safe") {
mstore(0, selector)
mstore(0x04, value)
revert(0, 0x24)
}
}
/// @dev Reverts with a custom error with a bytes32 argument in the scratch space
function revertWith(bytes4 selector, GroupId value) internal pure {
bytes32 valueBytes = GroupId.unwrap(value);
assembly("memory-safe") {
mstore(0, selector)
mstore(0x04, valueBytes)
revert(0, 0x24)
}
}
/// @dev Reverts with a custom error with a bytes32 and an address argument
function revertWith(bytes4 selector, GroupId value, address addr) internal pure {
bytes32 valueBytes = GroupId.unwrap(value);
assembly("memory-safe") {
mstore(0x00, selector)
mstore(0x04, valueBytes)
mstore(0x24, and(addr, 0xffffffffffffffffffffffffffffffffffffffff))
revert(0x00, 0x44)
}
}
/// @dev Reverts with a custom error with a bytes32, address, and uint256 arguments
function revertWith(bytes4 selector, GroupId value, address addr, uint256 amount) internal pure {
bytes32 valueBytes = GroupId.unwrap(value);
assembly("memory-safe") {
mstore(0x00, selector)
mstore(0x04, valueBytes)
mstore(0x24, and(addr, 0xffffffffffffffffffffffffffffffffffffffff))
mstore(0x44, amount)
revert(0x00, 0x64)
}
}
/// @notice bubble up the revert message returned by a call and revert with the selector provided
/// @dev this function should only be used with custom errors of the type `CustomError(address target, bytes revertReason)`
function bubbleUpAndRevertWith(bytes4 selector, address addr) internal pure {
assembly("memory-safe") {
let size := returndatasize()
let fmp := mload(0x40)
// Encode selector, address, offset, size, data
mstore(fmp, selector)
mstore(add(fmp, 0x04), addr)
mstore(add(fmp, 0x24), 0x40)
mstore(add(fmp, 0x44), size)
returndatacopy(add(fmp, 0x64), 0, size)
// Ensure the size is a multiple of 32 bytes
let encodedSize := add(0x64, mul(div(add(size, 31), 32), 32))
revert(fmp, encodedSize)
}
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { GroupKey } from "./GroupKey.sol";
type GroupId is bytes32;
// @notice library for computing the ID of a group
library GroupIdLibrary {
using GroupIdLibrary for GroupId;
function toId(GroupKey memory groupKey) internal pure returns (GroupId groupId) {
groupId = GroupId.wrap(keccak256(abi.encode(groupKey)));
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { DTokenRegistry } from "../declarations/DTokenRegistry.sol";
import { WordCodec } from "../libs/WordCodec.sol";
import { Address, AddressLibrary } from "../types/Address.sol";
import { Currency, CurrencyLibrary } from "../types/Currency.sol";
import { DefaultFeeParams, FeeParams, FeePermissions, CollateralInfo, GroupState } from "../types/CommonTypes.sol";
import { ITokenRegistry } from "../interfaces/ITokenRegistry.sol";
/**
* @title GroupStateHelper
* @notice Library for managing group settings and fee parameters in a space-efficient manner using bit packing
* @dev Uses WordCodec for bit manipulation operations to store and retrieve various parameters
*/
type GroupSettings is bytes32;
library GroupStateHelper {
using WordCodec for bytes32;
using CurrencyLibrary for Currency;
using AddressLibrary for address;
// Group Settings bit layout (total 256 bits):
uint256 private constant A_TOKEN_DECIMALS_OFFSET = 0; // [0-7]: aToken decimals (8 bits)
uint256 private constant X_TOKEN_DECIMALS_OFFSET = 8; // [8-15]: xToken decimals (8 bits)
uint256 private constant BASE_TOKEN_DECIMALS_OFFSET = 16; // [16-23]: baseToken decimals (8 bits)
uint256 private constant YIELD_BEARING_TOKEN_DECIMALS_OFFSET = 24; // [24-31]: yieldBearingToken decimals (8 bits)
uint256 private constant HOOK_PERMISSIONS_OFFSET = 32; // [32-47]: hook permissions (16 bits)
uint256 private constant FEE_PERMISSIONS_OFFSET = 48; // [48-49]: fee permissions (2 bits)
uint256 private constant FEE_MODEL_OFFSET = 50; // [50-57]: fee model (8 bits)
uint256 private constant WRAPPING_REQUIRED_OFFSET = 58; // [58]: wrapping required (1 bit)
uint256 private constant STABILITY_RATIO_OFFSET = 59; // [59-154]: stability ratio (96 bits)
uint256 private constant STABILITY_TRIGGER_RATIO_OFFSET = 155; // [155-250]: stability triggering ratio (96 bits)
// Fee Parameters bit layout (total 256 bits):
uint256 private constant MAX_FEE_OFFSET = 0; // [0-15]: max fee (16 bits)
uint256 private constant MIN_FEE_OFFSET = 16; // [16-31]: min fee (16 bits)
uint256 private constant BASE_FEE_OFFSET = 32; // [32-47]: base fee (16 bits)
uint256 private constant YIELD_FEE_VT_OFFSET = 48; // [48-63]: yield fee VT (16 bits)
uint256 private constant YIELD_FEE_YT_OFFSET = 64; // [64-79]: yield fee YT (16 bits)
uint256 private constant REDEEM_FEE_YT_OFFSET = 80; // [80-103]: redeem fee YT (24 bits)
uint256 private constant MINT_FEE_YT_OFFSET = 104; // [104-127]: mint fee YT (24 bits)
uint256 private constant REDEEM_FEE_VT_OFFSET = 128; // [128-151]: redeem fee VT (24 bits)
uint256 private constant MINT_FEE_VT_OFFSET = 152; // [152-175]: mint fee VT (24 bits)
uint256 private constant STABILITY_MINT_FEE_VT_OFFSET = 176; // [176-191]: stability mint fee VT (16 bits)
uint256 private constant STABILITY_MINT_FEE_YT_OFFSET = 192; // [192-207]: stability mint fee YT (16 bits)
uint256 private constant STABILITY_REDEEM_FEE_VT_OFFSET = 208; // [208-223]: stability redeem fee VT (16 bits)
uint256 private constant STABILITY_REDEEM_FEE_YT_OFFSET = 224; // [224-239]: stability redeem fee YT (16 bits)
uint256 private constant PROTOCOL_FEE_OFFSET = 240; // [240-255]: protocol fee (16 bits)
// Common bit lengths
uint256 private constant LENGTH_1BIT = 1;
uint256 private constant LENGTH_2BITS = 2;
uint256 private constant LENGTH_8BITS = 8;
uint256 private constant LENGTH_16BITS = 16;
uint256 private constant LENGTH_24BITS = 24;
uint256 private constant LENGTH_96BITS = 96;
/**
* @notice Retrieves the aToken decimals from group settings
* @param groupSettings The packed group settings
* @return The aToken decimals
*/
function getATokenDecimals(GroupSettings groupSettings) internal pure returns (uint8) {
return uint8(GroupSettings.unwrap(groupSettings).decodeUint(A_TOKEN_DECIMALS_OFFSET, LENGTH_8BITS));
}
/**
* @notice Retrieves the xToken decimals from group settings
* @param groupSettings The packed group settings
* @return The xToken decimals
*/
function getXTokenDecimals(GroupSettings groupSettings) internal pure returns (uint8) {
return uint8(GroupSettings.unwrap(groupSettings).decodeUint(X_TOKEN_DECIMALS_OFFSET, LENGTH_8BITS));
}
/**
* @notice Retrieves the baseToken decimals from group settings
* @param groupSettings The packed group settings
* @return The baseToken decimals
*/
function getBaseTokenDecimals(GroupSettings groupSettings) internal pure returns (uint8) {
return uint8(GroupSettings.unwrap(groupSettings).decodeUint(BASE_TOKEN_DECIMALS_OFFSET, LENGTH_8BITS));
}
/**
* @notice Retrieves the yieldBearingToken decimals from group settings
* @param groupSettings The packed group settings
* @return The yieldBearingToken decimals
*/
function getYieldBearingTokenDecimals(GroupSettings groupSettings) internal pure returns (uint8) {
return uint8(GroupSettings.unwrap(groupSettings).decodeUint(YIELD_BEARING_TOKEN_DECIMALS_OFFSET, LENGTH_8BITS));
}
/**
* @notice Retrieves the hook permissions from group settings
* @param groupSettings The packed group settings
* @return The hook permissions
*/
function getHookPermissions(GroupSettings groupSettings) internal pure returns (uint16) {
return uint16(GroupSettings.unwrap(groupSettings).decodeUint(HOOK_PERMISSIONS_OFFSET, LENGTH_16BITS));
}
/**
* @notice Retrieves the fee permissions from group settings
* @param groupSettings The packed group settings
* @return FeePermissions struct containing permission flags
*/
function getFeePermissions(GroupSettings groupSettings) internal pure returns (FeePermissions memory) {
bytes32 raw = GroupSettings.unwrap(groupSettings);
return
FeePermissions({ isDynamic: raw.decodeBool(FEE_PERMISSIONS_OFFSET + 1), allowDelegation: raw.decodeBool(FEE_PERMISSIONS_OFFSET) });
}
/**
* @notice Retrieves the fee model from group settings
* @param groupSettings The packed group settings
* @return The fee model
*/
function getFeeModel(GroupSettings groupSettings) internal pure returns (uint8) {
return uint8(GroupSettings.unwrap(groupSettings).decodeUint(FEE_MODEL_OFFSET, LENGTH_8BITS));
}
/**
* @notice Checks if wrapping is required from group settings
* @param groupSettings The packed group settings
* @return True if wrapping is required, false otherwise
*/
function isWrappingRequired(GroupSettings groupSettings) internal pure returns (bool) {
return GroupSettings.unwrap(groupSettings).decodeBool(WRAPPING_REQUIRED_OFFSET);
}
/**
* @notice Retrieves the stability ratio from group settings
* @param groupSettings The packed group settings
* @return The stability ratio (scaled by 1e18)
*/
function getStabilityRatio(GroupSettings groupSettings) internal pure returns (uint96) {
return uint96(GroupSettings.unwrap(groupSettings).decodeUint(STABILITY_RATIO_OFFSET, LENGTH_96BITS));
}
/**
* @notice Retrieves the stability triggering ratio from group settings
* @param groupSettings The packed group settings
* @return The stability triggering ratio (scaled by 1e18)
*/
function getStabilityTriggeringRatio(GroupSettings groupSettings) internal pure returns (uint96) {
return uint96(GroupSettings.unwrap(groupSettings).decodeUint(STABILITY_TRIGGER_RATIO_OFFSET, LENGTH_96BITS));
}
/**
* @notice Sets the aToken decimals in group settings
* @param groupSettings Current group settings
* @param decimals The new aToken decimals
* @return Updated group settings
*/
function setATokenDecimals(GroupSettings groupSettings, uint8 decimals) internal pure returns (GroupSettings) {
bytes32 updated = GroupSettings.unwrap(groupSettings).insertUint(uint256(decimals), A_TOKEN_DECIMALS_OFFSET, LENGTH_8BITS);
return GroupSettings.wrap(updated);
}
/**
* @notice Sets the xToken decimals in group settings
* @param groupSettings Current group settings
* @param decimals The new xToken decimals
* @return Updated group settings
*/
function setXTokenDecimals(GroupSettings groupSettings, uint8 decimals) internal pure returns (GroupSettings) {
bytes32 updated = GroupSettings.unwrap(groupSettings).insertUint(uint256(decimals), X_TOKEN_DECIMALS_OFFSET, LENGTH_8BITS);
return GroupSettings.wrap(updated);
}
/**
* @notice Sets the baseToken decimals in group settings
* @param groupSettings Current group settings
* @param decimals The new baseToken decimals
* @return Updated group settings
*/
function setBaseTokenDecimals(GroupSettings groupSettings, uint8 decimals) internal pure returns (GroupSettings) {
bytes32 updated = GroupSettings.unwrap(groupSettings).insertUint(uint256(decimals), BASE_TOKEN_DECIMALS_OFFSET, LENGTH_8BITS);
return GroupSettings.wrap(updated);
}
/**
* @notice Sets the yieldBearingToken decimals in group settings
* @param groupSettings Current group settings
* @param decimals The new yieldBearingToken decimals
* @return Updated group settings
*/
function setYieldBearingTokenDecimals(GroupSettings groupSettings, uint8 decimals) internal pure returns (GroupSettings) {
bytes32 updated = GroupSettings.unwrap(groupSettings).insertUint(uint256(decimals), YIELD_BEARING_TOKEN_DECIMALS_OFFSET, LENGTH_8BITS);
return GroupSettings.wrap(updated);
}
/**
* @notice Sets the hook permissions in group settings
* @param groupSettings Current group settings
* @param hookPermissions The new hook permissions
* @return Updated group settings
*/
function setHookPermissions(GroupSettings groupSettings, uint16 hookPermissions) internal pure returns (GroupSettings) {
bytes32 updated = GroupSettings.unwrap(groupSettings).insertUint(uint256(hookPermissions), HOOK_PERMISSIONS_OFFSET, LENGTH_16BITS);
return GroupSettings.wrap(updated);
}
/**
* @notice Sets the fee permissions in group settings
* @param groupSettings Current group settings
* @param permissions The new fee permissions
* @return Updated group settings
*/
function setFeePermissions(GroupSettings groupSettings, FeePermissions memory permissions) internal pure returns (GroupSettings) {
bytes32 raw = GroupSettings.unwrap(groupSettings);
raw = raw.insertBool(permissions.allowDelegation, FEE_PERMISSIONS_OFFSET);
raw = raw.insertBool(permissions.isDynamic, FEE_PERMISSIONS_OFFSET + 1);
return GroupSettings.wrap(raw);
}
/**
* @notice Sets the fee model in group settings
* @param groupSettings Current group settings
* @param feeModel The new fee model
* @return Updated group settings
*/
function setFeeModel(GroupSettings groupSettings, uint8 feeModel) internal pure returns (GroupSettings) {
bytes32 updated = GroupSettings.unwrap(groupSettings).insertUint(uint256(feeModel), FEE_MODEL_OFFSET, LENGTH_8BITS);
return GroupSettings.wrap(updated);
}
/**
* @notice Sets the wrapping required flag in group settings
* @param groupSettings Current group settings
* @param wrappingRequired The new wrapping required flag
* @return Updated group settings
*/
function setWrappingRequired(GroupSettings groupSettings, bool wrappingRequired) internal pure returns (GroupSettings) {
bytes32 updated = GroupSettings.unwrap(groupSettings).insertBool(wrappingRequired, WRAPPING_REQUIRED_OFFSET);
return GroupSettings.wrap(updated);
}
/**
* @notice Sets the stability ratio in group settings
* @param groupSettings Current group settings
* @param stabilityRatio The new stability ratio (scaled by 1e18)
* @return Updated group settings
*/
function setStabilityRatio(GroupSettings groupSettings, uint96 stabilityRatio) internal pure returns (GroupSettings) {
bytes32 updated = GroupSettings.unwrap(groupSettings).insertUint(uint256(stabilityRatio), STABILITY_RATIO_OFFSET, LENGTH_96BITS);
return GroupSettings.wrap(updated);
}
/**
* @notice Sets the stability triggering ratio in group settings
* @param groupSettings Current group settings
* @param stabilityTriggeringRatio The new stability triggering ratio (scaled by 1e18)
* @return Updated group settings
*/
function setStabilityTriggeringRatio(GroupSettings groupSettings, uint96 stabilityTriggeringRatio) internal pure returns (GroupSettings) {
bytes32 updated = GroupSettings.unwrap(groupSettings).insertUint(
uint256(stabilityTriggeringRatio),
STABILITY_TRIGGER_RATIO_OFFSET,
LENGTH_96BITS
);
return GroupSettings.wrap(updated);
}
/**
* @notice Retrieves all fee parameters from packed fees
* @param feesPacked The packed fees
* @return FeeParams struct containing all fee parameters
*/
function getFeeParams(bytes32 feesPacked) internal pure returns (FeeParams memory) {
return
FeeParams({
mintFeeVT: uint24(feesPacked.decodeUint(MINT_FEE_VT_OFFSET, LENGTH_24BITS)),
redeemFeeVT: uint24(feesPacked.decodeUint(REDEEM_FEE_VT_OFFSET, LENGTH_24BITS)),
mintFeeYT: uint24(feesPacked.decodeUint(MINT_FEE_YT_OFFSET, LENGTH_24BITS)),
redeemFeeYT: uint24(feesPacked.decodeUint(REDEEM_FEE_YT_OFFSET, LENGTH_24BITS)),
// Upcast 16-bit stability fees to 24-bit
stabilityMintFeeVT: uint24(uint16(feesPacked.decodeUint(STABILITY_MINT_FEE_VT_OFFSET, LENGTH_16BITS))),
stabilityMintFeeYT: uint24(uint16(feesPacked.decodeUint(STABILITY_MINT_FEE_YT_OFFSET, LENGTH_16BITS))),
stabilityRedeemFeeVT: uint24(uint16(feesPacked.decodeUint(STABILITY_REDEEM_FEE_VT_OFFSET, LENGTH_16BITS))),
stabilityRedeemFeeYT: uint24(uint16(feesPacked.decodeUint(STABILITY_REDEEM_FEE_YT_OFFSET, LENGTH_16BITS))),
// Upcast 16-bit yield fees to 24-bit
yieldFeeVT: uint24(uint16(feesPacked.decodeUint(YIELD_FEE_VT_OFFSET, LENGTH_16BITS))),
yieldFeeYT: uint24(uint16(feesPacked.decodeUint(YIELD_FEE_YT_OFFSET, LENGTH_16BITS))),
protocolFee: uint16(feesPacked.decodeUint(PROTOCOL_FEE_OFFSET, LENGTH_16BITS))
});
}
/**
* @notice Retrieves the default fee parameters from packed fees
* @param feesPacked The packed fees
* @return DefaultFeeParams struct containing min, max and base fees
*/
function getDefaultFeeParams(bytes32 feesPacked) internal pure returns (DefaultFeeParams memory) {
return
DefaultFeeParams({
baseFee: uint24(feesPacked.decodeUint(BASE_FEE_OFFSET, LENGTH_16BITS)),
maxFee: uint16(feesPacked.decodeUint(MAX_FEE_OFFSET, LENGTH_16BITS)),
minFee: uint16(feesPacked.decodeUint(MIN_FEE_OFFSET, LENGTH_16BITS))
});
}
/**
* @notice Retrieves the max fee from packed fees
* @param feesPacked The packed fees
* @return The max fee value (24-bit)
*/
function getMaxFee(bytes32 feesPacked) internal pure returns (uint24) {
return uint24(uint16(feesPacked.decodeUint(MAX_FEE_OFFSET, LENGTH_16BITS)));
}
/**
* @notice Retrieves the min fee from packed fees
* @param feesPacked The packed fees
* @return The min fee value (24-bit)
*/
function getMinFee(bytes32 feesPacked) internal pure returns (uint24) {
return uint24(uint16(feesPacked.decodeUint(MIN_FEE_OFFSET, LENGTH_16BITS)));
}
/**
* @notice Retrieves the base fee from packed fees
* @param feesPacked The packed fees
* @return The base fee value (24-bit)
*/
function getBaseFee(bytes32 feesPacked) internal pure returns (uint24) {
return uint24(uint16(feesPacked.decodeUint(BASE_FEE_OFFSET, LENGTH_16BITS)));
}
/**
* @notice Retrieves the yield fee VT from packed fees
* @param feesPacked The packed fees
* @return The yield fee VT value (24-bit)
*/
function getYieldFeeVT(bytes32 feesPacked) internal pure returns (uint24) {
return uint24(uint16(feesPacked.decodeUint(YIELD_FEE_VT_OFFSET, LENGTH_16BITS)));
}
/**
* @notice Retrieves the yield fee YT from packed fees
* @param feesPacked The packed fees
* @return The yield fee YT value (24-bit)
*/
function getYieldFeeYT(bytes32 feesPacked) internal pure returns (uint24) {
return uint24(uint16(feesPacked.decodeUint(YIELD_FEE_YT_OFFSET, LENGTH_16BITS)));
}
/**
* @notice Retrieves the mint fee VT from packed fees
* @param feesPacked The packed fees
* @return The mint fee VT value (24-bit)
*/
function getMintFeeVT(bytes32 feesPacked) internal pure returns (uint24) {
return uint24(feesPacked.decodeUint(MINT_FEE_VT_OFFSET, LENGTH_24BITS));
}
/**
* @notice Retrieves the redeem fee VT from packed fees
* @param feesPacked The packed fees
* @return The redeem fee VT value (24-bit)
*/
function getRedeemFeeVT(bytes32 feesPacked) internal pure returns (uint24) {
return uint24(feesPacked.decodeUint(REDEEM_FEE_VT_OFFSET, LENGTH_24BITS));
}
/**
* @notice Retrieves the mint fee YT from packed fees
* @param feesPacked The packed fees
* @return The mint fee YT value (24-bit)
*/
function getMintFeeYT(bytes32 feesPacked) internal pure returns (uint24) {
return uint24(feesPacked.decodeUint(MINT_FEE_YT_OFFSET, LENGTH_24BITS));
}
/**
* @notice Retrieves the redeem fee YT from packed fees
* @param feesPacked The packed fees
* @return The redeem fee YT value (24-bit)
*/
function getRedeemFeeYT(bytes32 feesPacked) internal pure returns (uint24) {
return uint24(feesPacked.decodeUint(REDEEM_FEE_YT_OFFSET, LENGTH_24BITS));
}
/**
* @notice Retrieves the stability mint fee VT from packed fees
* @param feesPacked The packed fees
* @return The stability mint fee VT value (24-bit)
*/
function getStabilityMintFeeVT(bytes32 feesPacked) internal pure returns (uint24) {
return uint24(uint16(feesPacked.decodeUint(STABILITY_MINT_FEE_VT_OFFSET, LENGTH_16BITS)));
}
/**
* @notice Retrieves the stability mint fee YT from packed fees
* @param feesPacked The packed fees
* @return The stability mint fee YT value (24-bit)
*/
function getStabilityMintFeeYT(bytes32 feesPacked) internal pure returns (uint24) {
return uint24(uint16(feesPacked.decodeUint(STABILITY_MINT_FEE_YT_OFFSET, LENGTH_16BITS)));
}
/**
* @notice Retrieves the stability redeem fee VT from packed fees
* @param feesPacked The packed fees
* @return The stability redeem fee VT value (24-bit)
*/
function getStabilityRedeemFeeVT(bytes32 feesPacked) internal pure returns (uint24) {
return uint24(uint16(feesPacked.decodeUint(STABILITY_REDEEM_FEE_VT_OFFSET, LENGTH_16BITS)));
}
/**
* @notice Retrieves the stability redeem fee YT from packed fees
* @param feesPacked The packed fees
* @return The stability redeem fee YT value (24-bit)
*/
function getStabilityRedeemFeeYT(bytes32 feesPacked) internal pure returns (uint24) {
return uint24(uint16(feesPacked.decodeUint(STABILITY_REDEEM_FEE_YT_OFFSET, LENGTH_16BITS)));
}
/**
* @notice Retrieves the protocol fee from packed fees
* @param feesPacked The packed fees
* @return The protocol fee value (24-bit)
*/
function getProtocolFee(bytes32 feesPacked) internal pure returns (uint24) {
return uint24(uint16(feesPacked.decodeUint(PROTOCOL_FEE_OFFSET, LENGTH_16BITS)));
}
/**
* @notice Sets the max fee in packed fees
* @param feesPacked Current packed fees
* @param maxFee The new max fee value (truncated to 16-bit)
* @return Updated packed fees
*/
function setMaxFee(bytes32 feesPacked, uint24 maxFee) internal pure returns (bytes32) {
return feesPacked.insertUint(uint256(uint16(maxFee)), MAX_FEE_OFFSET, LENGTH_16BITS);
}
/**
* @notice Sets the min fee in packed fees
* @param feesPacked Current packed fees
* @param minFee The new min fee value (truncated to 16-bit)
* @return Updated packed fees
*/
function setMinFee(bytes32 feesPacked, uint24 minFee) internal pure returns (bytes32) {
return feesPacked.insertUint(uint256(uint16(minFee)), MIN_FEE_OFFSET, LENGTH_16BITS);
}
/**
* @notice Sets the base fee in packed fees
* @param feesPacked Current packed fees
* @param baseFee The new base fee value (truncated to 16-bit)
* @return Updated packed fees
*/
function setBaseFee(bytes32 feesPacked, uint24 baseFee) internal pure returns (bytes32) {
return feesPacked.insertUint(uint256(uint16(baseFee)), BASE_FEE_OFFSET, LENGTH_16BITS);
}
/**
* @notice Sets the yield fee VT in packed fees
* @param feesPacked Current packed fees
* @param yieldFeeVT The new yield fee VT value (truncated to 16-bit)
* @return Updated packed fees
*/
function setYieldFeeVT(bytes32 feesPacked, uint24 yieldFeeVT) internal pure returns (bytes32) {
return feesPacked.insertUint(uint256(uint16(yieldFeeVT)), YIELD_FEE_VT_OFFSET, LENGTH_16BITS);
}
/**
* @notice Sets the yield fee YT in packed fees
* @param feesPacked Current packed fees
* @param yieldFeeYT The new yield fee YT value (truncated to 16-bit)
* @return Updated packed fees
*/
function setYieldFeeYT(bytes32 feesPacked, uint24 yieldFeeYT) internal pure returns (bytes32) {
return feesPacked.insertUint(uint256(uint16(yieldFeeYT)), YIELD_FEE_YT_OFFSET, LENGTH_16BITS);
}
/**
* @notice Sets the mint fee VT in packed fees
* @param feesPacked Current packed fees
* @param mintFeeVT The new mint fee VT value
* @return Updated packed fees
*/
function setMintFeeVT(bytes32 feesPacked, uint24 mintFeeVT) internal pure returns (bytes32) {
return feesPacked.insertUint(uint256(mintFeeVT), MINT_FEE_VT_OFFSET, LENGTH_24BITS);
}
/**
* @notice Sets the redeem fee VT in packed fees
* @param feesPacked Current packed fees
* @param redeemFeeVT The new redeem fee VT value
* @return Updated packed fees
*/
function setRedeemFeeVT(bytes32 feesPacked, uint24 redeemFeeVT) internal pure returns (bytes32) {
return feesPacked.insertUint(uint256(redeemFeeVT), REDEEM_FEE_VT_OFFSET, LENGTH_24BITS);
}
/**
* @notice Sets the mint fee YT in packed fees
* @param feesPacked Current packed fees
* @param mintFeeYT The new mint fee YT value
* @return Updated packed fees
*/
function setMintFeeYT(bytes32 feesPacked, uint24 mintFeeYT) internal pure returns (bytes32) {
return feesPacked.insertUint(uint256(mintFeeYT), MINT_FEE_YT_OFFSET, LENGTH_24BITS);
}
/**
* @notice Sets the redeem fee YT in packed fees
* @param feesPacked Current packed fees
* @param redeemFeeYT The new redeem fee YT value
* @return Updated packed fees
*/
function setRedeemFeeYT(bytes32 feesPacked, uint24 redeemFeeYT) internal pure returns (bytes32) {
return feesPacked.insertUint(uint256(redeemFeeYT), REDEEM_FEE_YT_OFFSET, LENGTH_24BITS);
}
/**
* @notice Sets the stability mint fee VT in packed fees
* @param feesPacked Current packed fees
* @param stabilityMintFeeVT The new stability mint fee VT value (truncated to 16-bit)
* @return Updated packed fees
*/
function setStabilityMintFeeVT(bytes32 feesPacked, uint24 stabilityMintFeeVT) internal pure returns (bytes32) {
return feesPacked.insertUint(uint256(uint16(stabilityMintFeeVT)), STABILITY_MINT_FEE_VT_OFFSET, LENGTH_16BITS);
}
/**
* @notice Sets the stability mint fee YT in packed fees
* @param feesPacked Current packed fees
* @param stabilityMintFeeYT The new stability mint fee YT value (truncated to 16-bit)
* @return Updated packed fees
*/
function setStabilityMintFeeYT(bytes32 feesPacked, uint24 stabilityMintFeeYT) internal pure returns (bytes32) {
return feesPacked.insertUint(uint256(uint16(stabilityMintFeeYT)), STABILITY_MINT_FEE_YT_OFFSET, LENGTH_16BITS);
}
/**
* @notice Sets the stability redeem fee VT in packed fees
* @param feesPacked Current packed fees
* @param stabilityRedeemFeeVT The new stability redeem fee VT value (truncated to 16-bit)
* @return Updated packed fees
*/
function setStabilityRedeemFeeVT(bytes32 feesPacked, uint24 stabilityRedeemFeeVT) internal pure returns (bytes32) {
return feesPacked.insertUint(uint256(uint16(stabilityRedeemFeeVT)), STABILITY_REDEEM_FEE_VT_OFFSET, LENGTH_16BITS);
}
/**
* @notice Sets the stability redeem fee YT in packed fees
* @param feesPacked Current packed fees
* @param stabilityRedeemFeeYT The new stability redeem fee YT value (truncated to 16-bit)
* @return Updated packed fees
*/
function setStabilityRedeemFeeYT(bytes32 feesPacked, uint24 stabilityRedeemFeeYT) internal pure returns (bytes32) {
return feesPacked.insertUint(uint256(uint16(stabilityRedeemFeeYT)), STABILITY_REDEEM_FEE_YT_OFFSET, LENGTH_16BITS);
}
/**
* @notice Sets the protocol fee in packed fees
* @param feesPacked Current packed fees
* @param protocolFee The new protocol fee value (truncated to 16-bit)
* @return Updated packed fees
*/
function setProtocolFee(bytes32 feesPacked, uint24 protocolFee) internal pure returns (bytes32) {
return feesPacked.insertUint(uint256(uint16(protocolFee)), PROTOCOL_FEE_OFFSET, LENGTH_16BITS);
}
// Group Core Components Getters
/**
* @notice Retrieves the hook contract from group state
* @param groupState The group state
* @return The hook contract address wrapper
*/
function getHookContract(GroupState memory groupState) internal pure returns (Address) {
return groupState.hookContract;
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { Currency } from "./Currency.sol";
import { DTokenRegistry } from "../declarations/DTokenRegistry.sol";
import { Address } from "../types/Address.sol";
// Enums representing different fee models that can be applied.
enum OperationTypes {
OP_TYPE_MINT_VT,
OP_TYPE_MINT_YT,
OP_TYPE_REDEEM_VT,
OP_TYPE_REDEEM_YT
}
enum FeeModel {
NONE, // No fees.
MANAGEMENT_FEE, // Management fee model.
VARIABLE_FUNDING_FEE, // Fee that varies based on funding.
FIXED_FUNDING_FEE, // Fixed fee for funding.
CURATED_PAIRS_FEE, // Curated pairs fee model.
BLANK_FEE // Blank fee model.
}
// Information about acceptable collaterals
struct CollateralInfo {
Currency token; // Token address for collateral
uint8 decimals; // Decimals for the collateral token
uint256 minAmount; // Minimum amount for both usage minting and redeeming (as desired token)
uint256 maxAmount; // Maximum amount for both usage minting and redeeming (as desired token)
}
// DefaultFeeParams defines the basic fee structure with base, min, and max fees.
struct DefaultFeeParams {
uint24 baseFee; // The base fee applied when no flags are present.
uint24 minFee; // The minimum fee allowed for dynamic fee models.
uint24 maxFee; // The maximum fee allowed.
}
// FeeParams defines specific fees for different token types and operations.
struct FeeParams {
uint24 mintFeeVT; // Minting fee for volatile tokens (VT).
uint24 redeemFeeVT; // Redemption fee for volatile tokens (VT).
uint24 mintFeeYT; // Minting fee for yield tokens (YT).
uint24 redeemFeeYT; // Redemption fee for yield tokens (YT).
uint24 stabilityMintFeeVT; // Stability minting fee for volatile tokens (VT).
uint24 stabilityMintFeeYT; // Stability minting fee for yield tokens (YT).
uint24 stabilityRedeemFeeVT; // Stability redemption fee for volatile tokens (VT).
uint24 stabilityRedeemFeeYT; // Stability redemption fee for yield tokens (YT).
uint24 yieldFeeVT; // Yield fee for volatile tokens (VT).
uint24 yieldFeeYT; // Yield fee for yield tokens (YT).
uint24 protocolFee; // Protocol fee.
}
// FeePermissions define the flexibility of the fee model (dynamic fees, delegation).
struct FeePermissions {
bool isDynamic; // Whether the fee model is dynamic.
bool allowDelegation; // Whether fee delegation is allowed.
}
struct GroupState {
DTokenRegistry.GroupCore core;
DTokenRegistry.GroupExtended extended;
bytes32 feesPacked;
CollateralInfo[] acceptableCollaterals;
Address hookContract;
bytes32 groupSettings;
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { DTokenRegistry } from "../declarations/DTokenRegistry.sol";
/**
* @title GroupKey
* @dev Struct representing the core group key information.
*/
struct GroupKey {
/**
* @dev The core group information from the DTokenRegistry.
*/
DTokenRegistry.GroupCore core;
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
/// @notice https://forum.openzeppelin.com/t/safeerc20-vs-safeerc20upgradeable/17326
import { SafeERC20Upgradeable, IERC20Upgradeable } from "@openzeppelin/contracts-upgradeable/token/ERC20/utils/SafeERC20Upgradeable.sol";
import { CustomRevert } from "../libs/CustomRevert.sol";
// Custom type `Currency` to represent either an ERC20 token or the native currency (e.g., ETH)
type Currency is address;
/**
* @title CurrencyLibrary
* @dev A library for handling operations related to currencies, supporting both ERC20 tokens and native currency.
* Provides utility functions for balance checks, transfers, approvals, and comparisons.
*/
library CurrencyLibrary {
// Use SafeERC20Upgradeable for IERC20Upgradeable token operations to ensure safety
using SafeERC20Upgradeable for IERC20Upgradeable;
// Use CustomRevert for standardized error handling via selectors
using CustomRevert for bytes4;
// Define a constant representing the native currency (address(0))
Currency public constant NATIVE = Currency.wrap(address(0));
// Custom error definitions for various invalid operations involving native currency
error NativeCurrencyTransfersNotAllowed();
error NativeCurrencyTransferFromNotAllowed();
error NativeCurrencyApprovalNotAllowed();
error NativeCurrencyDoesNotHaveTotalSupply();
error NativeCurrencyIncreaseAllowanceNotAllowed();
error NativeCurrencyDecreaseAllowanceNotAllowed();
error ArbitraryTransfersNotAllowed();
/**
* @notice Checks if two currencies are equal.
* @param currency The first currency to compare.
* @param other The second currency to compare.
* @return True if both currencies are the same, false otherwise.
*/
function equals(Currency currency, Currency other) internal pure returns (bool) {
return Currency.unwrap(currency) == Currency.unwrap(other);
}
/**
* @notice Retrieves the balance of the specified owner for the given currency.
* @param currency The currency to check (ERC20 token or native).
* @param owner The address of the owner whose balance is queried.
* @return The balance of the owner in the specified currency.
*/
function balanceOf(Currency currency, address owner) internal view returns (uint256) {
if (isNative(currency)) {
return owner.balance; // For native currency, return the ETH balance
} else {
return IERC20Upgradeable(Currency.unwrap(currency)).balanceOf(owner); // For ERC20 tokens, use balanceOf
}
}
/**
* @notice Safely transfers a specified amount of the currency to a recipient.
* @param currency The currency to transfer (must be an ERC20 token).
* @param to The recipient address.
* @param amount The amount to transfer.
* @dev Native currency transfers are not allowed and will revert.
*/
function safeTransfer(Currency currency, address to, uint256 amount) internal {
if (isNative(currency)) {
// Revert if attempting to transfer native currency using ERC20 methods
NativeCurrencyTransfersNotAllowed.selector.revertWith();
} else {
IERC20Upgradeable(Currency.unwrap(currency)).safeTransfer(to, amount);
}
}
/**
* @notice Safely transfers a specified amount of the currency from one address to another.
* @param currency The currency to transfer (must be an ERC20 token).
* @param safeFrom The address to transfer from.
* @param to The recipient address.
* @param amount The amount to transfer.
* @dev Native currency transfers are not allowed and will revert.
* @dev Arbitrary transfers (i.e., not initiated by the sender) are also not allowed and will revert.
* @notice Slither false positive. The function is internal and only used within the library
*/
//slither-disable-next-line arbitrary-send-erc20
function safeTransferFrom(Currency currency, address safeFrom, address to, uint256 amount) internal {
if (isNative(currency)) {
// Revert if attempting to transfer ERC20 tokens from an address other than the sender
// This is to prevent arbitrary transfers, which are not allowed in the context of this library
// This logic has the priority, so overrides any other inhereted logic
NativeCurrencyTransferFromNotAllowed.selector.revertWith();
} else {
if (safeFrom != msg.sender) ArbitraryTransfersNotAllowed.selector.revertWith();
IERC20Upgradeable(Currency.unwrap(currency)).safeTransferFrom(safeFrom, to, amount);
}
}
/**
* @notice Retrieves the allowance of a spender for the given owner's currency.
* @param currency The currency to check (must be an ERC20 token).
* @param owner The address of the owner of the currency.
* @param spender The address of the spender.
* @return The allowance of the spender for the owner's currency.
*/
function allowance(Currency currency, address owner, address spender) internal view returns (uint256) {
if (isNative(currency)) {
return 0; // For native currency, return 0 as allowance is not applicable
} else {
return IERC20Upgradeable(Currency.unwrap(currency)).allowance(owner, spender); // For ERC20 tokens, use allowance
}
}
/**
* @notice Safely approves a spender to spend a specified amount of the currency.
* @param currency The currency to approve (must be an ERC20 token).
* @param spender The address authorized to spend the tokens.
* @param amount The amount to approve.
* @dev Approving native currency is not allowed and will revert.
*/
function safeApprove(Currency currency, address spender, uint256 amount) internal {
if (!isNative(currency)) {
IERC20Upgradeable(Currency.unwrap(currency)).safeApprove(spender, amount);
} else {
// Revert if attempting to approve native currency
NativeCurrencyApprovalNotAllowed.selector.revertWith();
}
}
/**
* @notice Safely increases the allowance of a spender for the currency.
* @param currency The currency to modify allowance for (must be an ERC20 token).
* @param spender The address authorized to spend the tokens.
* @param addedValue The amount to increase the allowance by.
* @dev Increasing allowance for native currency is not allowed and will revert.
*/
function safeIncreaseAllowance(Currency currency, address spender, uint256 addedValue) internal {
if (!isNative(currency)) {
IERC20Upgradeable(Currency.unwrap(currency)).safeIncreaseAllowance(spender, addedValue);
} else {
// Revert if attempting to increase allowance for native currency
NativeCurrencyIncreaseAllowanceNotAllowed.selector.revertWith();
}
}
/**
* @notice Checks if the given currency is the native currency.
* @param currency The currency to check.
* @return True if `currency` is the native currency, false otherwise.
*/
function isNative(Currency currency) internal pure returns (bool) {
return Currency.unwrap(currency) == Currency.unwrap(NATIVE);
}
/**
* @notice Checks if the given currency is the zero address.
* @param currency The currency to check.
* @return True if `currency` is the zero address, false otherwise.
*/
function isZero(Currency currency) internal pure returns (bool) {
return Currency.unwrap(currency) == address(0);
}
/**
* @notice Converts the currency address to a unique identifier.
* @param currency The currency to convert.
* @return The uint256 representation of the currency's address.
*/
function toId(Currency currency) internal pure returns (uint256) {
return uint160(Currency.unwrap(currency));
}
/**
* @notice Unwraps the `Currency` type to retrieve the underlying address.
* @param currency The currency to unwrap.
* @return The underlying address of the currency.
*/
function toAddress(Currency currency) internal pure returns (address) {
return Currency.unwrap(currency);
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
library DProtocol {
struct MintParams {
uint8 operationType; // Type of mint operation
uint256 baseIn; // Amount of base token input
uint24 slippage; // The slippage tolerance
address paymentToken; // Token used for payment
bytes hookData; // Hook data if any
}
struct RedeemParams {
uint8 operationType; // Type of redeem operation
uint256 baseOut; // Amount of tokens to redeem
uint24 slippage; // The slippage tolerance
address desiredCollateral; // Desired collateral to receive
bytes hookData; // Hook data if any
}
struct MintResult {
uint256 ytMinted; // Amount of YT Tokens minted
uint256 vtMinted; // Amount of VT Tokens minted
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { Currency } from "../types/Currency.sol";
import { GroupId } from "../types/GroupId.sol";
import { Slot0 } from "../types/Slot0.sol";
import { DTokenRegistry } from "../declarations/DTokenRegistry.sol";
import { Address } from "../types/Address.sol";
import { GroupState, CollateralInfo } from "../types/CommonTypes.sol";
/**
* @title DOperationContext
* @notice Library containing the data structures used for operation contexts.
*/
library DOperationContext {
/**
* @notice Struct representing amounts used in the context.
*/
struct ContextAmounts {
uint256 amount0;
uint256 amount1;
}
/**
* @notice Struct representing a tight group state.
*/
struct TightGroupState {
DTokenRegistry.GroupCore core;
DTokenRegistry.GroupExtended extended;
bytes32 groupSettings;
bytes32 feesPacked;
Address hookContract;
}
/**
* @notice Struct containing parameters needed to create an operation context.
*/
struct Context {
uint8 operationType;
ContextAmounts amounts;
uint24 slippage;
address recipient;
CollateralInfo paymentToken;
GroupId groupId;
GroupState groupState;
bytes hookData;
}
/**
* @notice Struct representing parameters packed into Slot0 and Slot1 (memory bytes).
*/
struct Params {
Slot0 slot0;
bytes slot1;
}
/**
* @notice Struct representing the full operation context.
*/
struct OperationContext {
GroupId groupId;
Params params;
TightGroupState groupState;
uint256 amount0;
uint256 amount1;
uint24 slippage;
address recipient;
CollateralInfo paymentToken;
uint256 collateralRatio;
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { Currency } from "../types/Currency.sol";
import { GroupKey } from "../types/GroupKey.sol";
import { GroupId } from "../types/GroupId.sol";
import { DProtocol } from "../declarations/DProtocol.sol";
interface IProtocol {
/**
* @notice Emitted when ERC20 tokens are recovered.
* @param tokenAddress The address of the recovered token.
* @param amount The amount of tokens recovered.
*/
event ERC20Recovered(Currency indexed tokenAddress, uint256 indexed amount);
/**
* @notice Emitted when native tokens are withdrawn.
* @param amount The amount of native tokens withdrawn.
*/
event NativeTokenWithdrawn(uint256 indexed amount);
/**
* @notice Emitted when tokens are minted.
* @param groupId The group ID.
* @param token The token address.
* @param recipient The recipient address.
* @param sender The sender address.
* @param ytMinted The amount of aTokens minted.
* @param vtMinted The amount of vTokens minted.
*/
event MintToken(
GroupId indexed groupId,
address indexed token,
address indexed recipient,
address sender,
uint256 ytMinted,
uint256 vtMinted
);
/**
* @notice Emitted when tokens are redeemed.
* @param groupId The group ID.
* @param token The token address.
* @param recipient The recipient address.
* @param sender The sender address.
* @param baseOut The amount of base tokens received.
*/
event RedeemToken(
GroupId indexed groupId,
address indexed token,
address indexed recipient,
address sender,
uint256 baseOut
);
/**
* @notice Emitted when the minting is paused.
* @param groupId The group ID.
*/
event RedeemLockUpdated(GroupId indexed groupId, bool indexed enabled);
event FeeDelegated(address indexed hookContract, uint256 indexed protocolFeeAmount, uint256 indexed hookRemainingFee);
event FeeCollected(address indexed feeCollector, address indexed token, uint256 indexed amount);
/**
* @notice Emitted when fees are settled.
* @param hookContract The hook contract address.
* @param token The token address.
* @param amount The amount of fees settled.
*/
event FeesSettled(address indexed hookContract, address indexed token, uint256 indexed amount);
// Custom Errors
// todo: bytes32 groupId
error ZeroAddress();
error ZeroAmount();
error ErrorMinTokenAmount();
error NotPermitted();
error ZeroGroupId();
error ConfigNotReady();
error NoFeesOwed();
error InvalidOperationType();
error InvalidRatios();
error ErrorMintPaused();
error ErrorRedeemPaused();
error UnsupportedCollateralType();
error InvalidMinMaxCollateralAmount();
error InsufficientAllowance();
error InsufficientBalance(address token);
error ErrorInsufficientTokenOutput();
error ErrorInsufficientBaseOutput();
error InvalidSlippageRequested();
error TreasuryGroupUpdateFailed();
error InvalidMinimumAmount(uint256 minimumAmount, uint256 amountReceived);
error MaximumAmountExceeded(GroupId groupId, uint256 preparedAmount, uint256 maxAmount);
error InvalidPaymentAmount();
error TransferFailed(GroupId groupId, address recipient, uint256 amount);
error ErrorATokenMintingPausedInStabilityMode(GroupId groupId);
error RedeemLocked(GroupId groupId);
error EarlyExit(GroupId groupId);
error EmptyCollateralListOnUpdate(GroupId groupId);
error InvalidGroupConfiguration(GroupId groupId);
error NativeTokenWithdrawnFailed(uint256 amount);
/**
* @notice Mints tokens based on the provided parameters.
* @param groupKey The key identifying the group.
* @param params The mint parameters.
* @return result The result of the mint operation.
*/
function mintToken(
GroupKey calldata groupKey,
DProtocol.MintParams calldata params
) external payable returns (DProtocol.MintResult memory result);
/**
* @notice Redeems tokens based on the provided parameters.
* @param groupKey The key identifying the group.
* @param params The redeem parameters.
* @return baseOut The amount of base tokens received.
*/
function redeemToken(GroupKey calldata groupKey, DProtocol.RedeemParams calldata params) external returns (uint256 baseOut);
/**
* @notice Returns the stability ratio for a group.
* @param groupId The group identifier.
* @return The stability ratio.
*/
function stabilityRatio(GroupId groupId) external view returns (uint96);
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { GroupId } from "../types/GroupId.sol";
interface ITreasuryMinimum {
function isUnderCollateral(GroupId groupId) external view returns (bool);
function currentBaseTokenPrice(GroupId groupId) external view returns (uint256);
function totalBaseToken(GroupId groupId) external view returns (uint256);
function forceUpdateGroupCache(address tokenRegistry, GroupId groupId) external;
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { GroupId } from "../types/GroupId.sol";
import { DOperationContext } from "../declarations/DOperationContext.sol";
/**
* @title IHooks
* @notice Interface defining hooks that can be called during protocol operations
* @dev This interface must be implemented by any contract that wants to act as a hook
*/
interface IHooks {
/**
* @notice Returns the permissions of the hook contract
* @return A uint16 representing the permission flags
*/
function getHookPermissions() external view returns (uint16);
/**
* @notice Hook called before a mint operation
* @param sender The address initiating the operation
* @param groupId The group identifier
* @param context The operation context
* @param hookData The current hook data
* @return selector The function selector (should be `beforeMint.selector`)
* @return updatedHookData The updated hook data
* @return lpFeeOverride An optional LP fee override
*/
function beforeMint(
address sender,
GroupId groupId,
DOperationContext.OperationContext calldata context,
bytes calldata hookData
) external returns (bytes4 selector, bytes memory updatedHookData, uint24 lpFeeOverride);
/**
* @notice Hook called after a mint operation
* @param sender The address initiating the operation
* @param groupId The group identifier
* @param context The operation context
* @param hookData The current hook data
* @return selector The function selector (should be `afterMint.selector`)
* @return updatedHookData The updated hook data
*/
function afterMint(
address sender,
GroupId groupId,
DOperationContext.OperationContext calldata context,
bytes calldata hookData
) external returns (bytes4 selector, bytes memory updatedHookData);
/**
* @notice Hook called before a redeem operation
* @param sender The address initiating the operation
* @param groupId The group identifier
* @param context The operation context
* @param hookData The current hook data
* @return selector The function selector (should be `beforeRedeem.selector`)
* @return updatedHookData The updated hook data
* @return lpFeeOverride An optional LP fee override
*/
function beforeRedeem(
address sender,
GroupId groupId,
DOperationContext.OperationContext calldata context,
bytes calldata hookData
) external returns (bytes4 selector, bytes memory updatedHookData, uint24 lpFeeOverride);
/**
* @notice Hook called after a redeem operation
* @param sender The address initiating the operation
* @param groupId The group identifier
* @param context The operation context
* @param hookData The current hook data
* @return selector The function selector (should be `afterRedeem.selector`)
* @return updatedHookData The updated hook data
*/
function afterRedeem(
address sender,
GroupId groupId,
DOperationContext.OperationContext calldata context,
bytes calldata hookData
) external returns (bytes4 selector, bytes memory updatedHookData);
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (access/IAccessControl.sol)
pragma solidity ^0.8.0;
/**
* @dev External interface of AccessControl declared to support ERC165 detection.
*/
interface IAccessControlUpgradeable {
/**
* @dev Emitted when `newAdminRole` is set as ``role``'s admin role, replacing `previousAdminRole`
*
* `DEFAULT_ADMIN_ROLE` is the starting admin for all roles, despite
* {RoleAdminChanged} not being emitted signaling this.
*
* _Available since v3.1._
*/
event RoleAdminChanged(bytes32 indexed role, bytes32 indexed previousAdminRole, bytes32 indexed newAdminRole);
/**
* @dev Emitted when `account` is granted `role`.
*
* `sender` is the account that originated the contract call, an admin role
* bearer except when using {AccessControl-_setupRole}.
*/
event RoleGranted(bytes32 indexed role, address indexed account, address indexed sender);
/**
* @dev Emitted when `account` is revoked `role`.
*
* `sender` is the account that originated the contract call:
* - if using `revokeRole`, it is the admin role bearer
* - if using `renounceRole`, it is the role bearer (i.e. `account`)
*/
event RoleRevoked(bytes32 indexed role, address indexed account, address indexed sender);
/**
* @dev Returns `true` if `account` has been granted `role`.
*/
function hasRole(bytes32 role, address account) external view returns (bool);
/**
* @dev Returns the admin role that controls `role`. See {grantRole} and
* {revokeRole}.
*
* To change a role's admin, use {AccessControl-_setRoleAdmin}.
*/
function getRoleAdmin(bytes32 role) external view returns (bytes32);
/**
* @dev Grants `role` to `account`.
*
* If `account` had not been already granted `role`, emits a {RoleGranted}
* event.
*
* Requirements:
*
* - the caller must have ``role``'s admin role.
*/
function grantRole(bytes32 role, address account) external;
/**
* @dev Revokes `role` from `account`.
*
* If `account` had been granted `role`, emits a {RoleRevoked} event.
*
* Requirements:
*
* - the caller must have ``role``'s admin role.
*/
function revokeRole(bytes32 role, address account) external;
/**
* @dev Revokes `role` from the calling account.
*
* Roles are often managed via {grantRole} and {revokeRole}: this function's
* purpose is to provide a mechanism for accounts to lose their privileges
* if they are compromised (such as when a trusted device is misplaced).
*
* If the calling account had been granted `role`, emits a {RoleRevoked}
* event.
*
* Requirements:
*
* - the caller must be `account`.
*/
function renounceRole(bytes32 role, address account) external;
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.9.4) (utils/Context.sol)
pragma solidity ^0.8.0;
import {Initializable} from "../proxy/utils/Initializable.sol";
/**
* @dev Provides information about the current execution context, including the
* sender of the transaction and its data. While these are generally available
* via msg.sender and msg.data, they should not be accessed in such a direct
* manner, since when dealing with meta-transactions the account sending and
* paying for execution may not be the actual sender (as far as an application
* is concerned).
*
* This contract is only required for intermediate, library-like contracts.
*/
abstract contract ContextUpgradeable is Initializable {
function __Context_init() internal onlyInitializing {
}
function __Context_init_unchained() internal onlyInitializing {
}
function _msgSender() internal view virtual returns (address) {
return msg.sender;
}
function _msgData() internal view virtual returns (bytes calldata) {
return msg.data;
}
function _contextSuffixLength() internal view virtual returns (uint256) {
return 0;
}
/**
* @dev This empty reserved space is put in place to allow future versions to add new
* variables without shifting down storage in the inheritance chain.
* See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
*/
uint256[50] private __gap;
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.9.0) (utils/Strings.sol)
pragma solidity ^0.8.0;
import "./math/MathUpgradeable.sol";
import "./math/SignedMathUpgradeable.sol";
/**
* @dev String operations.
*/
library StringsUpgradeable {
bytes16 private constant _SYMBOLS = "0123456789abcdef";
uint8 private constant _ADDRESS_LENGTH = 20;
/**
* @dev Converts a `uint256` to its ASCII `string` decimal representation.
*/
function toString(uint256 value) internal pure returns (string memory) {
unchecked {
uint256 length = MathUpgradeable.log10(value) + 1;
string memory buffer = new string(length);
uint256 ptr;
/// @solidity memory-safe-assembly
assembly {
ptr := add(buffer, add(32, length))
}
while (true) {
ptr--;
/// @solidity memory-safe-assembly
assembly {
mstore8(ptr, byte(mod(value, 10), _SYMBOLS))
}
value /= 10;
if (value == 0) break;
}
return buffer;
}
}
/**
* @dev Converts a `int256` to its ASCII `string` decimal representation.
*/
function toString(int256 value) internal pure returns (string memory) {
return string(abi.encodePacked(value < 0 ? "-" : "", toString(SignedMathUpgradeable.abs(value))));
}
/**
* @dev Converts a `uint256` to its ASCII `string` hexadecimal representation.
*/
function toHexString(uint256 value) internal pure returns (string memory) {
unchecked {
return toHexString(value, MathUpgradeable.log256(value) + 1);
}
}
/**
* @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length.
*/
function toHexString(uint256 value, uint256 length) internal pure returns (string memory) {
bytes memory buffer = new bytes(2 * length + 2);
buffer[0] = "0";
buffer[1] = "x";
for (uint256 i = 2 * length + 1; i > 1; --i) {
buffer[i] = _SYMBOLS[value & 0xf];
value >>= 4;
}
require(value == 0, "Strings: hex length insufficient");
return string(buffer);
}
/**
* @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal representation.
*/
function toHexString(address addr) internal pure returns (string memory) {
return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH);
}
/**
* @dev Returns true if the two strings are equal.
*/
function equal(string memory a, string memory b) internal pure returns (bool) {
return keccak256(bytes(a)) == keccak256(bytes(b));
}
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (utils/introspection/ERC165.sol)
pragma solidity ^0.8.0;
import "./IERC165Upgradeable.sol";
import {Initializable} from "../../proxy/utils/Initializable.sol";
/**
* @dev Implementation of the {IERC165} interface.
*
* Contracts that want to implement ERC165 should inherit from this contract and override {supportsInterface} to check
* for the additional interface id that will be supported. For example:
*
* ```solidity
* function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
* return interfaceId == type(MyInterface).interfaceId || super.supportsInterface(interfaceId);
* }
* ```
*
* Alternatively, {ERC165Storage} provides an easier to use but more expensive implementation.
*/
abstract contract ERC165Upgradeable is Initializable, IERC165Upgradeable {
function __ERC165_init() internal onlyInitializing {
}
function __ERC165_init_unchained() internal onlyInitializing {
}
/**
* @dev See {IERC165-supportsInterface}.
*/
function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
return interfaceId == type(IERC165Upgradeable).interfaceId;
}
/**
* @dev This empty reserved space is put in place to allow future versions to add new
* variables without shifting down storage in the inheritance chain.
* See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
*/
uint256[50] private __gap;
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.9.0) (proxy/utils/Initializable.sol)
pragma solidity ^0.8.2;
import "../../utils/AddressUpgradeable.sol";
/**
* @dev This is a base contract to aid in writing upgradeable contracts, or any kind of contract that will be deployed
* behind a proxy. Since proxied contracts do not make use of a constructor, it's common to move constructor logic to an
* external initializer function, usually called `initialize`. It then becomes necessary to protect this initializer
* function so it can only be called once. The {initializer} modifier provided by this contract will have this effect.
*
* The initialization functions use a version number. Once a version number is used, it is consumed and cannot be
* reused. This mechanism prevents re-execution of each "step" but allows the creation of new initialization steps in
* case an upgrade adds a module that needs to be initialized.
*
* For example:
*
* [.hljs-theme-light.nopadding]
* ```solidity
* contract MyToken is ERC20Upgradeable {
* function initialize() initializer public {
* __ERC20_init("MyToken", "MTK");
* }
* }
*
* contract MyTokenV2 is MyToken, ERC20PermitUpgradeable {
* function initializeV2() reinitializer(2) public {
* __ERC20Permit_init("MyToken");
* }
* }
* ```
*
* TIP: To avoid leaving the proxy in an uninitialized state, the initializer function should be called as early as
* possible by providing the encoded function call as the `_data` argument to {ERC1967Proxy-constructor}.
*
* CAUTION: When used with inheritance, manual care must be taken to not invoke a parent initializer twice, or to ensure
* that all initializers are idempotent. This is not verified automatically as constructors are by Solidity.
*
* [CAUTION]
* ====
* Avoid leaving a contract uninitialized.
*
* An uninitialized contract can be taken over by an attacker. This applies to both a proxy and its implementation
* contract, which may impact the proxy. To prevent the implementation contract from being used, you should invoke
* the {_disableInitializers} function in the constructor to automatically lock it when it is deployed:
*
* [.hljs-theme-light.nopadding]
* ```
* /// @custom:oz-upgrades-unsafe-allow constructor
* constructor() {
* _disableInitializers();
* }
* ```
* ====
*/
abstract contract Initializable {
/**
* @dev Indicates that the contract has been initialized.
* @custom:oz-retyped-from bool
*/
uint8 private _initialized;
/**
* @dev Indicates that the contract is in the process of being initialized.
*/
bool private _initializing;
/**
* @dev Triggered when the contract has been initialized or reinitialized.
*/
event Initialized(uint8 version);
/**
* @dev A modifier that defines a protected initializer function that can be invoked at most once. In its scope,
* `onlyInitializing` functions can be used to initialize parent contracts.
*
* Similar to `reinitializer(1)`, except that functions marked with `initializer` can be nested in the context of a
* constructor.
*
* Emits an {Initialized} event.
*/
modifier initializer() {
bool isTopLevelCall = !_initializing;
require(
(isTopLevelCall && _initialized < 1) || (!AddressUpgradeable.isContract(address(this)) && _initialized == 1),
"Initializable: contract is already initialized"
);
_initialized = 1;
if (isTopLevelCall) {
_initializing = true;
}
_;
if (isTopLevelCall) {
_initializing = false;
emit Initialized(1);
}
}
/**
* @dev A modifier that defines a protected reinitializer function that can be invoked at most once, and only if the
* contract hasn't been initialized to a greater version before. In its scope, `onlyInitializing` functions can be
* used to initialize parent contracts.
*
* A reinitializer may be used after the original initialization step. This is essential to configure modules that
* are added through upgrades and that require initialization.
*
* When `version` is 1, this modifier is similar to `initializer`, except that functions marked with `reinitializer`
* cannot be nested. If one is invoked in the context of another, execution will revert.
*
* Note that versions can jump in increments greater than 1; this implies that if multiple reinitializers coexist in
* a contract, executing them in the right order is up to the developer or operator.
*
* WARNING: setting the version to 255 will prevent any future reinitialization.
*
* Emits an {Initialized} event.
*/
modifier reinitializer(uint8 version) {
require(!_initializing && _initialized < version, "Initializable: contract is already initialized");
_initialized = version;
_initializing = true;
_;
_initializing = false;
emit Initialized(version);
}
/**
* @dev Modifier to protect an initialization function so that it can only be invoked by functions with the
* {initializer} and {reinitializer} modifiers, directly or indirectly.
*/
modifier onlyInitializing() {
require(_initializing, "Initializable: contract is not initializing");
_;
}
/**
* @dev Locks the contract, preventing any future reinitialization. This cannot be part of an initializer call.
* Calling this in the constructor of a contract will prevent that contract from being initialized or reinitialized
* to any version. It is recommended to use this to lock implementation contracts that are designed to be called
* through proxies.
*
* Emits an {Initialized} event the first time it is successfully executed.
*/
function _disableInitializers() internal virtual {
require(!_initializing, "Initializable: contract is initializing");
if (_initialized != type(uint8).max) {
_initialized = type(uint8).max;
emit Initialized(type(uint8).max);
}
}
/**
* @dev Returns the highest version that has been initialized. See {reinitializer}.
*/
function _getInitializedVersion() internal view returns (uint8) {
return _initialized;
}
/**
* @dev Returns `true` if the contract is currently initializing. See {onlyInitializing}.
*/
function _isInitializing() internal view returns (bool) {
return _initializing;
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
// solhint-disable no-inline-assembly
/// @dev A subset copied from the following contracts:
///
/// + `balancer-labs/v2-solidity-utils/contracts/helpers/WordCodec.sol`
/// + `balancer-labs/v2-solidity-utils/contracts/helpers/WordCodecHelpers.sol`
library WordCodec {
/// @dev Inserts an unsigned integer of bitLength, shifted by an offset, into a 256 bit word,
/// replacing the old value. Returns the new word.
function insertUint(bytes32 word, uint256 value, uint256 offset, uint256 bitLength) internal pure returns (bytes32 result) {
// Equivalent to:
// uint256 mask = (1 << bitLength) - 1;
// bytes32 clearedWord = bytes32(uint256(word) & ~(mask << offset));
// result = clearedWord | bytes32(value << offset);
assembly("memory-safe") {
let mask := sub(shl(bitLength, 1), 1)
let clearedWord := and(word, not(shl(offset, mask)))
result := or(clearedWord, shl(offset, value))
}
}
/// @dev Decodes and returns an unsigned integer with `bitLength` bits, shifted by an offset, from a 256 bit word.
function decodeUint(bytes32 word, uint256 offset, uint256 bitLength) internal pure returns (uint256 result) {
// Equivalent to:
// result = uint256(word >> offset) & ((1 << bitLength) - 1);
assembly("memory-safe") {
result := and(shr(offset, word), sub(shl(bitLength, 1), 1))
}
}
/// @dev Inserts a signed integer shifted by an offset into a 256 bit word, replacing the old value. Returns
/// the new word.
///
function insertInt(bytes32 word, int256 value, uint256 offset, uint256 bitLength) internal pure returns (bytes32) {
unchecked {
uint256 mask = (1 << bitLength) - 1;
bytes32 clearedWord = bytes32(uint256(word) & ~(mask << offset));
// Integer values need masking to remove the upper bits of negative values.
return clearedWord | bytes32((uint256(value) & mask) << offset);
}
}
/// @dev Decodes and returns a signed integer with `bitLength` bits, shifted by an offset, from a 256 bit word.
function decodeInt(bytes32 word, uint256 offset, uint256 bitLength) internal pure returns (int256 result) {
unchecked {
int256 maxInt = int256((1 << (bitLength - 1)) - 1);
uint256 mask = (1 << bitLength) - 1;
int256 value = int256(uint256(word >> offset) & mask);
// In case the decoded value is greater than the max positive integer that can be represented with bitLength
// bits, we know it was originally a negative integer. Therefore, we mask it to restore the sign in the 256 bit
// representation.
//
// Equivalent to:
// result = value > maxInt ? (value | int256(~mask)) : value;
assembly {
result := or(mul(gt(value, maxInt), not(mask)), value)
}
}
}
/// @dev Decodes and returns a boolean shifted by an offset from a 256 bit word.
function decodeBool(bytes32 word, uint256 offset) internal pure returns (bool result) {
// Equivalent to:
// result = (uint256(word >> offset) & 1) == 1;
assembly {
result := and(shr(offset, word), 1)
}
}
/// @dev Inserts a boolean value shifted by an offset into a 256 bit word, replacing the old value. Returns the new
/// word.
function insertBool(bytes32 word, bool value, uint256 offset) internal pure returns (bytes32 result) {
// Equivalent to:
// bytes32 clearedWord = bytes32(uint256(word) & ~(1 << offset));
// bytes32 referenceInsertBool = clearedWord | bytes32(uint256(value ? 1 : 0) << offset);
assembly("memory-safe") {
let clearedWord := and(word, not(shl(offset, 1)))
result := or(clearedWord, shl(offset, value))
}
}
function clearWordAtPosition(bytes32 word, uint256 offset, uint256 bitLength) internal pure returns (bytes32 clearedWord) {
unchecked {
uint256 mask = (1 << bitLength) - 1;
clearedWord = bytes32(uint256(word) & ~(mask << offset));
}
}
/// @dev Encodes an address into a 256 bit word at a given offset.
function insertAddress(bytes32 word, address value, uint256 offset) internal pure returns (bytes32 result) {
assembly("memory-safe") {
let clearedWord := and(word, not(shl(offset, 0xffffffffffffffffffffffffffffffffffffffff)))
result := or(clearedWord, shl(offset, value))
}
}
/// @dev Decodes an address from a 256 bit word at a given offset.
function decodeAddress(bytes32 word, uint256 offset) internal pure returns (address result) {
assembly("memory-safe") {
result := and(shr(offset, word), 0xffffffffffffffffffffffffffffffffffffffff)
}
}
/// @dev Encodes an enum value into a 256 bit word at a given offset.
function insertEnum(bytes32 word, uint8 value, uint256 offset) internal pure returns (bytes32 result) {
assembly("memory-safe") {
let clearedWord := and(word, not(shl(offset, 0xff)))
result := or(clearedWord, shl(offset, value))
}
}
/// @dev Decodes an enum value from a 256 bit word at a given offset.
function decodeEnum(bytes32 word, uint256 offset) internal pure returns (uint8 result) {
assembly("memory-safe") {
result := and(shr(offset, word), 0xff)
}
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
// Import the CustomRevert library for standardized error handling
import { CustomRevert } from "../libs/CustomRevert.sol";
/**
* @title Address
* @dev Defines a user-defined value type `Address` that wraps the built-in `address` type.
*/
type Address is address;
/**
* @title AddressLibrary
* @dev A library for performing various operations on the `Address` type.
*/
library AddressLibrary {
// Apply the library functions to the `Address` type
using AddressLibrary for Address;
// Use the CustomRevert library for standardized error handling via selectors
using CustomRevert for bytes4;
// Custom error definitions for more descriptive revert reasons
error ZeroAddress();
error FailedCall(string reason);
error NonContractAddress();
error UnableToSendValue();
/**
* @notice Checks if the given `Address` is the zero address.
* @param addr The `Address` to check.
* @return True if `addr` is the zero address, false otherwise.
*/
function isZero(Address addr) internal pure returns (bool) {
return Address.unwrap(addr) == address(0);
}
/**
* @notice Determines if the given `Address` is a contract.
* @param addr The `Address` to check.
* @return True if `addr` is a contract, false otherwise.
*
* @dev This method relies on the fact that contracts have non-zero code size.
* It returns false for contracts in construction, since the code is only stored at the end of the constructor execution.
*/
function isContract(Address addr) internal view returns (bool) {
return Address.unwrap(addr).code.length > 0;
}
/**
* @notice Compares two `Address` instances for equality.
* @param a The first `Address`.
* @param b The second `Address`.
* @return True if both addresses are equal, false otherwise.
*/
function equals(Address a, Address b) internal pure returns (bool) {
address addrA = Address.unwrap(a);
address addrB = Address.unwrap(b);
return addrA == addrB;
}
/**
* @notice Converts an `Address` to a `uint160`.
* @param addr The `Address` to convert.
* @return The `uint160` representation of the address.
*/
function toUint160(Address addr) internal pure returns (uint160) {
return uint160(Address.unwrap(addr));
}
/**
* @notice Creates an `Address` from a `uint160`.
* @param addr The `uint160` value to convert.
* @return A new `Address` instance.
*/
function fromUint160(uint160 addr) internal pure returns (Address) {
return Address.wrap(address(addr));
}
/**
* @notice Unwraps the `Address` type to retrieve the underlying address.
* @param addr The `Address` to unwrap.
* @return The underlying `address`.
*/
function toAddress(Address addr) internal pure returns (address) {
return Address.unwrap(addr);
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { GroupId } from "../types/GroupId.sol";
import { GroupState } from "../types/CommonTypes.sol";
interface ITokenRegistryMinimum {
// Getter functions for group state
function getGroup(GroupId groupId) external view returns (GroupState memory);
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { Currency } from "../types/Currency.sol";
import { Address } from "../types/Address.sol";
import { DefaultFeeParams, FeeParams, FeePermissions, CollateralInfo } from "../types/CommonTypes.sol";
library DTokenRegistry {
// Core tokens of the group (immutable)
struct GroupCore {
Currency aToken; // Yield-bearing token
Currency xToken; // Leverage token
Currency baseToken; // Base token for the group - wrapped Avax
Currency yieldBearingToken; // Example: staked AVAX (immutable)
Currency wethToken; // WETH token address for the router
}
// Decimals for each core token
struct GroupDecimals {
uint8 aTokenDecimals;
uint8 xTokenDecimals;
uint8 baseTokenDecimals;
uint8 yieldBearingTokenDecimals;
}
// Extended group settings (mutable)
struct GroupExtended {
Address priceOracle; // Price Oracle address
Address rateProvider; // Rate provider address
Address swapRouter; // Swap Router
Address treasury; // Treasury address
Address feeCollector; // Fee collector address
Address strategy; // Strategy contract address
Currency rebalancePool; // Rebalance pool address
}
// Metadata for the group (partially mutable)
struct GroupMeta {
uint96 stabilityRatio; // Mutable stability ratio (this is 2^96-1 and we have max 5e18)
uint96 stabilityConditionsTriggeringRate; // Mutable stability fee trigger (this is 2^96-1 and we have max 5e18)
uint8 feeModel; // Immutable fee model used for this group
bool isWrappingRequired; // Immutable wrapping requirement flag
}
// Struct representing full group setup during creation
struct GroupSetup {
GroupCore core;
GroupDecimals decimals;
GroupExtended extended;
GroupMeta meta;
FeeParams fees;
DefaultFeeParams defaultFees;
FeePermissions feePermissions;
CollateralInfo[] acceptableCollaterals;
}
// Data used for updating mutable parts of the group
struct GroupUpdate {
GroupExtended extended;
GroupMeta meta;
CollateralInfo[] acceptableCollaterals;
FeeParams feeParams;
DefaultFeeParams defaultFees;
}
}// SPDX-License-Identifier: MI
pragma solidity 0.8.28;
import { CustomRevert } from "./CustomRevert.sol";
import { FeePermissions, DefaultFeeParams } from "../types/CommonTypes.sol";
/**
* @title GroupFeeLibrary
* @dev Provides utilities to manage fees for groups with their own fee structures and permissions.
* This library handles fee validation, processing, and manipulation based on defined flags and permissions.
*/
library GroupFeeLibrary {
// Utilize the CustomRevert library for standardized error handling via selectors.
using CustomRevert for bytes4;
/**
* @dev Thrown when a fee exceeds the allowed maximum.
* @param fee The fee value that is too large.
*/
error InvalidGroupFee(uint24 fee);
/**
* @dev Thrown when fee delegation is attempted without permission.
*/
error FeeDelegationNotAllowed();
error BaseFeeExceedslimit();
error MinFeeExceedsBaseFee();
error MaxFeeIsLessThanBaseFee();
error MaxFeeExceedsLimit();
/**
* @dev Thrown when a dynamic fee operation is called but not allowed.
*/
error NotDynamicFeeGroup();
// ====================
// ==== Constants ====
// ====================
/**
* @dev Flag indicating dynamic fees. Embedded in the fee's uint24 value.
* Example: fee | 0x800000
*/
uint24 public constant DYNAMIC_FEE_FLAG = 0x800000;
/**
* @dev Flag for overriding fees. Embedded in the fee's uint24 value.
* Example: fee |0x400000
*/
uint24 public constant OVERRIDE_FEE_FLAG = 0x400000;
/**
* @dev Flag allowing fee delegation. Embedded in the fee's uint24 value.
* Example: fee | 0x200000
*/
uint24 public constant DELEGATE_FEE_FLAG = 0x200000;
/**
* @dev Flag for explicit zero fee. Embedded in the fee's uint24 value.
* Example: fee |0x100000
* @notice Group min Fee settings has the priority over this flag, so it's impossible where minFee not equal to zero
*/
uint24 public constant ZERO_FEE_FLAG = 0x100000;
/**
* @dev Mask to remove all flags from the fee's uint24 value.
* Example: fee |0x1FFFFF
* All flags (0x800000, 0x400000, 0x200000, 0x100000) are in the upper 4 bits
* limiting the mask to 0x0FFFFF, all these flags are cleared.
*/
uint24 public constant REMOVE_ALL_FLAGS_MASK = 0x0FFFFF;
/**
* @dev Maximum allowed group fee (10,000 BPS). 50% expressed in basis points.
*/
uint24 public constant MAX_GROUP_FEE = 5_000;
// ====================
// ==== Internal Functions ====
// ====================
/**
* @notice Validates whether the given fee is within acceptable boundaries.
* @param self The DefaultFeeParams struct containing fee parameters.
* @param fee The fee value to validate.
* @return True if the fee is valid, false otherwise.
*
* @dev The cleaned fee (without flags) must be within the min/max fee limits and not exceed MAX_GROUP_FEE.
*/
function isValid(DefaultFeeParams memory self, uint24 fee) internal pure returns (bool) {
// Clean the fee by removing any flags (since flags do not affect fee validation).
uint24 cleanFee = fee & REMOVE_ALL_FLAGS_MASK;
// The cleaned fee must be within the min/max fee limits and not exceed MAX_GROUP_FEE.
/// @dev The cleaned fee must be within the min/max fee limits and not exceed MAX_GROUP_FEE, eventhough hook sent with ZERO_FEE_FLAG
return cleanFee >= self.minFee && cleanFee <= self.maxFee && cleanFee <= MAX_GROUP_FEE;
}
/**
* @notice Checks if the fee has the dynamic fee flag set.
* @param fee The fee value to check.
* @return True if the dynamic fee flag is set, false otherwise.
*/
function isDynamicFee(uint24 fee) internal pure returns (bool) {
return fee & DYNAMIC_FEE_FLAG != 0;
}
/**
* @notice Checks if the fee has an override flag.
* @param fee The fee value to check.
* @return True if the override flag is set, false otherwise.
*/
function isOverride(uint24 fee) internal pure returns (bool) {
return fee & OVERRIDE_FEE_FLAG != 0;
}
/**
* @notice Checks if the fee has the delegation flag set.
* @param fee The fee value to check.
* @return True if the delegation flag is set, false otherwise.
*/
function isDelegate(uint24 fee) internal pure returns (bool) {
return fee & DELEGATE_FEE_FLAG != 0;
}
/**
* @notice Removes all flags from the fee (i.e., cleans it to its base value).
* @param fee The fee value to clean.
* @return cleanedFee The fee value without any flags.
*/
function _removeAllFlags(uint24 fee) internal pure returns (uint24 cleanedFee) {
return fee & REMOVE_ALL_FLAGS_MASK;
}
/**
* @notice Validates the group fee based on default fee parameters and permissions.
* @param self The DefaultFeeParams struct containing fee parameters.
* @param permissions The FeePermissions struct containing permission flags.
* @param checkingFee The fee value to validate.
* @return The validated fee.
*
* @dev Ensures the fee is within allowed bounds and adheres to permissions.
*/
function _validateGroupFee(
DefaultFeeParams memory self,
FeePermissions memory permissions,
uint24 checkingFee
) internal pure returns (uint24) {
// Ensure fee delegation is allowed.
if (!permissions.allowDelegation && isDelegate(checkingFee)) {
FeeDelegationNotAllowed.selector.revertWith();
}
if (!permissions.isDynamic && isDynamicFee(checkingFee)) {
NotDynamicFeeGroup.selector.revertWith();
}
// Validate fee bounds
bool validated = isValid(self, checkingFee);
if (!validated) {
InvalidGroupFee.selector.revertWith(checkingFee);
}
return checkingFee;
}
/**
* @notice Processes the group fee, applying validations and possible overrides.
* @param self The DefaultFeeParams struct containing fee parameters.
* @param permissions The FeePermissions struct containing permission flags.
* @param overrideFee The fee value that may override the default base fee.
* @return cleanFee The processed and validated fee.
*/
function processGroupFee(
DefaultFeeParams memory self,
FeePermissions memory permissions,
uint24 overrideFee
) internal pure returns (uint24 cleanFee) {
cleanFee = _validateGroupFee(self, permissions, overrideFee);
}
/**
* @notice Validates the fee parameters to ensure they are logically correct.
* @param params The DefaultFeeParams struct containing fee parameters.
*
* @dev Ensures that:
* - Base fee does not exceed MAX_GROUP_FEE.
* - Minimum fee is not greater than the base fee.
* - Maximum fee is not less than the base fee.
* - Maximum fee does not exceed MAX_GROUP_FEE.
*/
function _validateFeeParams(DefaultFeeParams memory params) internal pure {
// Base fee should not exceed the maximum group fee.
if (params.baseFee > MAX_GROUP_FEE) {
BaseFeeExceedslimit.selector.revertWith();
}
// Min fee must be less than or equal to base fee.
if (params.minFee > params.baseFee) {
MinFeeExceedsBaseFee.selector.revertWith();
}
// Max fee must be greater than or equal to base fee.
if (params.maxFee < params.baseFee) {
MaxFeeIsLessThanBaseFee.selector.revertWith();
}
// Max fee should not exceed the maximum allowed group fee.
if (params.maxFee > MAX_GROUP_FEE) {
MaxFeeExceedsLimit.selector.revertWith();
}
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { GroupId } from "../types/GroupId.sol";
import { DTokenRegistry } from "../declarations/DTokenRegistry.sol";
import { DTreasury } from "../declarations/DTreasury.sol";
import { TreasuryHarvestLibrary } from "../libs/TreasuryHarvestLibrary.sol";
interface ITreasury {
// Events
event HarvestYield(GroupId indexed groupId, uint256 indexed yieldBaseTokens);
event HarvestFees(GroupId indexed groupId, uint256 indexed xTokenAmount, uint256 indexed aTokenAmount);
event FeesDistributed(GroupId indexed groupId, uint256 indexed xTokenAmount, uint256 indexed baseTokenAmount);
event UpdateBaseTokenCap(GroupId indexed groupId, uint256 indexed oldBaseTokenCap, uint256 indexed newBaseTokenCap);
event Settle(GroupId indexed groupId, uint256 indexed oldPrice, uint256 indexed newPrice);
event UpdateFeeCollector(address indexed oldFeeCollector, address indexed newFeeCollector);
event GroupInitialized(GroupId indexed groupId, uint256 indexed baseTokenPrice);
event RebalanceUpPerformed(GroupId indexed groupId, uint256 indexed newCollateralRatio);
event RebalanceDownPerformed(GroupId indexed groupId, uint256 indexed newCollateralRatio);
event TransferToStrategy(GroupId indexed groupId, uint256 indexed amount);
// Custom Errors
error ZeroAddress();
error ZeroAmount();
error OnlyStrategy();
error GroupAlreadyInitialized(GroupId groupId);
error ErrorUnderCollateral(GroupId groupId);
error ErrorExceedTotalCap(GroupId groupId);
error ErrorInvalidTwapPrice(GroupId groupId);
error NotPermitted();
error InvalidGroupConfiguration(GroupId groupId);
error InvalidRate(GroupId groupId, address rateProvider);
error InvalidRatio(GroupId groupId);
error InvalidPrice(GroupId groupId);
error InvalidBaseTokenCap(GroupId groupId);
error ErrorInsufficientBalance(GroupId groupId, address token);
error ErrorSwapFailed();
error ErrorDistributingYield(GroupId groupId);
error ErrorWithdrawFromStrategy();
error ErrorCollateralRatioTooSmall(GroupId groupId);
error MaximumAmountExceeded(GroupId groupId, uint256 maxAmount);
error StrategyUnderflow(GroupId groupId);
error InvalidTokenType();
error RenounceRoleProhibited();
// View Functions
function collateralRatio(GroupId groupId) external view returns (uint256);
function isUnderCollateral(GroupId groupId) external view returns (bool);
function leverageRatio(GroupId groupId) external view returns (uint256);
function currentBaseTokenPrice(GroupId groupId) external view returns (uint256);
// function isBaseTokenPriceValid(GroupId groupId) external view returns (bool _isValid);
function totalBaseToken(GroupId groupId) external view returns (uint256);
function maxMintableAToken(
GroupId groupId,
uint256 _newCollateralRatio
) external view returns (uint256 _maxBaseIn, uint256 _maxATokenMintable);
function maxRedeemableAToken(
GroupId groupId,
uint256 _newCollateralRatio
) external view returns (uint256 _maxBaseOut, uint256 _maxATokenRedeemable);
// State-Changing Functions
function mintAToken(GroupId groupId, uint256 _baseIn, address _recipient) external returns (uint256 _aTokenOut);
function mintXToken(GroupId groupId, uint256 _baseIn, address _recipient) external returns (uint256 _xTokenOut);
function redeem(GroupId groupId, uint256 _aTokenIn, uint256 _xTokenIn, address _owner) external returns (uint256 _baseOut);
function transferToStrategy(GroupId groupId, uint256 _amount) external;
// Harvest Functions
function harvestYield(GroupId groupId, TreasuryHarvestLibrary.HarvestParams memory params) external;
function harvestFees(GroupId groupId, TreasuryHarvestLibrary.HarvestParams memory params) external;
// Management Functions
function updateBaseTokenCap(GroupId groupId, uint256 _baseTokenCap) external;
function updateFeeCollector(address _newFeeCollector) external;
function forceUpdateGroupCache(address tokenRegistry, GroupId groupId) external;
function getTreasuryState(GroupId groupId) external view returns (DTreasury.TreasuryState memory);
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { IERC20Upgradeable } from "@openzeppelin/contracts-upgradeable/token/ERC20/utils/SafeERC20Upgradeable.sol";
import { GroupId } from "../types/GroupId.sol";
/**
* @title IWToken
* @notice Interface for the WToken contract, enabling wrapping and unwrapping of an underlying token.
*/
interface IWToken is IERC20Upgradeable {
/**
* @dev Emitted when tokens are wrapped into WTokens.
* @param user The address of the user wrapping tokens.
* @param underlyingAmount The amount of underlying tokens wrapped.
* @param amount The amount of tokens wrapped.
*/
event TokenWrapped(address indexed user, uint256 indexed underlyingAmount, uint256 indexed amount);
/**
* @dev Emitted when WTokens are unwrapped into underlying tokens.
* @param user The address of the user unwrapping tokens.
* @param underlyingAmount The amount of underlying tokens unwrapped.
* @param amount The amount of WTokens unwrapped.
*/
event TokenUnwrapped(address indexed user, uint256 indexed underlyingAmount, uint256 indexed amount);
/**
* @dev Emitted when the treasury address is updated.
* @param oldTreasury The address of the previous treasury contract.
* @param newTreasury The address of the new treasury contract.
*/
event UpdateTreasuryAddress(address indexed oldTreasury, address indexed newTreasury);
/**
* @dev Emitted when the associated group ID is updated.
* @param groupId New group identifier.
*/
event UpdateGroup(GroupId indexed groupId);
event Rebase(uint256 indexed epoch, uint256 indexed newScalar);
event RateProviderUpdated(address indexed rateProvider);
// Custom Errors
error ErrorZeroAddress();
error InvalidDecimals();
error ErrorZeroAmount();
error ErrorNotPermitted();
error ErrorCannotRecoverToken();
error ErrorMaxUnderlyingExceeded();
/**
* @notice Wraps underlying tokens into WTokens.
* @param underlyingAmount The amount of underlying tokens to wrap.
* @return wrappedAmount amount of WTokens minted.
*/
function wrap(uint256 underlyingAmount) external returns (uint256 wrappedAmount);
/**
* @notice Unwraps WTokens back into underlying tokens.
* @param burnAmount The amount of WTokens to burn.
* @return amount of underlying tokens returned.
*/
function unwrap(uint256 burnAmount) external returns (uint256 amount);
/**
* @notice rebase the WToken
*/
function rebase() external;
function updateRateProvider(address _rateProvider) external;
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
interface IDEXRouter {
function swapExactTokensForTokens(
uint256 amountIn,
uint256 amountOutMin,
address[] calldata path,
address to,
uint256 deadline
) external returns (uint256[] memory amounts);
function swapExactETHForTokens(
uint256 amountOutMin,
address[] calldata path,
address to,
uint256 deadline
) external payable returns (uint256[] memory amounts);
function swapExactTokensForETH(
uint256 amountIn,
uint256 amountOutMin,
address[] calldata path,
address to,
uint256 deadline
) external returns (uint256[] memory amounts);
// function getAmountsOut(
// uint256 amountIn,
// address[] calldata path
// ) external view returns (uint256[] memory amounts);
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { IERC4626Upgradeable } from "@openzeppelin/contracts-upgradeable/token/ERC20/extensions/ERC4626Upgradeable.sol";
/**
* @title IRebalancePool
* @notice Interface for the RebalancePool contract implementing ERC4626
*/
interface IRebalancePool is IERC4626Upgradeable {
/// @notice Errors
error ZeroAddress();
error ZeroDeposit();
error InvalidDepositAmount();
error ZeroRedeem();
error TokenAlreadyAdded(address token);
error ZeroShares();
error InvalidPriceFromOracle();
error NotPermitted();
error InvalidYieldFee();
error RedeemingNotAvailableYet();
error CannotRecoverToken();
error UpdatingNotAvailableYet();
error InvalidCoolingPeriod();
error CoolingPeriodTriggered();
/// @notice Events
event YieldDistributed(uint256 indexed assets, uint256 indexed shares);
event NAVUpdated(address indexed caller, uint256 indexed totalAssets, uint256 indexed totalSupply);
event CoolingOffPeriodUpdated(uint256 indexed oldPeriod, uint256 indexed newPeriod);
event FeeCollectorUpdated(address indexed oldCollector, address indexed newCollector);
event PriceOracleUpdated(address indexed oldOracle, address indexed newOracle);
event OtherERC20Withdrawn(address indexed receiver, address indexed token, uint256 indexed amount);
event DepositMade(address indexed depositor, address indexed receiver, uint256 indexed amount);
event YieldFeeUpdated(uint256 indexed newFee);
/// @notice Function to update NAV externally
function updateNAV() external;
/// @notice Get NAV per share
function getNavPerShare() external view returns (uint256);
/// @notice Override decimals
function decimals() external view override returns (uint8);
/// @notice Get cooling off period
function coolingOffPeriod() external view returns (uint256);
/// @notice Transfer tokens to treasury
function transferTokenToTreasury(address token, uint256 amount) external;
/// @notice Get yieldFeePercentage
function yieldFeePercentage() external view returns (uint256);
/// @notice Get base points
function BASE_POINTS() external view returns (uint256);
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { ITreasury } from "../interfaces/ITreasury.sol";
import { OperationContextLibrary } from "./OperationContextLibrary.sol";
import { DOperationContext } from "../declarations/DOperationContext.sol";
import { GroupId } from "../types/GroupId.sol";
import { Address, AddressLibrary } from "../types/Address.sol";
import { FeeModel, OperationTypes } from "../types/CommonTypes.sol";
library StabilityFeeOperations {
using OperationContextLibrary for DOperationContext.OperationContext;
using AddressLibrary for Address;
uint256 private constant PRECISION = 1e18;
uint256 private constant BASIS_POINTS = 10_000;
error ErrorStabilityModeMintNotAllowed(GroupId groupId, uint8 operationType, uint8 feeModel, bool isInStabilityMode);
error ErrorStabilityModeRedeemNotAllowed(GroupId groupId, uint8 operationType, uint8 feeModel, bool isInStabilityMode);
struct StabilityState {
uint256 maxBaseInOrOut; // Maximum base token input/output before stability mode
uint256 maxTokenAmount; // Maximum token amount for mint/redeem
bool isInStabilityMode; // Whether system is in stability mode
uint256 boundedAmount; // Amount bounded by stability restrictions
uint24 appliedFeeRate; // Fee rate to apply based on stability state
}
/**
* @notice Gets stability state information for the current operation.
* @param context The operation context.
* @param amount The requested amount for the operation.
* @return state The stability state containing relevant information.
*/
function getStabilityState(
DOperationContext.OperationContext memory context,
uint256 amount
) internal view returns (StabilityState memory state) {
ITreasury treasury = ITreasury(context.getTreasury().toAddress());
uint256 stabilityRatio = context.getStabilityRatio();
uint256 collateralRatio = context.getContextCollateralRatio();
uint8 operationType = context.getOperationType();
// Step 1: Determine if we're in stability mode
state.isInStabilityMode = collateralRatio < stabilityRatio;
// Step 2: Validate stability mode restrictions
if (context.isMintOperation()) {
if (operationType == uint8(OperationTypes.OP_TYPE_MINT_YT)) {
validateMintStabilityMode(context, state);
// Only get maxMintable amounts if we pass stability validation
(state.maxBaseInOrOut, state.maxTokenAmount) = treasury.maxMintableAToken(context.groupId, stabilityRatio);
state.boundedAmount = boundAmount(amount, state.maxBaseInOrOut);
} else {
// For VT minting, no bounds needed
state.maxBaseInOrOut = type(uint256).max;
state.maxTokenAmount = type(uint256).max;
state.boundedAmount = amount;
}
} else {
// Redemption operations
// Step 3: Check stability mode restrictions for redemptions
validateRedeemStabilityMode(context, state);
// Set boundedAmount directly to the requested amount
state.boundedAmount = amount;
}
return state;
}
/**
* @notice Validates mint operations in stability mode.
* @param context The operation context.
* @param state The current stability state.
*/
function validateMintStabilityMode(DOperationContext.OperationContext memory context, StabilityState memory state) private pure {
if (!state.isInStabilityMode) return;
// Only check YT minting restrictions
if (context.getOperationType() == uint8(OperationTypes.OP_TYPE_MINT_YT)) {
uint8 feeModel = context.getFeeModel();
// Allow minting only for specific fee models in stability mode
if (feeModel != uint8(FeeModel.CURATED_PAIRS_FEE) && feeModel != uint8(FeeModel.BLANK_FEE)) {
revert ErrorStabilityModeMintNotAllowed(context.groupId, context.getOperationType(), feeModel, state.isInStabilityMode);
}
}
}
/**
* @notice Validates redeem operations in stability mode.
* @param context The operation context.
* @param state The current stability state.
*/
function validateRedeemStabilityMode(DOperationContext.OperationContext memory context, StabilityState memory state) private pure {
if (!state.isInStabilityMode) return;
// Restrict VT redemption in stability mode
if (context.getOperationType() == uint8(OperationTypes.OP_TYPE_REDEEM_VT)) {
uint8 feeModel = context.getFeeModel();
revert ErrorStabilityModeRedeemNotAllowed(context.groupId, feeModel, context.getFeeModel(), state.isInStabilityMode);
}
}
/**
* @notice Bounds mint amount based on stability limits.
* @param amount The requested amount.
* @param maxAmount The maximum allowed amount.
* @return The bounded amount.
*/
function boundAmount(uint256 amount, uint256 maxAmount) private pure returns (uint256) {
return amount > maxAmount ? maxAmount : amount;
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
/**
* @title IPriceOracle
* @notice Interface for the PriceOracle contract, managing and updating token price feeds.
*/
interface IPriceOracle {
/**
* @dev Emitted when a price feed is added for a token.
* @param token The address of the token.
* @param feed The address of the Chainlink price feed.
*/
event FeedAdded(address indexed token, address indexed feed);
/**
* @dev Emitted when a price feed is removed for a token.
* @param token The address of the token.
*/
event FeedRemoved(address indexed token);
/**
* @dev Emitted when a price feed is updated for a token.
* @param token The address of the token.
* @param oldFeed The address of the old price feed.
* @param newFeed The address of the new price feed.
*/
event FeedUpdated(address indexed token, address indexed oldFeed, address indexed newFeed);
// Custom Errors
error ZeroAddress();
error InvalidAddress();
error InvalidOperation();
error InvalidAmount();
error StalePrice();
error InvalidInput();
error DuplicateEntry();
/**
* @notice Gets the latest price data for a specified token.
* @param token The address of the token.
* @return isValid Whether the price is valid.
* @return price The last updated price (18 decimals).
*/
function getPrice(address token) external view returns (bool isValid, uint256 price);
/**
* @notice gets the token decimals from the feed
*/
function getFeedDecimals(address token) external view returns (uint8 decimals);
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
// solhint-disable
/// @title Contains 512-bit math functions
/// @notice Facilitates multiplication and division that can have overflow of an intermediate value without any loss of precision
/// @dev Handles "phantom overflow" i.e., allows multiplication and division where an intermediate value overflows 256 bits
library FullMath {
/// @notice Calculates floor(a×b÷denominator) with full precision. Throws if result overflows a uint256 or denominator == 0
/// @param a The multiplicand
/// @param b The multiplier
/// @param denominator The divisor
/// @return result The 256-bit result
/// @dev Credit to Remco Bloemen under MIT license https://xn--2-umb.com/21/muldiv
function mulDiv(uint256 a, uint256 b, uint256 denominator) internal pure returns (uint256 result) {
unchecked {
// 512-bit multiply [prod1 prod0] = a * b
// Compute the product mod 2**256 and mod 2**256 - 1
// then use the Chinese Remainder Theorem to reconstruct
// the 512 bit result. The result is stored in two 256
// variables such that product = prod1 * 2**256 + prod0
uint256 prod0 = a * b; // Least significant 256 bits of the product
uint256 prod1; // Most significant 256 bits of the product
assembly ("memory-safe") {
let mm := mulmod(a, b, not(0))
prod1 := sub(sub(mm, prod0), lt(mm, prod0))
}
// Make sure the result is less than 2**256.
// Also prevents denominator == 0
require(denominator > prod1, "FullMath: denominator too small");
// Handle non-overflow cases, 256 by 256 division
if (prod1 == 0) {
assembly ("memory-safe") {
result := div(prod0, denominator)
}
return result;
}
///////////////////////////////////////////////
// 512 by 256 division.
///////////////////////////////////////////////
// Make division exact by subtracting the remainder from [prod1 prod0]
// Compute remainder using mulmod
uint256 remainder;
assembly ("memory-safe") {
remainder := mulmod(a, b, denominator)
}
// Subtract 256 bit number from 512 bit number
assembly ("memory-safe") {
prod1 := sub(prod1, gt(remainder, prod0))
prod0 := sub(prod0, remainder)
}
// Factor powers of two out of denominator
// Compute largest power of two divisor of denominator.
// Always >= 1.
uint256 twos = (0 - denominator) & denominator;
// Divide denominator by power of two
assembly ("memory-safe") {
denominator := div(denominator, twos)
}
// Divide [prod1 prod0] by the factors of two
assembly ("memory-safe") {
prod0 := div(prod0, twos)
}
// Shift in bits from prod1 into prod0. For this we need
// to flip `twos` such that it is 2**256 / twos.
// If twos is zero, then it becomes one
assembly ("memory-safe") {
twos := add(div(sub(0, twos), twos), 1)
}
prod0 |= prod1 * twos;
// Invert denominator mod 2**256
// Now that denominator is an odd number, it has an inverse
// modulo 2**256 such that denominator * inv = 1 mod 2**256.
// Compute the inverse by starting with a seed that is correct
// correct for four bits. That is, denominator * inv = 1 mod 2**4
uint256 inv = (3 * denominator) ^ 2;
// Now use Newton-Raphson iteration to improve the precision.
// Thanks to Hensel's lifting lemma, this also works in modular
// arithmetic, doubling the correct bits in each step.
inv *= 2 - denominator * inv; // inverse mod 2**8
inv *= 2 - denominator * inv; // inverse mod 2**16
inv *= 2 - denominator * inv; // inverse mod 2**32
inv *= 2 - denominator * inv; // inverse mod 2**64
inv *= 2 - denominator * inv; // inverse mod 2**128
inv *= 2 - denominator * inv; // inverse mod 2**256
// Because the division is now exact we can divide by multiplying
// with the modular inverse of denominator. This will give us the
// correct result modulo 2**256. Since the preconditions guarantee
// that the outcome is less than 2**256, this is the final result.
// We don't need to compute the high bits of the result and prod1
// is no longer required.
result = prod0 * inv;
return result;
}
}
/// @notice Calculates ceil(a×b÷denominator) with full precision. Throws if result overflows a uint256 or denominator == 0
/// @param a The multiplicand
/// @param b The multiplier
/// @param denominator The divisor
/// @return result The 256-bit result
function mulDivRoundingUp(uint256 a, uint256 b, uint256 denominator) internal pure returns (uint256 result) {
unchecked {
result = mulDiv(a, b, denominator);
if (mulmod(a, b, denominator) != 0) {
require(++result > 0, "FullMath: addition overflow");
}
}
}
}// SPDX-License-Identifier: MIT
pragma solidity >=0.4.22 <0.9.0;
library console {
address constant CONSOLE_ADDRESS = address(0x000000000000000000636F6e736F6c652e6c6f67);
function _castLogPayloadViewToPure(
function(bytes memory) internal view fnIn
) internal pure returns (function(bytes memory) internal pure fnOut) {
assembly {
fnOut := fnIn
}
}
function _sendLogPayload(bytes memory payload) internal pure {
_castLogPayloadViewToPure(_sendLogPayloadView)(payload);
}
function _sendLogPayloadView(bytes memory payload) private view {
uint256 payloadLength = payload.length;
address consoleAddress = CONSOLE_ADDRESS;
/// @solidity memory-safe-assembly
assembly {
let payloadStart := add(payload, 32)
let r := staticcall(gas(), consoleAddress, payloadStart, payloadLength, 0, 0)
}
}
function log() internal pure {
_sendLogPayload(abi.encodeWithSignature("log()"));
}
function logInt(int p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(int)", p0));
}
function logUint(uint p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint)", p0));
}
function logString(string memory p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string)", p0));
}
function logBool(bool p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool)", p0));
}
function logAddress(address p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address)", p0));
}
function logBytes(bytes memory p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes)", p0));
}
function logBytes1(bytes1 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes1)", p0));
}
function logBytes2(bytes2 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes2)", p0));
}
function logBytes3(bytes3 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes3)", p0));
}
function logBytes4(bytes4 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes4)", p0));
}
function logBytes5(bytes5 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes5)", p0));
}
function logBytes6(bytes6 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes6)", p0));
}
function logBytes7(bytes7 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes7)", p0));
}
function logBytes8(bytes8 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes8)", p0));
}
function logBytes9(bytes9 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes9)", p0));
}
function logBytes10(bytes10 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes10)", p0));
}
function logBytes11(bytes11 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes11)", p0));
}
function logBytes12(bytes12 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes12)", p0));
}
function logBytes13(bytes13 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes13)", p0));
}
function logBytes14(bytes14 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes14)", p0));
}
function logBytes15(bytes15 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes15)", p0));
}
function logBytes16(bytes16 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes16)", p0));
}
function logBytes17(bytes17 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes17)", p0));
}
function logBytes18(bytes18 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes18)", p0));
}
function logBytes19(bytes19 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes19)", p0));
}
function logBytes20(bytes20 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes20)", p0));
}
function logBytes21(bytes21 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes21)", p0));
}
function logBytes22(bytes22 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes22)", p0));
}
function logBytes23(bytes23 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes23)", p0));
}
function logBytes24(bytes24 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes24)", p0));
}
function logBytes25(bytes25 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes25)", p0));
}
function logBytes26(bytes26 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes26)", p0));
}
function logBytes27(bytes27 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes27)", p0));
}
function logBytes28(bytes28 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes28)", p0));
}
function logBytes29(bytes29 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes29)", p0));
}
function logBytes30(bytes30 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes30)", p0));
}
function logBytes31(bytes31 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes31)", p0));
}
function logBytes32(bytes32 p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bytes32)", p0));
}
function log(uint p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint)", p0));
}
function log(int p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(int)", p0));
}
function log(string memory p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string)", p0));
}
function log(bool p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool)", p0));
}
function log(address p0) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address)", p0));
}
function log(uint p0, uint p1) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,uint)", p0, p1));
}
function log(uint p0, string memory p1) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,string)", p0, p1));
}
function log(uint p0, bool p1) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,bool)", p0, p1));
}
function log(uint p0, address p1) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,address)", p0, p1));
}
function log(string memory p0, uint p1) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,uint)", p0, p1));
}
function log(string memory p0, int p1) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,int)", p0, p1));
}
function log(string memory p0, string memory p1) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,string)", p0, p1));
}
function log(string memory p0, bool p1) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,bool)", p0, p1));
}
function log(string memory p0, address p1) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,address)", p0, p1));
}
function log(bool p0, uint p1) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,uint)", p0, p1));
}
function log(bool p0, string memory p1) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,string)", p0, p1));
}
function log(bool p0, bool p1) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,bool)", p0, p1));
}
function log(bool p0, address p1) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,address)", p0, p1));
}
function log(address p0, uint p1) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,uint)", p0, p1));
}
function log(address p0, string memory p1) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,string)", p0, p1));
}
function log(address p0, bool p1) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,bool)", p0, p1));
}
function log(address p0, address p1) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,address)", p0, p1));
}
function log(uint p0, uint p1, uint p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,uint,uint)", p0, p1, p2));
}
function log(uint p0, uint p1, string memory p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,uint,string)", p0, p1, p2));
}
function log(uint p0, uint p1, bool p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,uint,bool)", p0, p1, p2));
}
function log(uint p0, uint p1, address p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,uint,address)", p0, p1, p2));
}
function log(uint p0, string memory p1, uint p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,string,uint)", p0, p1, p2));
}
function log(uint p0, string memory p1, string memory p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,string,string)", p0, p1, p2));
}
function log(uint p0, string memory p1, bool p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,string,bool)", p0, p1, p2));
}
function log(uint p0, string memory p1, address p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,string,address)", p0, p1, p2));
}
function log(uint p0, bool p1, uint p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,bool,uint)", p0, p1, p2));
}
function log(uint p0, bool p1, string memory p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,bool,string)", p0, p1, p2));
}
function log(uint p0, bool p1, bool p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,bool,bool)", p0, p1, p2));
}
function log(uint p0, bool p1, address p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,bool,address)", p0, p1, p2));
}
function log(uint p0, address p1, uint p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,address,uint)", p0, p1, p2));
}
function log(uint p0, address p1, string memory p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,address,string)", p0, p1, p2));
}
function log(uint p0, address p1, bool p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,address,bool)", p0, p1, p2));
}
function log(uint p0, address p1, address p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,address,address)", p0, p1, p2));
}
function log(string memory p0, uint p1, uint p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,uint,uint)", p0, p1, p2));
}
function log(string memory p0, uint p1, string memory p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,uint,string)", p0, p1, p2));
}
function log(string memory p0, uint p1, bool p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,uint,bool)", p0, p1, p2));
}
function log(string memory p0, uint p1, address p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,uint,address)", p0, p1, p2));
}
function log(string memory p0, string memory p1, uint p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,string,uint)", p0, p1, p2));
}
function log(string memory p0, string memory p1, string memory p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,string,string)", p0, p1, p2));
}
function log(string memory p0, string memory p1, bool p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,string,bool)", p0, p1, p2));
}
function log(string memory p0, string memory p1, address p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,string,address)", p0, p1, p2));
}
function log(string memory p0, bool p1, uint p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,bool,uint)", p0, p1, p2));
}
function log(string memory p0, bool p1, string memory p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,bool,string)", p0, p1, p2));
}
function log(string memory p0, bool p1, bool p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,bool,bool)", p0, p1, p2));
}
function log(string memory p0, bool p1, address p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,bool,address)", p0, p1, p2));
}
function log(string memory p0, address p1, uint p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,address,uint)", p0, p1, p2));
}
function log(string memory p0, address p1, string memory p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,address,string)", p0, p1, p2));
}
function log(string memory p0, address p1, bool p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,address,bool)", p0, p1, p2));
}
function log(string memory p0, address p1, address p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,address,address)", p0, p1, p2));
}
function log(bool p0, uint p1, uint p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,uint,uint)", p0, p1, p2));
}
function log(bool p0, uint p1, string memory p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,uint,string)", p0, p1, p2));
}
function log(bool p0, uint p1, bool p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,uint,bool)", p0, p1, p2));
}
function log(bool p0, uint p1, address p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,uint,address)", p0, p1, p2));
}
function log(bool p0, string memory p1, uint p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,string,uint)", p0, p1, p2));
}
function log(bool p0, string memory p1, string memory p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,string,string)", p0, p1, p2));
}
function log(bool p0, string memory p1, bool p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,string,bool)", p0, p1, p2));
}
function log(bool p0, string memory p1, address p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,string,address)", p0, p1, p2));
}
function log(bool p0, bool p1, uint p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,bool,uint)", p0, p1, p2));
}
function log(bool p0, bool p1, string memory p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,bool,string)", p0, p1, p2));
}
function log(bool p0, bool p1, bool p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,bool,bool)", p0, p1, p2));
}
function log(bool p0, bool p1, address p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,bool,address)", p0, p1, p2));
}
function log(bool p0, address p1, uint p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,address,uint)", p0, p1, p2));
}
function log(bool p0, address p1, string memory p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,address,string)", p0, p1, p2));
}
function log(bool p0, address p1, bool p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,address,bool)", p0, p1, p2));
}
function log(bool p0, address p1, address p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,address,address)", p0, p1, p2));
}
function log(address p0, uint p1, uint p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,uint,uint)", p0, p1, p2));
}
function log(address p0, uint p1, string memory p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,uint,string)", p0, p1, p2));
}
function log(address p0, uint p1, bool p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,uint,bool)", p0, p1, p2));
}
function log(address p0, uint p1, address p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,uint,address)", p0, p1, p2));
}
function log(address p0, string memory p1, uint p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,string,uint)", p0, p1, p2));
}
function log(address p0, string memory p1, string memory p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,string,string)", p0, p1, p2));
}
function log(address p0, string memory p1, bool p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,string,bool)", p0, p1, p2));
}
function log(address p0, string memory p1, address p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,string,address)", p0, p1, p2));
}
function log(address p0, bool p1, uint p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,bool,uint)", p0, p1, p2));
}
function log(address p0, bool p1, string memory p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,bool,string)", p0, p1, p2));
}
function log(address p0, bool p1, bool p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,bool,bool)", p0, p1, p2));
}
function log(address p0, bool p1, address p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,bool,address)", p0, p1, p2));
}
function log(address p0, address p1, uint p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,address,uint)", p0, p1, p2));
}
function log(address p0, address p1, string memory p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,address,string)", p0, p1, p2));
}
function log(address p0, address p1, bool p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,address,bool)", p0, p1, p2));
}
function log(address p0, address p1, address p2) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,address,address)", p0, p1, p2));
}
function log(uint p0, uint p1, uint p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,uint,uint,uint)", p0, p1, p2, p3));
}
function log(uint p0, uint p1, uint p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,uint,uint,string)", p0, p1, p2, p3));
}
function log(uint p0, uint p1, uint p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,uint,uint,bool)", p0, p1, p2, p3));
}
function log(uint p0, uint p1, uint p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,uint,uint,address)", p0, p1, p2, p3));
}
function log(uint p0, uint p1, string memory p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,uint,string,uint)", p0, p1, p2, p3));
}
function log(uint p0, uint p1, string memory p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,uint,string,string)", p0, p1, p2, p3));
}
function log(uint p0, uint p1, string memory p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,uint,string,bool)", p0, p1, p2, p3));
}
function log(uint p0, uint p1, string memory p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,uint,string,address)", p0, p1, p2, p3));
}
function log(uint p0, uint p1, bool p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,uint,bool,uint)", p0, p1, p2, p3));
}
function log(uint p0, uint p1, bool p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,uint,bool,string)", p0, p1, p2, p3));
}
function log(uint p0, uint p1, bool p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,uint,bool,bool)", p0, p1, p2, p3));
}
function log(uint p0, uint p1, bool p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,uint,bool,address)", p0, p1, p2, p3));
}
function log(uint p0, uint p1, address p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,uint,address,uint)", p0, p1, p2, p3));
}
function log(uint p0, uint p1, address p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,uint,address,string)", p0, p1, p2, p3));
}
function log(uint p0, uint p1, address p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,uint,address,bool)", p0, p1, p2, p3));
}
function log(uint p0, uint p1, address p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,uint,address,address)", p0, p1, p2, p3));
}
function log(uint p0, string memory p1, uint p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,string,uint,uint)", p0, p1, p2, p3));
}
function log(uint p0, string memory p1, uint p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,string,uint,string)", p0, p1, p2, p3));
}
function log(uint p0, string memory p1, uint p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,string,uint,bool)", p0, p1, p2, p3));
}
function log(uint p0, string memory p1, uint p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,string,uint,address)", p0, p1, p2, p3));
}
function log(uint p0, string memory p1, string memory p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,string,string,uint)", p0, p1, p2, p3));
}
function log(uint p0, string memory p1, string memory p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,string,string,string)", p0, p1, p2, p3));
}
function log(uint p0, string memory p1, string memory p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,string,string,bool)", p0, p1, p2, p3));
}
function log(uint p0, string memory p1, string memory p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,string,string,address)", p0, p1, p2, p3));
}
function log(uint p0, string memory p1, bool p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,string,bool,uint)", p0, p1, p2, p3));
}
function log(uint p0, string memory p1, bool p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,string,bool,string)", p0, p1, p2, p3));
}
function log(uint p0, string memory p1, bool p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,string,bool,bool)", p0, p1, p2, p3));
}
function log(uint p0, string memory p1, bool p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,string,bool,address)", p0, p1, p2, p3));
}
function log(uint p0, string memory p1, address p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,string,address,uint)", p0, p1, p2, p3));
}
function log(uint p0, string memory p1, address p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,string,address,string)", p0, p1, p2, p3));
}
function log(uint p0, string memory p1, address p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,string,address,bool)", p0, p1, p2, p3));
}
function log(uint p0, string memory p1, address p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,string,address,address)", p0, p1, p2, p3));
}
function log(uint p0, bool p1, uint p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,bool,uint,uint)", p0, p1, p2, p3));
}
function log(uint p0, bool p1, uint p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,bool,uint,string)", p0, p1, p2, p3));
}
function log(uint p0, bool p1, uint p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,bool,uint,bool)", p0, p1, p2, p3));
}
function log(uint p0, bool p1, uint p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,bool,uint,address)", p0, p1, p2, p3));
}
function log(uint p0, bool p1, string memory p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,bool,string,uint)", p0, p1, p2, p3));
}
function log(uint p0, bool p1, string memory p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,bool,string,string)", p0, p1, p2, p3));
}
function log(uint p0, bool p1, string memory p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,bool,string,bool)", p0, p1, p2, p3));
}
function log(uint p0, bool p1, string memory p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,bool,string,address)", p0, p1, p2, p3));
}
function log(uint p0, bool p1, bool p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,bool,bool,uint)", p0, p1, p2, p3));
}
function log(uint p0, bool p1, bool p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,bool,bool,string)", p0, p1, p2, p3));
}
function log(uint p0, bool p1, bool p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,bool,bool,bool)", p0, p1, p2, p3));
}
function log(uint p0, bool p1, bool p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,bool,bool,address)", p0, p1, p2, p3));
}
function log(uint p0, bool p1, address p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,bool,address,uint)", p0, p1, p2, p3));
}
function log(uint p0, bool p1, address p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,bool,address,string)", p0, p1, p2, p3));
}
function log(uint p0, bool p1, address p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,bool,address,bool)", p0, p1, p2, p3));
}
function log(uint p0, bool p1, address p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,bool,address,address)", p0, p1, p2, p3));
}
function log(uint p0, address p1, uint p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,address,uint,uint)", p0, p1, p2, p3));
}
function log(uint p0, address p1, uint p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,address,uint,string)", p0, p1, p2, p3));
}
function log(uint p0, address p1, uint p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,address,uint,bool)", p0, p1, p2, p3));
}
function log(uint p0, address p1, uint p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,address,uint,address)", p0, p1, p2, p3));
}
function log(uint p0, address p1, string memory p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,address,string,uint)", p0, p1, p2, p3));
}
function log(uint p0, address p1, string memory p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,address,string,string)", p0, p1, p2, p3));
}
function log(uint p0, address p1, string memory p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,address,string,bool)", p0, p1, p2, p3));
}
function log(uint p0, address p1, string memory p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,address,string,address)", p0, p1, p2, p3));
}
function log(uint p0, address p1, bool p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,address,bool,uint)", p0, p1, p2, p3));
}
function log(uint p0, address p1, bool p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,address,bool,string)", p0, p1, p2, p3));
}
function log(uint p0, address p1, bool p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,address,bool,bool)", p0, p1, p2, p3));
}
function log(uint p0, address p1, bool p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,address,bool,address)", p0, p1, p2, p3));
}
function log(uint p0, address p1, address p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,address,address,uint)", p0, p1, p2, p3));
}
function log(uint p0, address p1, address p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,address,address,string)", p0, p1, p2, p3));
}
function log(uint p0, address p1, address p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,address,address,bool)", p0, p1, p2, p3));
}
function log(uint p0, address p1, address p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(uint,address,address,address)", p0, p1, p2, p3));
}
function log(string memory p0, uint p1, uint p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,uint,uint,uint)", p0, p1, p2, p3));
}
function log(string memory p0, uint p1, uint p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,uint,uint,string)", p0, p1, p2, p3));
}
function log(string memory p0, uint p1, uint p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,uint,uint,bool)", p0, p1, p2, p3));
}
function log(string memory p0, uint p1, uint p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,uint,uint,address)", p0, p1, p2, p3));
}
function log(string memory p0, uint p1, string memory p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,uint,string,uint)", p0, p1, p2, p3));
}
function log(string memory p0, uint p1, string memory p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,uint,string,string)", p0, p1, p2, p3));
}
function log(string memory p0, uint p1, string memory p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,uint,string,bool)", p0, p1, p2, p3));
}
function log(string memory p0, uint p1, string memory p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,uint,string,address)", p0, p1, p2, p3));
}
function log(string memory p0, uint p1, bool p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,uint,bool,uint)", p0, p1, p2, p3));
}
function log(string memory p0, uint p1, bool p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,uint,bool,string)", p0, p1, p2, p3));
}
function log(string memory p0, uint p1, bool p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,uint,bool,bool)", p0, p1, p2, p3));
}
function log(string memory p0, uint p1, bool p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,uint,bool,address)", p0, p1, p2, p3));
}
function log(string memory p0, uint p1, address p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,uint,address,uint)", p0, p1, p2, p3));
}
function log(string memory p0, uint p1, address p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,uint,address,string)", p0, p1, p2, p3));
}
function log(string memory p0, uint p1, address p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,uint,address,bool)", p0, p1, p2, p3));
}
function log(string memory p0, uint p1, address p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,uint,address,address)", p0, p1, p2, p3));
}
function log(string memory p0, string memory p1, uint p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,string,uint,uint)", p0, p1, p2, p3));
}
function log(string memory p0, string memory p1, uint p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,string,uint,string)", p0, p1, p2, p3));
}
function log(string memory p0, string memory p1, uint p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,string,uint,bool)", p0, p1, p2, p3));
}
function log(string memory p0, string memory p1, uint p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,string,uint,address)", p0, p1, p2, p3));
}
function log(string memory p0, string memory p1, string memory p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,string,string,uint)", p0, p1, p2, p3));
}
function log(string memory p0, string memory p1, string memory p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,string,string,string)", p0, p1, p2, p3));
}
function log(string memory p0, string memory p1, string memory p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,string,string,bool)", p0, p1, p2, p3));
}
function log(string memory p0, string memory p1, string memory p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,string,string,address)", p0, p1, p2, p3));
}
function log(string memory p0, string memory p1, bool p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,string,bool,uint)", p0, p1, p2, p3));
}
function log(string memory p0, string memory p1, bool p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,string,bool,string)", p0, p1, p2, p3));
}
function log(string memory p0, string memory p1, bool p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,string,bool,bool)", p0, p1, p2, p3));
}
function log(string memory p0, string memory p1, bool p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,string,bool,address)", p0, p1, p2, p3));
}
function log(string memory p0, string memory p1, address p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,string,address,uint)", p0, p1, p2, p3));
}
function log(string memory p0, string memory p1, address p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,string,address,string)", p0, p1, p2, p3));
}
function log(string memory p0, string memory p1, address p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,string,address,bool)", p0, p1, p2, p3));
}
function log(string memory p0, string memory p1, address p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,string,address,address)", p0, p1, p2, p3));
}
function log(string memory p0, bool p1, uint p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,bool,uint,uint)", p0, p1, p2, p3));
}
function log(string memory p0, bool p1, uint p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,bool,uint,string)", p0, p1, p2, p3));
}
function log(string memory p0, bool p1, uint p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,bool,uint,bool)", p0, p1, p2, p3));
}
function log(string memory p0, bool p1, uint p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,bool,uint,address)", p0, p1, p2, p3));
}
function log(string memory p0, bool p1, string memory p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,bool,string,uint)", p0, p1, p2, p3));
}
function log(string memory p0, bool p1, string memory p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,bool,string,string)", p0, p1, p2, p3));
}
function log(string memory p0, bool p1, string memory p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,bool,string,bool)", p0, p1, p2, p3));
}
function log(string memory p0, bool p1, string memory p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,bool,string,address)", p0, p1, p2, p3));
}
function log(string memory p0, bool p1, bool p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,bool,bool,uint)", p0, p1, p2, p3));
}
function log(string memory p0, bool p1, bool p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,bool,bool,string)", p0, p1, p2, p3));
}
function log(string memory p0, bool p1, bool p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,bool,bool,bool)", p0, p1, p2, p3));
}
function log(string memory p0, bool p1, bool p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,bool,bool,address)", p0, p1, p2, p3));
}
function log(string memory p0, bool p1, address p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,bool,address,uint)", p0, p1, p2, p3));
}
function log(string memory p0, bool p1, address p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,bool,address,string)", p0, p1, p2, p3));
}
function log(string memory p0, bool p1, address p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,bool,address,bool)", p0, p1, p2, p3));
}
function log(string memory p0, bool p1, address p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,bool,address,address)", p0, p1, p2, p3));
}
function log(string memory p0, address p1, uint p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,address,uint,uint)", p0, p1, p2, p3));
}
function log(string memory p0, address p1, uint p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,address,uint,string)", p0, p1, p2, p3));
}
function log(string memory p0, address p1, uint p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,address,uint,bool)", p0, p1, p2, p3));
}
function log(string memory p0, address p1, uint p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,address,uint,address)", p0, p1, p2, p3));
}
function log(string memory p0, address p1, string memory p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,address,string,uint)", p0, p1, p2, p3));
}
function log(string memory p0, address p1, string memory p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,address,string,string)", p0, p1, p2, p3));
}
function log(string memory p0, address p1, string memory p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,address,string,bool)", p0, p1, p2, p3));
}
function log(string memory p0, address p1, string memory p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,address,string,address)", p0, p1, p2, p3));
}
function log(string memory p0, address p1, bool p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,address,bool,uint)", p0, p1, p2, p3));
}
function log(string memory p0, address p1, bool p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,address,bool,string)", p0, p1, p2, p3));
}
function log(string memory p0, address p1, bool p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,address,bool,bool)", p0, p1, p2, p3));
}
function log(string memory p0, address p1, bool p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,address,bool,address)", p0, p1, p2, p3));
}
function log(string memory p0, address p1, address p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,address,address,uint)", p0, p1, p2, p3));
}
function log(string memory p0, address p1, address p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,address,address,string)", p0, p1, p2, p3));
}
function log(string memory p0, address p1, address p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,address,address,bool)", p0, p1, p2, p3));
}
function log(string memory p0, address p1, address p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(string,address,address,address)", p0, p1, p2, p3));
}
function log(bool p0, uint p1, uint p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,uint,uint,uint)", p0, p1, p2, p3));
}
function log(bool p0, uint p1, uint p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,uint,uint,string)", p0, p1, p2, p3));
}
function log(bool p0, uint p1, uint p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,uint,uint,bool)", p0, p1, p2, p3));
}
function log(bool p0, uint p1, uint p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,uint,uint,address)", p0, p1, p2, p3));
}
function log(bool p0, uint p1, string memory p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,uint,string,uint)", p0, p1, p2, p3));
}
function log(bool p0, uint p1, string memory p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,uint,string,string)", p0, p1, p2, p3));
}
function log(bool p0, uint p1, string memory p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,uint,string,bool)", p0, p1, p2, p3));
}
function log(bool p0, uint p1, string memory p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,uint,string,address)", p0, p1, p2, p3));
}
function log(bool p0, uint p1, bool p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,uint,bool,uint)", p0, p1, p2, p3));
}
function log(bool p0, uint p1, bool p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,uint,bool,string)", p0, p1, p2, p3));
}
function log(bool p0, uint p1, bool p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,uint,bool,bool)", p0, p1, p2, p3));
}
function log(bool p0, uint p1, bool p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,uint,bool,address)", p0, p1, p2, p3));
}
function log(bool p0, uint p1, address p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,uint,address,uint)", p0, p1, p2, p3));
}
function log(bool p0, uint p1, address p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,uint,address,string)", p0, p1, p2, p3));
}
function log(bool p0, uint p1, address p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,uint,address,bool)", p0, p1, p2, p3));
}
function log(bool p0, uint p1, address p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,uint,address,address)", p0, p1, p2, p3));
}
function log(bool p0, string memory p1, uint p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,string,uint,uint)", p0, p1, p2, p3));
}
function log(bool p0, string memory p1, uint p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,string,uint,string)", p0, p1, p2, p3));
}
function log(bool p0, string memory p1, uint p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,string,uint,bool)", p0, p1, p2, p3));
}
function log(bool p0, string memory p1, uint p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,string,uint,address)", p0, p1, p2, p3));
}
function log(bool p0, string memory p1, string memory p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,string,string,uint)", p0, p1, p2, p3));
}
function log(bool p0, string memory p1, string memory p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,string,string,string)", p0, p1, p2, p3));
}
function log(bool p0, string memory p1, string memory p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,string,string,bool)", p0, p1, p2, p3));
}
function log(bool p0, string memory p1, string memory p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,string,string,address)", p0, p1, p2, p3));
}
function log(bool p0, string memory p1, bool p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,string,bool,uint)", p0, p1, p2, p3));
}
function log(bool p0, string memory p1, bool p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,string,bool,string)", p0, p1, p2, p3));
}
function log(bool p0, string memory p1, bool p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,string,bool,bool)", p0, p1, p2, p3));
}
function log(bool p0, string memory p1, bool p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,string,bool,address)", p0, p1, p2, p3));
}
function log(bool p0, string memory p1, address p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,string,address,uint)", p0, p1, p2, p3));
}
function log(bool p0, string memory p1, address p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,string,address,string)", p0, p1, p2, p3));
}
function log(bool p0, string memory p1, address p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,string,address,bool)", p0, p1, p2, p3));
}
function log(bool p0, string memory p1, address p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,string,address,address)", p0, p1, p2, p3));
}
function log(bool p0, bool p1, uint p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,bool,uint,uint)", p0, p1, p2, p3));
}
function log(bool p0, bool p1, uint p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,bool,uint,string)", p0, p1, p2, p3));
}
function log(bool p0, bool p1, uint p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,bool,uint,bool)", p0, p1, p2, p3));
}
function log(bool p0, bool p1, uint p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,bool,uint,address)", p0, p1, p2, p3));
}
function log(bool p0, bool p1, string memory p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,bool,string,uint)", p0, p1, p2, p3));
}
function log(bool p0, bool p1, string memory p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,bool,string,string)", p0, p1, p2, p3));
}
function log(bool p0, bool p1, string memory p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,bool,string,bool)", p0, p1, p2, p3));
}
function log(bool p0, bool p1, string memory p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,bool,string,address)", p0, p1, p2, p3));
}
function log(bool p0, bool p1, bool p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,bool,bool,uint)", p0, p1, p2, p3));
}
function log(bool p0, bool p1, bool p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,bool,bool,string)", p0, p1, p2, p3));
}
function log(bool p0, bool p1, bool p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,bool,bool,bool)", p0, p1, p2, p3));
}
function log(bool p0, bool p1, bool p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,bool,bool,address)", p0, p1, p2, p3));
}
function log(bool p0, bool p1, address p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,bool,address,uint)", p0, p1, p2, p3));
}
function log(bool p0, bool p1, address p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,bool,address,string)", p0, p1, p2, p3));
}
function log(bool p0, bool p1, address p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,bool,address,bool)", p0, p1, p2, p3));
}
function log(bool p0, bool p1, address p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,bool,address,address)", p0, p1, p2, p3));
}
function log(bool p0, address p1, uint p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,address,uint,uint)", p0, p1, p2, p3));
}
function log(bool p0, address p1, uint p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,address,uint,string)", p0, p1, p2, p3));
}
function log(bool p0, address p1, uint p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,address,uint,bool)", p0, p1, p2, p3));
}
function log(bool p0, address p1, uint p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,address,uint,address)", p0, p1, p2, p3));
}
function log(bool p0, address p1, string memory p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,address,string,uint)", p0, p1, p2, p3));
}
function log(bool p0, address p1, string memory p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,address,string,string)", p0, p1, p2, p3));
}
function log(bool p0, address p1, string memory p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,address,string,bool)", p0, p1, p2, p3));
}
function log(bool p0, address p1, string memory p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,address,string,address)", p0, p1, p2, p3));
}
function log(bool p0, address p1, bool p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,address,bool,uint)", p0, p1, p2, p3));
}
function log(bool p0, address p1, bool p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,address,bool,string)", p0, p1, p2, p3));
}
function log(bool p0, address p1, bool p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,address,bool,bool)", p0, p1, p2, p3));
}
function log(bool p0, address p1, bool p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,address,bool,address)", p0, p1, p2, p3));
}
function log(bool p0, address p1, address p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,address,address,uint)", p0, p1, p2, p3));
}
function log(bool p0, address p1, address p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,address,address,string)", p0, p1, p2, p3));
}
function log(bool p0, address p1, address p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,address,address,bool)", p0, p1, p2, p3));
}
function log(bool p0, address p1, address p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(bool,address,address,address)", p0, p1, p2, p3));
}
function log(address p0, uint p1, uint p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,uint,uint,uint)", p0, p1, p2, p3));
}
function log(address p0, uint p1, uint p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,uint,uint,string)", p0, p1, p2, p3));
}
function log(address p0, uint p1, uint p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,uint,uint,bool)", p0, p1, p2, p3));
}
function log(address p0, uint p1, uint p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,uint,uint,address)", p0, p1, p2, p3));
}
function log(address p0, uint p1, string memory p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,uint,string,uint)", p0, p1, p2, p3));
}
function log(address p0, uint p1, string memory p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,uint,string,string)", p0, p1, p2, p3));
}
function log(address p0, uint p1, string memory p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,uint,string,bool)", p0, p1, p2, p3));
}
function log(address p0, uint p1, string memory p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,uint,string,address)", p0, p1, p2, p3));
}
function log(address p0, uint p1, bool p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,uint,bool,uint)", p0, p1, p2, p3));
}
function log(address p0, uint p1, bool p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,uint,bool,string)", p0, p1, p2, p3));
}
function log(address p0, uint p1, bool p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,uint,bool,bool)", p0, p1, p2, p3));
}
function log(address p0, uint p1, bool p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,uint,bool,address)", p0, p1, p2, p3));
}
function log(address p0, uint p1, address p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,uint,address,uint)", p0, p1, p2, p3));
}
function log(address p0, uint p1, address p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,uint,address,string)", p0, p1, p2, p3));
}
function log(address p0, uint p1, address p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,uint,address,bool)", p0, p1, p2, p3));
}
function log(address p0, uint p1, address p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,uint,address,address)", p0, p1, p2, p3));
}
function log(address p0, string memory p1, uint p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,string,uint,uint)", p0, p1, p2, p3));
}
function log(address p0, string memory p1, uint p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,string,uint,string)", p0, p1, p2, p3));
}
function log(address p0, string memory p1, uint p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,string,uint,bool)", p0, p1, p2, p3));
}
function log(address p0, string memory p1, uint p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,string,uint,address)", p0, p1, p2, p3));
}
function log(address p0, string memory p1, string memory p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,string,string,uint)", p0, p1, p2, p3));
}
function log(address p0, string memory p1, string memory p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,string,string,string)", p0, p1, p2, p3));
}
function log(address p0, string memory p1, string memory p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,string,string,bool)", p0, p1, p2, p3));
}
function log(address p0, string memory p1, string memory p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,string,string,address)", p0, p1, p2, p3));
}
function log(address p0, string memory p1, bool p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,string,bool,uint)", p0, p1, p2, p3));
}
function log(address p0, string memory p1, bool p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,string,bool,string)", p0, p1, p2, p3));
}
function log(address p0, string memory p1, bool p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,string,bool,bool)", p0, p1, p2, p3));
}
function log(address p0, string memory p1, bool p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,string,bool,address)", p0, p1, p2, p3));
}
function log(address p0, string memory p1, address p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,string,address,uint)", p0, p1, p2, p3));
}
function log(address p0, string memory p1, address p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,string,address,string)", p0, p1, p2, p3));
}
function log(address p0, string memory p1, address p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,string,address,bool)", p0, p1, p2, p3));
}
function log(address p0, string memory p1, address p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,string,address,address)", p0, p1, p2, p3));
}
function log(address p0, bool p1, uint p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,bool,uint,uint)", p0, p1, p2, p3));
}
function log(address p0, bool p1, uint p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,bool,uint,string)", p0, p1, p2, p3));
}
function log(address p0, bool p1, uint p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,bool,uint,bool)", p0, p1, p2, p3));
}
function log(address p0, bool p1, uint p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,bool,uint,address)", p0, p1, p2, p3));
}
function log(address p0, bool p1, string memory p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,bool,string,uint)", p0, p1, p2, p3));
}
function log(address p0, bool p1, string memory p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,bool,string,string)", p0, p1, p2, p3));
}
function log(address p0, bool p1, string memory p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,bool,string,bool)", p0, p1, p2, p3));
}
function log(address p0, bool p1, string memory p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,bool,string,address)", p0, p1, p2, p3));
}
function log(address p0, bool p1, bool p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,bool,bool,uint)", p0, p1, p2, p3));
}
function log(address p0, bool p1, bool p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,bool,bool,string)", p0, p1, p2, p3));
}
function log(address p0, bool p1, bool p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,bool,bool,bool)", p0, p1, p2, p3));
}
function log(address p0, bool p1, bool p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,bool,bool,address)", p0, p1, p2, p3));
}
function log(address p0, bool p1, address p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,bool,address,uint)", p0, p1, p2, p3));
}
function log(address p0, bool p1, address p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,bool,address,string)", p0, p1, p2, p3));
}
function log(address p0, bool p1, address p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,bool,address,bool)", p0, p1, p2, p3));
}
function log(address p0, bool p1, address p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,bool,address,address)", p0, p1, p2, p3));
}
function log(address p0, address p1, uint p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,address,uint,uint)", p0, p1, p2, p3));
}
function log(address p0, address p1, uint p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,address,uint,string)", p0, p1, p2, p3));
}
function log(address p0, address p1, uint p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,address,uint,bool)", p0, p1, p2, p3));
}
function log(address p0, address p1, uint p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,address,uint,address)", p0, p1, p2, p3));
}
function log(address p0, address p1, string memory p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,address,string,uint)", p0, p1, p2, p3));
}
function log(address p0, address p1, string memory p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,address,string,string)", p0, p1, p2, p3));
}
function log(address p0, address p1, string memory p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,address,string,bool)", p0, p1, p2, p3));
}
function log(address p0, address p1, string memory p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,address,string,address)", p0, p1, p2, p3));
}
function log(address p0, address p1, bool p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,address,bool,uint)", p0, p1, p2, p3));
}
function log(address p0, address p1, bool p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,address,bool,string)", p0, p1, p2, p3));
}
function log(address p0, address p1, bool p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,address,bool,bool)", p0, p1, p2, p3));
}
function log(address p0, address p1, bool p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,address,bool,address)", p0, p1, p2, p3));
}
function log(address p0, address p1, address p2, uint p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,address,address,uint)", p0, p1, p2, p3));
}
function log(address p0, address p1, address p2, string memory p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,address,address,string)", p0, p1, p2, p3));
}
function log(address p0, address p1, address p2, bool p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,address,address,bool)", p0, p1, p2, p3));
}
function log(address p0, address p1, address p2, address p3) internal pure {
_sendLogPayload(abi.encodeWithSignature("log(address,address,address,address)", p0, p1, p2, p3));
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { WordCodec } from "../libs/WordCodec.sol";
import { CustomRevert } from "../libs/CustomRevert.sol";
import { GroupFeeLibrary } from "../libs/GroupFeeLibrary.sol";
import { OperationTypes } from "./CommonTypes.sol";
type Slot0 is bytes32;
/**
* @title Slot0Library
* @notice Library for managing operation types, fees, and rates in a space-efficient manner
* @dev Uses bit packing to store multiple values in a single bytes32 slot
*/
library Slot0Library {
using WordCodec for bytes32;
using CustomRevert for bytes4;
// Operation Type Constants
uint8 internal constant MAX_OPERATION_TYPE = 3;
// Fee Range Constants
/// @dev Same as GroupFeeLibrary
uint24 internal constant MAX_FEE = 10_000; /// 100% in basis points (1e4)
uint24 internal constant MAX_OVERRIDING_FEE = 5_000; /// 50% in basis points (5e3)
uint64 internal constant MAX_RATE = type(uint64).max; // ~18.44e18
uint96 internal constant MAX_STABILITY_TRIGGER = type(uint96).max;
// Offsets
uint256 internal constant OPERATION_TYPE_OFFSET = 0; // 8 bits
uint256 internal constant CONTEXT_PROTOCOL_FEE_OFFSET = 8; // 24 bits
uint256 internal constant CONTEXT_FEE_OFFSET = 32; // 24 bits
uint256 internal constant OVERRIDDEN_FEE_OFFSET = 56; // 24 bits
uint256 internal constant STABILITY_FEE_TRIGGER_OFFSET = 80; // 96 bits for stability fee trigger
// Bit lengths for the values
uint256 internal constant OPERATION_TYPE_LENGTH = 8;
uint256 internal constant CONTEXT_PROTOCOL_FEE_LENGTH = 24;
uint256 internal constant CONTEXT_FEE_LENGTH = 24;
uint256 internal constant OVERRIDDEN_FEE_LENGTH = 24;
uint256 internal constant RATE_LENGTH = 64; /// @dev ~18.44e18 is sufficient for rate
uint256 internal constant STABILITY_FEE_TRIGGER_LENGTH = 96;
// Custom errors
error InvalidOperationType(uint8 operationType);
error InvalidFee(uint24 fee);
error InvalidOverridenFee(uint24 fee);
error InvalidRate(uint64 rate);
error InvalidStabilityTrigger(uint96 trigger);
error InvalidPackedData();
error OperationTypeMismatch();
// Validation modifiers
/**
* @dev Validates that a protocol fees is within acceptable bounds
*/
modifier validateProtocolFee(uint24 fee) {
if (fee & GroupFeeLibrary.REMOVE_ALL_FLAGS_MASK > MAX_FEE) revert InvalidFee(fee);
_;
}
/**
* @dev Validates that hook fee is within acceptable bounds
*/
modifier validateOverridenFee(uint24 fee) {
if ((fee & GroupFeeLibrary.REMOVE_ALL_FLAGS_MASK) > MAX_OVERRIDING_FEE) revert InvalidFee(fee);
_;
}
/**
* @dev Validates that an operation type is valid
*/
modifier validateOperationType(uint8 opType) {
if (opType > MAX_OPERATION_TYPE) revert InvalidOperationType(opType);
_;
}
// Getter Functions
/// @notice Retrieves the operation type from Slot0.
/// @param _packed The packed Slot0 data
/// @return The operation type
function operationType(Slot0 _packed) internal pure returns (uint8) {
return uint8(Slot0.unwrap(_packed).decodeUint(OPERATION_TYPE_OFFSET, OPERATION_TYPE_LENGTH));
}
/// @notice Retrieves the context protocol fee from Slot0.
/// @param _packed The packed Slot0 data
/// @return The context protocol fee
function contextProtocolFee(Slot0 _packed) internal pure returns (uint24) {
uint24 fee = uint24(Slot0.unwrap(_packed).decodeUint(CONTEXT_PROTOCOL_FEE_OFFSET, CONTEXT_PROTOCOL_FEE_LENGTH));
if ((fee & GroupFeeLibrary.REMOVE_ALL_FLAGS_MASK) > MAX_FEE) revert InvalidFee(fee);
return fee;
}
/// @notice Retrieves the context LP fee from Slot0.
/// @param _packed The packed Slot0 data
/// @return The context LP fee
function contextFee(Slot0 _packed) internal pure returns (uint24) {
uint24 fee = uint24(Slot0.unwrap(_packed).decodeUint(CONTEXT_FEE_OFFSET, CONTEXT_FEE_LENGTH));
if ((fee & GroupFeeLibrary.REMOVE_ALL_FLAGS_MASK) > MAX_FEE) revert InvalidFee(fee);
return fee;
}
/// @notice Retrieves the overridden fee from Slot0.
/// @param _packed The packed Slot0 data
/// @return The overridden fee
function overriddenFee(Slot0 _packed) internal pure returns (uint24) {
uint24 fee = uint24(Slot0.unwrap(_packed).decodeUint(OVERRIDDEN_FEE_OFFSET, OVERRIDDEN_FEE_LENGTH));
if ((fee & GroupFeeLibrary.REMOVE_ALL_FLAGS_MASK) > MAX_OVERRIDING_FEE) revert InvalidFee(fee);
return fee;
}
/// @notice Retrieves the stability fee trigger ratio from Slot0.
/// @param _packed The packed Slot0 data
/// @return The stability fee trigger ratio
function stabilityConditionsTriggeringRate(Slot0 _packed) internal pure returns (uint256) {
return uint256(Slot0.unwrap(_packed).decodeUint(STABILITY_FEE_TRIGGER_OFFSET, STABILITY_FEE_TRIGGER_LENGTH));
}
// Pack Function
/// @notice Packs all Slot0 data into a single bytes32.
/// @param _operationType The operation type to pack
/// @param _contextProtocolFee The context protocol fee to pack
/// @param _contextFee The context fee to pack
/// @param _overriddenFee The overridden fee to pack
/// @param _stabilityFeeTrigger The stability fee trigger to pack
/// @return _packed The packed Slot0 data
function pack(
uint8 _operationType,
uint24 _contextProtocolFee,
uint24 _contextFee,
uint24 _overriddenFee,
uint96 _stabilityFeeTrigger
)
internal
pure
validateOperationType(_operationType)
validateProtocolFee(_contextProtocolFee)
validateProtocolFee(_contextFee)
validateOverridenFee(_overriddenFee)
returns (Slot0 _packed)
{
bytes32 packed = 0;
packed = packed.insertUint(_operationType, OPERATION_TYPE_OFFSET, OPERATION_TYPE_LENGTH);
packed = packed.insertUint(_contextProtocolFee, CONTEXT_PROTOCOL_FEE_OFFSET, CONTEXT_PROTOCOL_FEE_LENGTH);
packed = packed.insertUint(_contextFee, CONTEXT_FEE_OFFSET, CONTEXT_FEE_LENGTH);
packed = packed.insertUint(_overriddenFee, OVERRIDDEN_FEE_OFFSET, OVERRIDDEN_FEE_LENGTH);
packed = packed.insertUint(_stabilityFeeTrigger, STABILITY_FEE_TRIGGER_OFFSET, STABILITY_FEE_TRIGGER_LENGTH);
if (!isValidPacked(packed)) revert InvalidPackedData();
return Slot0.wrap(packed);
}
// Setter Functions
/// @notice Sets the overridden fee in Slot0.
/// @param _packed Current Slot0 data
/// @param _overriddenFee New overridden fee value
/// @return Updated Slot0 data
function setOverriddenFee(Slot0 _packed, uint24 _overriddenFee) internal pure validateOverridenFee(_overriddenFee) returns (Slot0) {
bytes32 packed = Slot0.unwrap(_packed).insertUint(_overriddenFee, OVERRIDDEN_FEE_OFFSET, OVERRIDDEN_FEE_LENGTH);
if (!isValidPacked(packed)) revert InvalidPackedData();
return Slot0.wrap(packed);
}
/// @notice Sets the context fee in Slot0.
/// @param _packed Current Slot0 data
/// @param _contextFee New context fee value
/// @return Updated Slot0 data
function setContextFee(Slot0 _packed, uint24 _contextFee) internal pure validateProtocolFee(_contextFee) returns (Slot0) {
bytes32 packed = Slot0.unwrap(_packed).insertUint(_contextFee, CONTEXT_FEE_OFFSET, CONTEXT_FEE_LENGTH);
if (!isValidPacked(packed)) revert InvalidPackedData();
return Slot0.wrap(packed);
}
/// @notice Sets the context protocol fee in Slot0.
/// @param _packed Current Slot0 data
/// @param _contextProtocolFee New context protocol fee value
/// @return Updated Slot0 data
function setContextProtocolFee(
Slot0 _packed,
uint24 _contextProtocolFee
) internal pure validateProtocolFee(_contextProtocolFee) returns (Slot0) {
bytes32 packed = Slot0.unwrap(_packed).insertUint(_contextProtocolFee, CONTEXT_PROTOCOL_FEE_OFFSET, CONTEXT_PROTOCOL_FEE_LENGTH);
if (!isValidPacked(packed)) revert InvalidPackedData();
return Slot0.wrap(packed);
}
// Internal Helper Functions
/// @dev Validates that packed data contains valid values
/// @param packed The packed data to validate
/// @return True if the packed data is valid
function isValidPacked(bytes32 packed) internal pure returns (bool) {
uint8 _opType = uint8(packed.decodeUint(OPERATION_TYPE_OFFSET, OPERATION_TYPE_LENGTH));
uint24 _protocolFee = uint24(packed.decodeUint(CONTEXT_PROTOCOL_FEE_OFFSET, CONTEXT_PROTOCOL_FEE_LENGTH));
uint24 _contextFee = uint24(packed.decodeUint(CONTEXT_FEE_OFFSET, CONTEXT_FEE_LENGTH));
uint24 _overriddenFee = uint24(packed.decodeUint(OVERRIDDEN_FEE_OFFSET, OVERRIDDEN_FEE_LENGTH));
return
_opType <= MAX_OPERATION_TYPE &&
(_protocolFee & GroupFeeLibrary.REMOVE_ALL_FLAGS_MASK) <= MAX_FEE &&
(_contextFee & GroupFeeLibrary.REMOVE_ALL_FLAGS_MASK) <= MAX_FEE &&
(_overriddenFee & GroupFeeLibrary.REMOVE_ALL_FLAGS_MASK) <= MAX_OVERRIDING_FEE;
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
interface IRateProvider {
function getRate() external view returns (uint256);
function rateDecimals() external view returns (uint256);
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { Currency } from "../types/Currency.sol";
import { GroupId } from "../types/GroupId.sol";
import { GroupKey } from "../types/GroupKey.sol";
import { DTokenRegistry } from "../declarations/DTokenRegistry.sol";
import { GroupState, FeePermissions, FeeParams, DefaultFeeParams, CollateralInfo } from "../types/CommonTypes.sol";
interface ITokenRegistry {
// Events
event GroupAdded(GroupId indexed groupId);
event GroupUpdated(GroupId indexed groupId);
event CollateralAdded(GroupId indexed groupId, Currency indexed token, uint8 indexed decimals);
event CollateralRemoved(GroupId indexed groupId, Currency indexed token);
event HookContractSet(GroupId indexed groupId, address indexed newHookContract, bool indexed isDynamic, bool isDelegated);
event HookPermissionsSet(GroupId indexed groupId, uint256 indexed newPermissions);
event UpdateFeePermissions(GroupId indexed groupId, bool indexed isDynamic, bool indexed allowDelegation);
// Custom Errors
error AddressNotContract(address addr);
error NotPermitted();
error GroupNotFound(GroupId groupId);
error GroupAlreadyExists(GroupId groupId);
error InvalidDecimals();
error InvalidMinMaxCollateralAmount();
error CollateralAlreadyExists();
error MultipleNativeTokensNotAllowed();
error CollateralNotFound();
error InvalidGroupSetup();
error ZeroAddress();
error EmptyCollateralList();
error MaxCollateralsLimitReached();
error StabilityRatioTooLarge();
error stabilityConditionsTriggeringRateTooLarge();
error InvalidStabilityTriggeringRatio(GroupId groupId);
error FeesTooHigh();
error InvalidMaxFee();
error InvalidMinFee();
error InvalidBaseFee();
error InvalidYieldFee();
error InvalidStabilityFee();
error InvalidProtocolFee();
error InvalidFeeFlags();
error FeesAreNotCompatibleWithDefaultFees();
error CannotRecoverToken();
// Group management functions (Manager Role)
function addGroup(
DTokenRegistry.GroupSetup calldata setup, // Full group setup
address hookContract, // Hook contract for the group
uint256 hookPermissions // Hook permissions
) external;
// Collateral management functions (Manager Role)
function addCollateral(GroupKey calldata key, CollateralInfo calldata collateral) external;
function removeCollateral(GroupKey calldata key, Currency token) external;
// Getter functions for group state
function getGroup(GroupId groupId) external view returns (GroupState memory);
function getStabilityRatio(GroupId groupId) external view returns (uint96);
function getStabilityTriggeringRatio(GroupId groupId) external view returns (uint96);
function getProtocolFee(GroupId groupId) external view returns (uint24);
function getFeePermissions(GroupId groupId) external view returns (FeePermissions memory);
function getFeeParams(GroupId groupId) external view returns (FeeParams memory);
function getMintFeeVT(GroupId groupId) external view returns (uint24);
function getRedeemFeeVT(GroupId groupId) external view returns (uint24);
function getMintFeeYT(GroupId groupId) external view returns (uint24);
function getRedeemFeeYT(GroupId groupId) external view returns (uint24);
function getYieldFeeYT(GroupId groupId) external view returns (uint24);
function getStabilityMintFeeVT(GroupId groupId) external view returns (uint24);
function getStabilityMintFeeYT(GroupId groupId) external view returns (uint24);
function getStabilityRedeemFeeVT(GroupId groupId) external view returns (uint24);
function getStabilityRedeemFeeYT(GroupId groupId) external view returns (uint24);
function getHookContract(GroupId groupId) external view returns (address);
function getHookPermissions(GroupId groupId) external view returns (uint256);
function validateCollateral(GroupId groupId, Currency currency) external view returns (bool);
function isWrappingRequired(GroupId groupId) external view returns (bool);
function getFeeModel(GroupId groupId) external view returns (uint8);
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.9.3) (token/ERC20/utils/SafeERC20.sol)
pragma solidity ^0.8.0;
import "../IERC20Upgradeable.sol";
import "../extensions/IERC20PermitUpgradeable.sol";
import "../../../utils/AddressUpgradeable.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 SafeERC20Upgradeable {
using AddressUpgradeable for address;
/**
* @dev Transfer `value` amount of `token` from the calling contract to `to`. If `token` returns no value,
* non-reverting calls are assumed to be successful.
*/
function safeTransfer(IERC20Upgradeable token, address to, uint256 value) internal {
_callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value));
}
/**
* @dev Transfer `value` amount of `token` from `from` to `to`, spending the approval given by `from` to the
* calling contract. If `token` returns no value, non-reverting calls are assumed to be successful.
*/
function safeTransferFrom(IERC20Upgradeable 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(IERC20Upgradeable 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));
}
/**
* @dev Increase the calling contract's allowance toward `spender` by `value`. If `token` returns no value,
* non-reverting calls are assumed to be successful.
*/
function safeIncreaseAllowance(IERC20Upgradeable token, address spender, uint256 value) internal {
uint256 oldAllowance = token.allowance(address(this), spender);
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, oldAllowance + value));
}
/**
* @dev Decrease the calling contract's allowance toward `spender` by `value`. If `token` returns no value,
* non-reverting calls are assumed to be successful.
*/
function safeDecreaseAllowance(IERC20Upgradeable token, address spender, uint256 value) internal {
unchecked {
uint256 oldAllowance = token.allowance(address(this), spender);
require(oldAllowance >= value, "SafeERC20: decreased allowance below zero");
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, oldAllowance - value));
}
}
/**
* @dev Set the calling contract's allowance toward `spender` to `value`. If `token` returns no value,
* non-reverting calls are assumed to be successful. Meant to be used with tokens that require the approval
* to be set to zero before setting it to a non-zero value, such as USDT.
*/
function forceApprove(IERC20Upgradeable token, address spender, uint256 value) internal {
bytes memory approvalCall = abi.encodeWithSelector(token.approve.selector, spender, value);
if (!_callOptionalReturnBool(token, approvalCall)) {
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, 0));
_callOptionalReturn(token, approvalCall);
}
}
/**
* @dev Use a ERC-2612 signature to set the `owner` approval toward `spender` on `token`.
* Revert on invalid signature.
*/
function safePermit(
IERC20PermitUpgradeable token,
address owner,
address spender,
uint256 value,
uint256 deadline,
uint8 v,
bytes32 r,
bytes32 s
) internal {
uint256 nonceBefore = token.nonces(owner);
token.permit(owner, spender, value, deadline, v, r, s);
uint256 nonceAfter = token.nonces(owner);
require(nonceAfter == nonceBefore + 1, "SafeERC20: permit did not succeed");
}
/**
* @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement
* on the return value: the return value is optional (but if data is returned, it must not be false).
* @param token The token targeted by the call.
* @param data The call data (encoded using abi.encode or one of its variants).
*/
function _callOptionalReturn(IERC20Upgradeable 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");
require(returndata.length == 0 || abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
}
/**
* @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement
* on the return value: the return value is optional (but if data is returned, it must not be false).
* @param token The token targeted by the call.
* @param data The call data (encoded using abi.encode or one of its variants).
*
* This is a variant of {_callOptionalReturn} that silents catches all reverts and returns a bool instead.
*/
function _callOptionalReturnBool(IERC20Upgradeable token, bytes memory data) private returns (bool) {
// 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 cannot use {Address-functionCall} here since this should return false
// and not revert is the subcall reverts.
(bool success, bytes memory returndata) = address(token).call(data);
return
success && (returndata.length == 0 || abi.decode(returndata, (bool))) && AddressUpgradeable.isContract(address(token));
}
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.9.0) (utils/math/Math.sol)
pragma solidity ^0.8.0;
/**
* @dev Standard math utilities missing in the Solidity language.
*/
library MathUpgradeable {
enum Rounding {
Down, // Toward negative infinity
Up, // Toward infinity
Zero // Toward zero
}
/**
* @dev Returns the largest of two numbers.
*/
function max(uint256 a, uint256 b) internal pure returns (uint256) {
return a > b ? a : b;
}
/**
* @dev Returns the smallest of two numbers.
*/
function min(uint256 a, uint256 b) internal pure returns (uint256) {
return a < b ? a : b;
}
/**
* @dev Returns the average of two numbers. The result is rounded towards
* zero.
*/
function average(uint256 a, uint256 b) internal pure returns (uint256) {
// (a + b) / 2 can overflow.
return (a & b) + (a ^ b) / 2;
}
/**
* @dev Returns the ceiling of the division of two numbers.
*
* This differs from standard division with `/` in that it rounds up instead
* of rounding down.
*/
function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) {
// (a + b - 1) / b can overflow on addition, so we distribute.
return a == 0 ? 0 : (a - 1) / b + 1;
}
/**
* @notice Calculates floor(x * y / denominator) with full precision. Throws if result overflows a uint256 or denominator == 0
* @dev Original credit to Remco Bloemen under MIT license (https://xn--2-umb.com/21/muldiv)
* with further edits by Uniswap Labs also under MIT license.
*/
function mulDiv(uint256 x, uint256 y, uint256 denominator) internal pure returns (uint256 result) {
unchecked {
// 512-bit multiply [prod1 prod0] = x * y. Compute the product mod 2^256 and mod 2^256 - 1, then use
// use the Chinese Remainder Theorem to reconstruct the 512 bit result. The result is stored in two 256
// variables such that product = prod1 * 2^256 + prod0.
uint256 prod0; // Least significant 256 bits of the product
uint256 prod1; // Most significant 256 bits of the product
assembly {
let mm := mulmod(x, y, not(0))
prod0 := mul(x, y)
prod1 := sub(sub(mm, prod0), lt(mm, prod0))
}
// Handle non-overflow cases, 256 by 256 division.
if (prod1 == 0) {
// Solidity will revert if denominator == 0, unlike the div opcode on its own.
// The surrounding unchecked block does not change this fact.
// See https://docs.soliditylang.org/en/latest/control-structures.html#checked-or-unchecked-arithmetic.
return prod0 / denominator;
}
// Make sure the result is less than 2^256. Also prevents denominator == 0.
require(denominator > prod1, "Math: mulDiv overflow");
///////////////////////////////////////////////
// 512 by 256 division.
///////////////////////////////////////////////
// Make division exact by subtracting the remainder from [prod1 prod0].
uint256 remainder;
assembly {
// Compute remainder using mulmod.
remainder := mulmod(x, y, denominator)
// Subtract 256 bit number from 512 bit number.
prod1 := sub(prod1, gt(remainder, prod0))
prod0 := sub(prod0, remainder)
}
// Factor powers of two out of denominator and compute largest power of two divisor of denominator. Always >= 1.
// See https://cs.stackexchange.com/q/138556/92363.
// Does not overflow because the denominator cannot be zero at this stage in the function.
uint256 twos = denominator & (~denominator + 1);
assembly {
// Divide denominator by twos.
denominator := div(denominator, twos)
// Divide [prod1 prod0] by twos.
prod0 := div(prod0, twos)
// Flip twos such that it is 2^256 / twos. If twos is zero, then it becomes one.
twos := add(div(sub(0, twos), twos), 1)
}
// Shift in bits from prod1 into prod0.
prod0 |= prod1 * twos;
// Invert denominator mod 2^256. Now that denominator is an odd number, it has an inverse modulo 2^256 such
// that denominator * inv = 1 mod 2^256. Compute the inverse by starting with a seed that is correct for
// four bits. That is, denominator * inv = 1 mod 2^4.
uint256 inverse = (3 * denominator) ^ 2;
// Use the Newton-Raphson iteration to improve the precision. Thanks to Hensel's lifting lemma, this also works
// in modular arithmetic, doubling the correct bits in each step.
inverse *= 2 - denominator * inverse; // inverse mod 2^8
inverse *= 2 - denominator * inverse; // inverse mod 2^16
inverse *= 2 - denominator * inverse; // inverse mod 2^32
inverse *= 2 - denominator * inverse; // inverse mod 2^64
inverse *= 2 - denominator * inverse; // inverse mod 2^128
inverse *= 2 - denominator * inverse; // inverse mod 2^256
// Because the division is now exact we can divide by multiplying with the modular inverse of denominator.
// This will give us the correct result modulo 2^256. Since the preconditions guarantee that the outcome is
// less than 2^256, this is the final result. We don't need to compute the high bits of the result and prod1
// is no longer required.
result = prod0 * inverse;
return result;
}
}
/**
* @notice Calculates x * y / denominator with full precision, following the selected rounding direction.
*/
function mulDiv(uint256 x, uint256 y, uint256 denominator, Rounding rounding) internal pure returns (uint256) {
uint256 result = mulDiv(x, y, denominator);
if (rounding == Rounding.Up && mulmod(x, y, denominator) > 0) {
result += 1;
}
return result;
}
/**
* @dev Returns the square root of a number. If the number is not a perfect square, the value is rounded down.
*
* Inspired by Henry S. Warren, Jr.'s "Hacker's Delight" (Chapter 11).
*/
function sqrt(uint256 a) internal pure returns (uint256) {
if (a == 0) {
return 0;
}
// For our first guess, we get the biggest power of 2 which is smaller than the square root of the target.
//
// We know that the "msb" (most significant bit) of our target number `a` is a power of 2 such that we have
// `msb(a) <= a < 2*msb(a)`. This value can be written `msb(a)=2**k` with `k=log2(a)`.
//
// This can be rewritten `2**log2(a) <= a < 2**(log2(a) + 1)`
// → `sqrt(2**k) <= sqrt(a) < sqrt(2**(k+1))`
// → `2**(k/2) <= sqrt(a) < 2**((k+1)/2) <= 2**(k/2 + 1)`
//
// Consequently, `2**(log2(a) / 2)` is a good first approximation of `sqrt(a)` with at least 1 correct bit.
uint256 result = 1 << (log2(a) >> 1);
// At this point `result` is an estimation with one bit of precision. We know the true value is a uint128,
// since it is the square root of a uint256. Newton's method converges quadratically (precision doubles at
// every iteration). We thus need at most 7 iteration to turn our partial result with one bit of precision
// into the expected uint128 result.
unchecked {
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
return min(result, a / result);
}
}
/**
* @notice Calculates sqrt(a), following the selected rounding direction.
*/
function sqrt(uint256 a, Rounding rounding) internal pure returns (uint256) {
unchecked {
uint256 result = sqrt(a);
return result + (rounding == Rounding.Up && result * result < a ? 1 : 0);
}
}
/**
* @dev Return the log in base 2, rounded down, of a positive value.
* Returns 0 if given 0.
*/
function log2(uint256 value) internal pure returns (uint256) {
uint256 result = 0;
unchecked {
if (value >> 128 > 0) {
value >>= 128;
result += 128;
}
if (value >> 64 > 0) {
value >>= 64;
result += 64;
}
if (value >> 32 > 0) {
value >>= 32;
result += 32;
}
if (value >> 16 > 0) {
value >>= 16;
result += 16;
}
if (value >> 8 > 0) {
value >>= 8;
result += 8;
}
if (value >> 4 > 0) {
value >>= 4;
result += 4;
}
if (value >> 2 > 0) {
value >>= 2;
result += 2;
}
if (value >> 1 > 0) {
result += 1;
}
}
return result;
}
/**
* @dev Return the log in base 2, following the selected rounding direction, of a positive value.
* Returns 0 if given 0.
*/
function log2(uint256 value, Rounding rounding) internal pure returns (uint256) {
unchecked {
uint256 result = log2(value);
return result + (rounding == Rounding.Up && 1 << result < value ? 1 : 0);
}
}
/**
* @dev Return the log in base 10, rounded down, of a positive value.
* Returns 0 if given 0.
*/
function log10(uint256 value) internal pure returns (uint256) {
uint256 result = 0;
unchecked {
if (value >= 10 ** 64) {
value /= 10 ** 64;
result += 64;
}
if (value >= 10 ** 32) {
value /= 10 ** 32;
result += 32;
}
if (value >= 10 ** 16) {
value /= 10 ** 16;
result += 16;
}
if (value >= 10 ** 8) {
value /= 10 ** 8;
result += 8;
}
if (value >= 10 ** 4) {
value /= 10 ** 4;
result += 4;
}
if (value >= 10 ** 2) {
value /= 10 ** 2;
result += 2;
}
if (value >= 10 ** 1) {
result += 1;
}
}
return result;
}
/**
* @dev Return the log in base 10, following the selected rounding direction, of a positive value.
* Returns 0 if given 0.
*/
function log10(uint256 value, Rounding rounding) internal pure returns (uint256) {
unchecked {
uint256 result = log10(value);
return result + (rounding == Rounding.Up && 10 ** result < value ? 1 : 0);
}
}
/**
* @dev Return the log in base 256, rounded down, of a positive value.
* Returns 0 if given 0.
*
* Adding one to the result gives the number of pairs of hex symbols needed to represent `value` as a hex string.
*/
function log256(uint256 value) internal pure returns (uint256) {
uint256 result = 0;
unchecked {
if (value >> 128 > 0) {
value >>= 128;
result += 16;
}
if (value >> 64 > 0) {
value >>= 64;
result += 8;
}
if (value >> 32 > 0) {
value >>= 32;
result += 4;
}
if (value >> 16 > 0) {
value >>= 16;
result += 2;
}
if (value >> 8 > 0) {
result += 1;
}
}
return result;
}
/**
* @dev Return the log in base 256, following the selected rounding direction, of a positive value.
* Returns 0 if given 0.
*/
function log256(uint256 value, Rounding rounding) internal pure returns (uint256) {
unchecked {
uint256 result = log256(value);
return result + (rounding == Rounding.Up && 1 << (result << 3) < value ? 1 : 0);
}
}
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.8.0) (utils/math/SignedMath.sol)
pragma solidity ^0.8.0;
/**
* @dev Standard signed math utilities missing in the Solidity language.
*/
library SignedMathUpgradeable {
/**
* @dev Returns the largest of two signed numbers.
*/
function max(int256 a, int256 b) internal pure returns (int256) {
return a > b ? a : b;
}
/**
* @dev Returns the smallest of two signed numbers.
*/
function min(int256 a, int256 b) internal pure returns (int256) {
return a < b ? a : b;
}
/**
* @dev Returns the average of two signed numbers without overflow.
* The result is rounded towards zero.
*/
function average(int256 a, int256 b) internal pure returns (int256) {
// Formula from the book "Hacker's Delight"
int256 x = (a & b) + ((a ^ b) >> 1);
return x + (int256(uint256(x) >> 255) & (a ^ b));
}
/**
* @dev Returns the absolute unsigned value of a signed value.
*/
function abs(int256 n) internal pure returns (uint256) {
unchecked {
// must be unchecked in order to support `n = type(int256).min`
return uint256(n >= 0 ? n : -n);
}
}
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (utils/introspection/IERC165.sol)
pragma solidity ^0.8.0;
/**
* @dev Interface of the ERC165 standard, as defined in the
* https://eips.ethereum.org/EIPS/eip-165[EIP].
*
* Implementers can declare support of contract interfaces, which can then be
* queried by others ({ERC165Checker}).
*
* For an implementation, see {ERC165}.
*/
interface IERC165Upgradeable {
/**
* @dev Returns true if this contract implements the interface defined by
* `interfaceId`. See the corresponding
* https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section]
* to learn more about how these ids are created.
*
* This function call must use less than 30 000 gas.
*/
function supportsInterface(bytes4 interfaceId) external view returns (bool);
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.9.0) (utils/Address.sol)
pragma solidity ^0.8.1;
/**
* @dev Collection of functions related to the address type
*/
library AddressUpgradeable {
/**
* @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
*
* Furthermore, `isContract` will also return true if the target contract within
* the same transaction is already scheduled for destruction by `SELFDESTRUCT`,
* which only has an effect at the end of a transaction.
* ====
*
* [IMPORTANT]
* ====
* You shouldn't rely on `isContract` to protect against flash loan attacks!
*
* Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets
* like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract
* constructor.
* ====
*/
function isContract(address account) internal view returns (bool) {
// This method relies on extcodesize/address.code.length, which returns 0
// for contracts in construction, since the code is only stored at the end
// of the constructor execution.
return account.code.length > 0;
}
/**
* @dev Replacement for Solidity's `transfer`: sends `amount` wei to
* `recipient`, forwarding all available gas and reverting on errors.
*
* https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
* of certain opcodes, possibly making contracts go over the 2300 gas limit
* imposed by `transfer`, making them unable to receive funds via
* `transfer`. {sendValue} removes this limitation.
*
* https://consensys.net/diligence/blog/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.8.0/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
*/
function sendValue(address payable recipient, uint256 amount) internal {
require(address(this).balance >= amount, "Address: insufficient balance");
(bool success, ) = recipient.call{value: amount}("");
require(success, "Address: unable to send value, recipient may have reverted");
}
/**
* @dev Performs a Solidity function call using a low level `call`. A
* plain `call` is an unsafe replacement for a function call: use this
* function instead.
*
* If `target` reverts with a revert reason, it is bubbled up by this
* function (like regular Solidity function calls).
*
* Returns the raw returned data. To convert to the expected return value,
* use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
*
* Requirements:
*
* - `target` must be a contract.
* - calling `target` with `data` must not revert.
*
* _Available since v3.1._
*/
function functionCall(address target, bytes memory data) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, "Address: low-level call failed");
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
* `errorMessage` as a fallback revert reason when `target` reverts.
*
* _Available since v3.1._
*/
function functionCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, errorMessage);
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
* but also transferring `value` wei to `target`.
*
* Requirements:
*
* - the calling contract must have an ETH balance of at least `value`.
* - the called Solidity function must be `payable`.
*
* _Available since v3.1._
*/
function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
}
/**
* @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
* with `errorMessage` as a fallback revert reason when `target` reverts.
*
* _Available since v3.1._
*/
function functionCallWithValue(
address target,
bytes memory data,
uint256 value,
string memory errorMessage
) internal returns (bytes memory) {
require(address(this).balance >= value, "Address: insufficient balance for call");
(bool success, bytes memory returndata) = target.call{value: value}(data);
return verifyCallResultFromTarget(target, success, returndata, errorMessage);
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
* but performing a static call.
*
* _Available since v3.3._
*/
function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
return functionStaticCall(target, data, "Address: low-level static call failed");
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
* but performing a static call.
*
* _Available since v3.3._
*/
function functionStaticCall(
address target,
bytes memory data,
string memory errorMessage
) internal view returns (bytes memory) {
(bool success, bytes memory returndata) = target.staticcall(data);
return verifyCallResultFromTarget(target, success, returndata, errorMessage);
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
* but performing a delegate call.
*
* _Available since v3.4._
*/
function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
return functionDelegateCall(target, data, "Address: low-level delegate call failed");
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
* but performing a delegate call.
*
* _Available since v3.4._
*/
function functionDelegateCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
(bool success, bytes memory returndata) = target.delegatecall(data);
return verifyCallResultFromTarget(target, success, returndata, errorMessage);
}
/**
* @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling
* the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract.
*
* _Available since v4.8._
*/
function verifyCallResultFromTarget(
address target,
bool success,
bytes memory returndata,
string memory errorMessage
) internal view returns (bytes memory) {
if (success) {
if (returndata.length == 0) {
// only check isContract if the call was successful and the return data is empty
// otherwise we already know that it was a contract
require(isContract(target), "Address: call to non-contract");
}
return returndata;
} else {
_revert(returndata, errorMessage);
}
}
/**
* @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the
* revert reason or using the provided one.
*
* _Available since v4.3._
*/
function verifyCallResult(
bool success,
bytes memory returndata,
string memory errorMessage
) internal pure returns (bytes memory) {
if (success) {
return returndata;
} else {
_revert(returndata, errorMessage);
}
}
function _revert(bytes memory returndata, string memory errorMessage) private pure {
// Look for revert reason and bubble it up if present
if (returndata.length > 0) {
// The easiest way to bubble the revert reason is using memory via assembly
/// @solidity memory-safe-assembly
assembly {
let returndata_size := mload(returndata)
revert(add(32, returndata), returndata_size)
}
} else {
revert(errorMessage);
}
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { ExponentialMovingAverageV8 } from "../libs/math/ExponentialMovingAverageV8.sol";
library DTreasury {
using ExponentialMovingAverageV8 for ExponentialMovingAverageV8.EMAStorage;
struct GroupUpdateParams {
uint256 baseTokenCaps;
uint256 baseIn;
bool beta;
}
struct TreasuryState {
ExponentialMovingAverageV8.EMAStorage emaLeverageRatio;
uint256 totalBaseTokens;
uint256 baseTokenCaps;
uint256 lastSettlementTimestamp;
uint256 baseTokenPrice;
bool inited;
uint256 strategyUnderlying;
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { Currency, CurrencyLibrary } from "../types/Currency.sol";
import { GroupStateHelper, GroupSettings } from "../types/GroupStateHelper.sol";
import { GroupState, FeeModel } from "../types/CommonTypes.sol";
import { GroupId } from "../types/GroupId.sol";
import { DTreasury } from "../declarations/DTreasury.sol";
import { TreasuryStateLibrary } from "./TreasuryStateLibrary.sol";
import { IDXToken } from "../interfaces/IDXToken.sol";
import { IAToken } from "../interfaces/IAToken.sol";
import { IWToken } from "../interfaces/IWToken.sol";
import { IRebalancePool } from "../interfaces/IRebalancePool.sol";
import { CustomRevert } from "../libs/CustomRevert.sol";
import { FullMath } from "../libs/math/FullMath.sol";
import { IProtocolMinimum as IProtocol } from "../interfaces/IProtocolMinimum.sol";
/**
* @title TreasuryHarvestLibrary
* @notice This library orchestrates fee and yield harvesting for a specific group.
* @dev It handles collecting fees, distributing base tokens, minting aTokens, and managing
* yield fees. The library relies on external calls to tokens and Rebalance Pools, and
* carefully handles potential revert scenarios via `try/catch` and bubble-up reverts.
*/
library TreasuryHarvestLibrary {
using CurrencyLibrary for Currency;
using GroupStateHelper for GroupSettings;
using TreasuryStateLibrary for *;
using CustomRevert for bytes4;
// =========================
// ======== Errors =========
// =========================
error ZeroAmount();
error WrongFeeModel(GroupId groupId);
error ErrorDistributingYield();
error ErrorFeeCollectionFailed(GroupId groupId);
error ErrorDistributingTokens(GroupId groupId);
// =========================
// ======== Events =========
// =========================
event FeesCollected(GroupId indexed groupId, uint256 amount);
event FeesHarvested(GroupId indexed groupId, uint256 dxTokenBalance, address feeCollector);
event YieldHarvested(GroupId indexed groupId, uint256 yieldAmount, address feeCollector);
event ATokensMintedToPool(GroupId indexed groupId, uint256 aTokenAmount, address recipient);
event TokensSwappedToPool(GroupId indexed groupId, address fromToken, address toToken, uint256 amountIn, uint256 amountOut);
// =========================
// ======== Structs ========
// =========================
/**
* @notice Parameters used to harvest yield or fees.
* @param sendTokens If true, will swap base tokens into stablecoins
* instead of attempting to mint aTokens (when mint capacity is exceeded).
* @param swapRouter Address of the DEX/router used to swap base tokens for stablecoins.
* @param stablecoin The address of the stablecoin into which base tokens may be swapped.
* @param minAmountOut The minimum amount of stablecoins that must be received from the swap
* for the transaction not to revert.
*/
struct HarvestParams {
bool sendTokens;
address swapRouter;
address stablecoin;
uint256 minAmountOut;
}
/**
* @notice Collects any accrued fees from dxToken.
* @dev Calls `collectFees` on the dxToken. Reverts if no fees were collected.
* @param groupState The current group state, which includes the dxToken address.
* @return The amount of dxToken fees collected.
*/
function collectFees(GroupState memory groupState, GroupId groupId) internal returns (uint256) {
IDXToken dxToken = IDXToken(groupState.core.xToken.toAddress());
uint256 dxTokenBalanceBefore = dxToken.balanceOf(address(this));
// Attempt to collect fees from dxToken
try dxToken.collectFees() {
// Get the balance of dxToken after fees have been collected
uint256 dxTokenBalanceAfter = dxToken.balanceOf(address(this));
uint256 collectedAmount = dxTokenBalanceAfter - dxTokenBalanceBefore;
// Revert if no fees were actually collected
if (collectedAmount == 0) ZeroAmount.selector.revertWith();
// Emit an event to log the fees collected
emit FeesCollected(groupId, collectedAmount);
return collectedAmount;
} catch {
CustomRevert.bubbleUpAndRevertWith(dxToken.collectFees.selector, address(dxToken));
}
}
/**
* @notice Harvests fees (not yield) based on the group's fee model.
* @dev For management fees, the dxTokens are simply transferred to the feeCollector.
* For funding fees, dxTokens are burned and then aTokens are minted (or stablecoins are swapped).
* @param groupId The ID of the group for which fees are harvested.
* @param groupState The current group state data.
* @param treasuryState The TreasuryState storage reference for updating internal state.
* @param harvestParams Parameters controlling how the harvested tokens might be swapped or minted.
* @param dxTokenBalance The amount of dxTokens available for fee harvesting.
* @param feeCollector The address to which management fees should be sent.
* @param protocol The Protocol contract address (for calculating stability ratio, etc).
*/
function harvestFees(
GroupId groupId,
GroupState memory groupState,
DTreasury.TreasuryState storage treasuryState,
HarvestParams memory harvestParams,
uint256 dxTokenBalance,
address feeCollector,
address protocol
) internal {
IDXToken dxToken = IDXToken(groupState.core.xToken.toAddress());
// Determine the fee model for this group
FeeModel feeModel = dxToken.getFeeModel();
if (feeModel == FeeModel.MANAGEMENT_FEE) {
// Simply transfer dxTokens to fee collector
groupState.core.xToken.safeTransfer(feeCollector, dxTokenBalance);
return;
}
if (feeModel == FeeModel.VARIABLE_FUNDING_FEE || feeModel == FeeModel.FIXED_FUNDING_FEE) {
// Calculate baseTokenAmount from dxTokens
uint256 baseTokenAmount = dxToken.dxTokenToBaseToken(dxTokenBalance);
uint256 feeAmount;
// For funding fees, burn dxTokens
try dxToken.burn(address(this), dxTokenBalance) {} catch {
CustomRevert.bubbleUpAndRevertWith(dxToken.burn.selector, address(dxToken));
}
if (feeModel == FeeModel.VARIABLE_FUNDING_FEE) {
// apply yield and get net harvestable amount
(baseTokenAmount, feeAmount) = _applyYield(groupState, baseTokenAmount, feeCollector);
}
// Distribute base tokens either as aToken or stablecoin, depending on mint capacity
treasuryState.totalBaseTokens -= baseTokenAmount + feeAmount; /// this is added because of size constraints (but we don't need to add basetokens)
_distributeBaseTokens(
groupState,
groupId,
treasuryState,
baseTokenAmount,
harvestParams,
protocol
);
emit FeesHarvested(groupId, dxTokenBalance, feeCollector);
return;
}
// If none of the recognized fee models applies, revert.
WrongFeeModel.selector.revertWith(groupId);
}
/**
* @notice Harvests yield for a group, applies any yield fees, and distributes the net amount.
* @dev A portion of the yield is sent to the feeCollector as a yield fee (if applicable).
* @param groupId The ID of the group for which yield is being harvested.
* @param groupState The current group state data.
* @param treasuryState The TreasuryState storage reference for updating internal state.
* @param harvestParams Parameters controlling how the base tokens might be swapped or minted.
* @param harvestableAmount The total base token amount of yield harvested.
* @param feeCollector The address to which the yield fee is sent.
* @param protocol The Protocol contract address (for calculating stability ratio, etc).
*/
function harvestYield(
GroupId groupId,
GroupState memory groupState,
DTreasury.TreasuryState storage treasuryState,
HarvestParams memory harvestParams,
uint256 harvestableAmount,
address feeCollector,
address protocol
) internal {
// Apply yield fees and get net harvestable amount
(harvestableAmount, ) = _applyYield(groupState, harvestableAmount, feeCollector);
// Distribute the net base tokens (either as stablecoin or aTokens)
_distributeBaseTokens(
groupState,
groupId,
treasuryState,
harvestableAmount,
harvestParams,
protocol
);
emit YieldHarvested(groupId, harvestableAmount, feeCollector);
}
/**
* @notice Calculates how many aTokens should be minted given a certain amount of base tokens,
* and also how many aTokens can be minted at maximum without breaching collateral targets.
* @param groupState The current group state data.
* @param groupId The ID of the group for which aTokens will be minted.
* @param baseTokenAmountNormalized The normalized amount of base tokens to convert into aTokens.
* @param aToken The IAToken contract reference (whose NAV is used to compute minted supply).
* @param protocol The Protocol contract address (for stability ratio checks).
* @return aTokenToMint How many aTokens should be minted given the current NAV.
* @return aTokenMintableMax The maximum number of aTokens that can be minted without breaching collateral.
*/
function calculateMintingParams(
DTreasury.TreasuryState storage treasuryState,
GroupState memory groupState,
GroupId groupId,
uint256 baseTokenAmountNormalized,
IAToken aToken,
address protocol
) internal view returns (uint256 aTokenToMint, uint256 aTokenMintableMax) {
// Fetch the current base token price from an oracle or similar feed
(, uint256 newPrice) = TreasuryStateLibrary.fetchBaseTokenPrice(treasuryState, groupState);
// NAV of the aToken is used to determine how many aTokens 1 base token is worth
uint256 aTokenNav = aToken.nav();
// aTokenToMint = (baseTokenAmountNormalized * newPrice) / aTokenNav
aTokenToMint = FullMath.mulDiv(baseTokenAmountNormalized, newPrice, aTokenNav);
// Check how many aTokens we can mint before breaching collateral/stability constraints
(, aTokenMintableMax) = TreasuryStateLibrary.maxMintableAToken(
treasuryState,
groupState,
uint256(IProtocol(protocol).stabilityRatio(groupId))
);
}
/**
* @notice Distributes base tokens to the Rebalance Pool either by minting aTokens or swapping
* them into stablecoins, depending on the mint capacity and user preference.
* @dev If the aToken mint capacity is insufficient and `sendTokens == true`, base tokens
* are swapped into stablecoins. Otherwise, the function reverts.
* @param groupState The current group state data.
* @param groupId The ID of the group for which distribution is happening.
* @param treasuryState The TreasuryState storage reference for updating internal state.
* @param baseTokenAmount The actual (denormalized) amount of base tokens.
* @param harvestParams Struct containing swap and stablecoin parameters.
* @param protocol The Protocol contract address (for stability ratio checks).
*/
function _distributeBaseTokens(
GroupState memory groupState,
GroupId groupId,
DTreasury.TreasuryState storage treasuryState,
uint256 baseTokenAmount,
HarvestParams memory harvestParams,
address protocol
) private {
IAToken aToken = IAToken(groupState.core.aToken.toAddress());
Currency stableCoinCurrency = Currency.wrap(harvestParams.stablecoin);
address rPool = groupState.extended.rebalancePool.toAddress();
// Normalize the base token amount for internal accounting
uint8 baseTokenDecimals = GroupSettings.wrap(groupState.groupSettings).getBaseTokenDecimals();
uint256 baseTokenAmountNormalized = TreasuryStateLibrary.normalizeDecimals(
baseTokenAmount,
baseTokenDecimals
);
// Compute how many aTokens we *need* to mint and how many are *allowed* to mint
(uint256 aTokenToMint, uint256 aTokenMintableMax) = calculateMintingParams(
treasuryState,
groupState,
groupId,
baseTokenAmountNormalized,
aToken,
protocol
);
// If we cannot mint the entire required amount of aTokens
if (aTokenMintableMax < aTokenToMint) {
// If we are NOT allowed to send stablecoins, revert
if (!harvestParams.sendTokens) ErrorDistributingTokens.selector.revertWith(groupId);
// Decrease the treasury's totalBaseTokens because we'll unwrap and swap them
treasuryState.totalBaseTokens -= baseTokenAmountNormalized;
// 1. Unwrap the wrapped base tokens into their underlying (e.g., WETH -> ETH or WMATIC -> MATIC).
uint256 unwrappedBaseTokens = IWToken(groupState.core.baseToken.toAddress()).unwrap(baseTokenAmount);
// 2. Swap from base tokens -> stablecoins
uint256 stablecoinAmount = TreasuryStateLibrary.swapTokens(
address(this),
harvestParams.swapRouter,
// We should swap from the *yield-bearing token* (not the base token)
groupState.core.yieldBearingToken.toAddress(),
stableCoinCurrency.toAddress(),
unwrappedBaseTokens,
harvestParams.minAmountOut
);
// 3. Transfer the stablecoins to the Rebalance Pool
stableCoinCurrency.safeTransfer(
rPool,
stablecoinAmount
);
emit TokensSwappedToPool(groupId, groupState.core.baseToken.toAddress(), harvestParams.stablecoin, unwrappedBaseTokens, stablecoinAmount);
} else {
// We can mint the full aToken amount from the base tokens
// Increase the treasury's totalBaseTokens accordingly
treasuryState.totalBaseTokens += baseTokenAmountNormalized;
// Convert aTokenToMint from normalized (18 decimals) to the actual aToken decimals
uint8 aTokenDecimals = GroupSettings.wrap(groupState.groupSettings).getATokenDecimals();
uint256 aTokenToMintDenorm = TreasuryStateLibrary.denormalizeDecimals(aTokenToMint, aTokenDecimals);
// Mint aTokens directly to the Rebalance Pool
aToken.mint(rPool, aTokenToMintDenorm);
// Update the Rebalance Pool’s NAV to reflect the newly minted aTokens
IRebalancePool(rPool).updateNAV();
emit ATokensMintedToPool(groupId, aTokenToMintDenorm, rPool);
}
}
/**
* @notice Applies a yield fee to the harvestable amount (if any) and transfers that fee
* portion to the feeCollector. Returns the net amount after fee.
* @param groupState The current group state data (for accessing baseToken).
* @param harvestableAmount The total yield (in base tokens) to be distributed.
* @param feeCollector The address to which yield fees should be sent.
* @return _harvestableAmount The net (post-fee) amount of base tokens.
* @return _feeAmount The fee amount that was deducted and sent to the feeCollector.
*/
function _applyYield(
GroupState memory groupState,
uint256 harvestableAmount,
address feeCollector
)
private
returns (uint256 _harvestableAmount, uint256 _feeAmount)
{
IRebalancePool rebalancePool = IRebalancePool(groupState.extended.rebalancePool.toAddress());
uint256 yieldFeePercentage = rebalancePool.yieldFeePercentage();
uint256 BASE_POINTS = 10_000;
// If there's a yield fee, calculate and transfer it
if (yieldFeePercentage > 0) {
_feeAmount = (harvestableAmount * yieldFeePercentage) / BASE_POINTS;
_harvestableAmount = harvestableAmount - _feeAmount;
if (_feeAmount > 0) {
groupState.core.baseToken.safeTransfer(feeCollector, _feeAmount);
}
} else {
// If no yield fee, the entire amount is harvestable
_harvestableAmount = harvestableAmount;
}
return (_harvestableAmount, _feeAmount);
}
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/extensions/ERC4626.sol)
pragma solidity ^0.8.0;
import "../ERC20Upgradeable.sol";
import "../utils/SafeERC20Upgradeable.sol";
import "../../../interfaces/IERC4626Upgradeable.sol";
import "../../../utils/math/MathUpgradeable.sol";
import {Initializable} from "../../../proxy/utils/Initializable.sol";
/**
* @dev Implementation of the ERC4626 "Tokenized Vault Standard" as defined in
* https://eips.ethereum.org/EIPS/eip-4626[EIP-4626].
*
* This extension allows the minting and burning of "shares" (represented using the ERC20 inheritance) in exchange for
* underlying "assets" through standardized {deposit}, {mint}, {redeem} and {burn} workflows. This contract extends
* the ERC20 standard. Any additional extensions included along it would affect the "shares" token represented by this
* contract and not the "assets" token which is an independent contract.
*
* [CAUTION]
* ====
* In empty (or nearly empty) ERC-4626 vaults, deposits are at high risk of being stolen through frontrunning
* with a "donation" to the vault that inflates the price of a share. This is variously known as a donation or inflation
* attack and is essentially a problem of slippage. Vault deployers can protect against this attack by making an initial
* deposit of a non-trivial amount of the asset, such that price manipulation becomes infeasible. Withdrawals may
* similarly be affected by slippage. Users can protect against this attack as well as unexpected slippage in general by
* verifying the amount received is as expected, using a wrapper that performs these checks such as
* https://github.com/fei-protocol/ERC4626#erc4626router-and-base[ERC4626Router].
*
* Since v4.9, this implementation uses virtual assets and shares to mitigate that risk. The `_decimalsOffset()`
* corresponds to an offset in the decimal representation between the underlying asset's decimals and the vault
* decimals. This offset also determines the rate of virtual shares to virtual assets in the vault, which itself
* determines the initial exchange rate. While not fully preventing the attack, analysis shows that the default offset
* (0) makes it non-profitable, as a result of the value being captured by the virtual shares (out of the attacker's
* donation) matching the attacker's expected gains. With a larger offset, the attack becomes orders of magnitude more
* expensive than it is profitable. More details about the underlying math can be found
* xref:erc4626.adoc#inflation-attack[here].
*
* The drawback of this approach is that the virtual shares do capture (a very small) part of the value being accrued
* to the vault. Also, if the vault experiences losses, the users try to exit the vault, the virtual shares and assets
* will cause the first user to exit to experience reduced losses in detriment to the last users that will experience
* bigger losses. Developers willing to revert back to the pre-v4.9 behavior just need to override the
* `_convertToShares` and `_convertToAssets` functions.
*
* To learn more, check out our xref:ROOT:erc4626.adoc[ERC-4626 guide].
* ====
*
* _Available since v4.7._
*/
abstract contract ERC4626Upgradeable is Initializable, ERC20Upgradeable, IERC4626Upgradeable {
using MathUpgradeable for uint256;
IERC20Upgradeable private _asset;
uint8 private _underlyingDecimals;
/**
* @dev Set the underlying asset contract. This must be an ERC20-compatible contract (ERC20 or ERC777).
*/
function __ERC4626_init(IERC20Upgradeable asset_) internal onlyInitializing {
__ERC4626_init_unchained(asset_);
}
function __ERC4626_init_unchained(IERC20Upgradeable asset_) internal onlyInitializing {
(bool success, uint8 assetDecimals) = _tryGetAssetDecimals(asset_);
_underlyingDecimals = success ? assetDecimals : 18;
_asset = asset_;
}
/**
* @dev Attempts to fetch the asset decimals. A return value of false indicates that the attempt failed in some way.
*/
function _tryGetAssetDecimals(IERC20Upgradeable asset_) private view returns (bool, uint8) {
(bool success, bytes memory encodedDecimals) = address(asset_).staticcall(
abi.encodeWithSelector(IERC20MetadataUpgradeable.decimals.selector)
);
if (success && encodedDecimals.length >= 32) {
uint256 returnedDecimals = abi.decode(encodedDecimals, (uint256));
if (returnedDecimals <= type(uint8).max) {
return (true, uint8(returnedDecimals));
}
}
return (false, 0);
}
/**
* @dev Decimals are computed by adding the decimal offset on top of the underlying asset's decimals. This
* "original" value is cached during construction of the vault contract. If this read operation fails (e.g., the
* asset has not been created yet), a default of 18 is used to represent the underlying asset's decimals.
*
* See {IERC20Metadata-decimals}.
*/
function decimals() public view virtual override(IERC20MetadataUpgradeable, ERC20Upgradeable) returns (uint8) {
return _underlyingDecimals + _decimalsOffset();
}
/** @dev See {IERC4626-asset}. */
function asset() public view virtual override returns (address) {
return address(_asset);
}
/** @dev See {IERC4626-totalAssets}. */
function totalAssets() public view virtual override returns (uint256) {
return _asset.balanceOf(address(this));
}
/** @dev See {IERC4626-convertToShares}. */
function convertToShares(uint256 assets) public view virtual override returns (uint256) {
return _convertToShares(assets, MathUpgradeable.Rounding.Down);
}
/** @dev See {IERC4626-convertToAssets}. */
function convertToAssets(uint256 shares) public view virtual override returns (uint256) {
return _convertToAssets(shares, MathUpgradeable.Rounding.Down);
}
/** @dev See {IERC4626-maxDeposit}. */
function maxDeposit(address) public view virtual override returns (uint256) {
return type(uint256).max;
}
/** @dev See {IERC4626-maxMint}. */
function maxMint(address) public view virtual override returns (uint256) {
return type(uint256).max;
}
/** @dev See {IERC4626-maxWithdraw}. */
function maxWithdraw(address owner) public view virtual override returns (uint256) {
return _convertToAssets(balanceOf(owner), MathUpgradeable.Rounding.Down);
}
/** @dev See {IERC4626-maxRedeem}. */
function maxRedeem(address owner) public view virtual override returns (uint256) {
return balanceOf(owner);
}
/** @dev See {IERC4626-previewDeposit}. */
function previewDeposit(uint256 assets) public view virtual override returns (uint256) {
return _convertToShares(assets, MathUpgradeable.Rounding.Down);
}
/** @dev See {IERC4626-previewMint}. */
function previewMint(uint256 shares) public view virtual override returns (uint256) {
return _convertToAssets(shares, MathUpgradeable.Rounding.Up);
}
/** @dev See {IERC4626-previewWithdraw}. */
function previewWithdraw(uint256 assets) public view virtual override returns (uint256) {
return _convertToShares(assets, MathUpgradeable.Rounding.Up);
}
/** @dev See {IERC4626-previewRedeem}. */
function previewRedeem(uint256 shares) public view virtual override returns (uint256) {
return _convertToAssets(shares, MathUpgradeable.Rounding.Down);
}
/** @dev See {IERC4626-deposit}. */
function deposit(uint256 assets, address receiver) public virtual override returns (uint256) {
require(assets <= maxDeposit(receiver), "ERC4626: deposit more than max");
uint256 shares = previewDeposit(assets);
_deposit(_msgSender(), receiver, assets, shares);
return shares;
}
/** @dev See {IERC4626-mint}.
*
* As opposed to {deposit}, minting is allowed even if the vault is in a state where the price of a share is zero.
* In this case, the shares will be minted without requiring any assets to be deposited.
*/
function mint(uint256 shares, address receiver) public virtual override returns (uint256) {
require(shares <= maxMint(receiver), "ERC4626: mint more than max");
uint256 assets = previewMint(shares);
_deposit(_msgSender(), receiver, assets, shares);
return assets;
}
/** @dev See {IERC4626-withdraw}. */
function withdraw(uint256 assets, address receiver, address owner) public virtual override returns (uint256) {
require(assets <= maxWithdraw(owner), "ERC4626: withdraw more than max");
uint256 shares = previewWithdraw(assets);
_withdraw(_msgSender(), receiver, owner, assets, shares);
return shares;
}
/** @dev See {IERC4626-redeem}. */
function redeem(uint256 shares, address receiver, address owner) public virtual override returns (uint256) {
require(shares <= maxRedeem(owner), "ERC4626: redeem more than max");
uint256 assets = previewRedeem(shares);
_withdraw(_msgSender(), receiver, owner, assets, shares);
return assets;
}
/**
* @dev Internal conversion function (from assets to shares) with support for rounding direction.
*/
function _convertToShares(uint256 assets, MathUpgradeable.Rounding rounding) internal view virtual returns (uint256) {
return assets.mulDiv(totalSupply() + 10 ** _decimalsOffset(), totalAssets() + 1, rounding);
}
/**
* @dev Internal conversion function (from shares to assets) with support for rounding direction.
*/
function _convertToAssets(uint256 shares, MathUpgradeable.Rounding rounding) internal view virtual returns (uint256) {
return shares.mulDiv(totalAssets() + 1, totalSupply() + 10 ** _decimalsOffset(), rounding);
}
/**
* @dev Deposit/mint common workflow.
*/
function _deposit(address caller, address receiver, uint256 assets, uint256 shares) internal virtual {
// If _asset is ERC777, `transferFrom` can trigger a reentrancy BEFORE the transfer happens through the
// `tokensToSend` hook. On the other hand, the `tokenReceived` hook, that is triggered after the transfer,
// calls the vault, which is assumed not malicious.
//
// Conclusion: we need to do the transfer before we mint so that any reentrancy would happen before the
// assets are transferred and before the shares are minted, which is a valid state.
// slither-disable-next-line reentrancy-no-eth
SafeERC20Upgradeable.safeTransferFrom(_asset, caller, address(this), assets);
_mint(receiver, shares);
emit Deposit(caller, receiver, assets, shares);
}
/**
* @dev Withdraw/redeem common workflow.
*/
function _withdraw(
address caller,
address receiver,
address owner,
uint256 assets,
uint256 shares
) internal virtual {
if (caller != owner) {
_spendAllowance(owner, caller, shares);
}
// If _asset is ERC777, `transfer` can trigger a reentrancy AFTER the transfer happens through the
// `tokensReceived` hook. On the other hand, the `tokensToSend` hook, that is triggered before the transfer,
// calls the vault, which is assumed not malicious.
//
// Conclusion: we need to do the transfer after the burn so that any reentrancy would happen after the
// shares are burned and after the assets are transferred, which is a valid state.
_burn(owner, shares);
SafeERC20Upgradeable.safeTransfer(_asset, receiver, assets);
emit Withdraw(caller, receiver, owner, assets, shares);
}
function _decimalsOffset() internal view virtual returns (uint8) {
return 0;
}
/**
* @dev This empty reserved space is put in place to allow future versions to add new
* variables without shifting down storage in the inheritance chain.
* See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
*/
uint256[49] private __gap;
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/IERC20.sol)
pragma solidity ^0.8.0;
/**
* @dev Interface of the ERC20 standard as defined in the EIP.
*/
interface IERC20Upgradeable {
/**
* @dev Emitted when `value` tokens are moved from one account (`from`) to
* another (`to`).
*
* Note that `value` may be zero.
*/
event Transfer(address indexed from, address indexed to, uint256 value);
/**
* @dev Emitted when the allowance of a `spender` for an `owner` is set by
* a call to {approve}. `value` is the new allowance.
*/
event Approval(address indexed owner, address indexed spender, uint256 value);
/**
* @dev Returns the amount of tokens in existence.
*/
function totalSupply() external view returns (uint256);
/**
* @dev Returns the amount of tokens owned by `account`.
*/
function balanceOf(address account) external view returns (uint256);
/**
* @dev Moves `amount` tokens from the caller's account to `to`.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* Emits a {Transfer} event.
*/
function transfer(address to, uint256 amount) external returns (bool);
/**
* @dev Returns the remaining number of tokens that `spender` will be
* allowed to spend on behalf of `owner` through {transferFrom}. This is
* zero by default.
*
* This value changes when {approve} or {transferFrom} are called.
*/
function allowance(address owner, address spender) external view returns (uint256);
/**
* @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* IMPORTANT: Beware that changing an allowance with this method brings the risk
* that someone may use both the old and the new allowance by unfortunate
* transaction ordering. One possible solution to mitigate this race
* condition is to first reduce the spender's allowance to 0 and set the
* desired value afterwards:
* https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
*
* Emits an {Approval} event.
*/
function approve(address spender, uint256 amount) external returns (bool);
/**
* @dev Moves `amount` tokens from `from` to `to` using the
* allowance mechanism. `amount` is then deducted from the caller's
* allowance.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* Emits a {Transfer} event.
*/
function transferFrom(address from, address to, uint256 amount) external returns (bool);
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.9.4) (token/ERC20/extensions/IERC20Permit.sol)
pragma solidity ^0.8.0;
/**
* @dev Interface of the ERC20 Permit extension allowing approvals to be made via signatures, as defined in
* https://eips.ethereum.org/EIPS/eip-2612[EIP-2612].
*
* Adds the {permit} method, which can be used to change an account's ERC20 allowance (see {IERC20-allowance}) by
* presenting a message signed by the account. By not relying on {IERC20-approve}, the token holder account doesn't
* need to send a transaction, and thus is not required to hold Ether at all.
*
* ==== Security Considerations
*
* There are two important considerations concerning the use of `permit`. The first is that a valid permit signature
* expresses an allowance, and it should not be assumed to convey additional meaning. In particular, it should not be
* considered as an intention to spend the allowance in any specific way. The second is that because permits have
* built-in replay protection and can be submitted by anyone, they can be frontrun. A protocol that uses permits should
* take this into consideration and allow a `permit` call to fail. Combining these two aspects, a pattern that may be
* generally recommended is:
*
* ```solidity
* function doThingWithPermit(..., uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s) public {
* try token.permit(msg.sender, address(this), value, deadline, v, r, s) {} catch {}
* doThing(..., value);
* }
*
* function doThing(..., uint256 value) public {
* token.safeTransferFrom(msg.sender, address(this), value);
* ...
* }
* ```
*
* Observe that: 1) `msg.sender` is used as the owner, leaving no ambiguity as to the signer intent, and 2) the use of
* `try/catch` allows the permit to fail and makes the code tolerant to frontrunning. (See also
* {SafeERC20-safeTransferFrom}).
*
* Additionally, note that smart contract wallets (such as Argent or Safe) are not able to produce permit signatures, so
* contracts should have entry points that don't rely on permit.
*/
interface IERC20PermitUpgradeable {
/**
* @dev Sets `value` as the allowance of `spender` over ``owner``'s tokens,
* given ``owner``'s signed approval.
*
* IMPORTANT: The same issues {IERC20-approve} has related to transaction
* ordering also apply here.
*
* Emits an {Approval} event.
*
* Requirements:
*
* - `spender` cannot be the zero address.
* - `deadline` must be a timestamp in the future.
* - `v`, `r` and `s` must be a valid `secp256k1` signature from `owner`
* over the EIP712-formatted function arguments.
* - the signature must use ``owner``'s current nonce (see {nonces}).
*
* For more information on the signature format, see the
* https://eips.ethereum.org/EIPS/eip-2612#specification[relevant EIP
* section].
*
* CAUTION: See Security Considerations above.
*/
function permit(
address owner,
address spender,
uint256 value,
uint256 deadline,
uint8 v,
bytes32 r,
bytes32 s
) external;
/**
* @dev Returns the current nonce for `owner`. This value must be
* included whenever a signature is generated for {permit}.
*
* Every successful call to {permit} increases ``owner``'s nonce by one. This
* prevents a signature from being used multiple times.
*/
function nonces(address owner) external view returns (uint256);
/**
* @dev Returns the domain separator used in the encoding of the signature for {permit}, as defined by {EIP712}.
*/
// solhint-disable-next-line func-name-mixedcase
function DOMAIN_SEPARATOR() external view returns (bytes32);
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { LogExpMathV8 } from "./LogExpMathV8.sol";
// solhint-disable not-rely-on-time
/// @dev See https://en.wikipedia.org/wiki/Exponential_smoothing
/// It is the same as `ExponentialMovingAverageV7` with `unchecked` scope.
library ExponentialMovingAverageV8 {
/*************
* Constants *
*************/
/// @dev The precision used to compute EMA.
uint256 private constant PRECISION = 1e18;
/***********
* Structs *
***********/
/// @dev Compiler will pack this into single `uint256`.
/// @param lastTime The last timestamp when the storage is updated.
/// @param sampleInterval The sampling time interval used in the EMA.
/// @param lastValue The last value in the data sequence, with precision 1e18.
/// @param lastEmaValue The last EMA value computed, with precision 1e18.
struct EMAStorage {
uint40 lastTime;
uint24 sampleInterval;
uint96 lastValue;
uint96 lastEmaValue;
}
/// @dev Save value of EMA storage.
/// @param s The EMA storage.
/// @param value The new value, with precision 1e18.
function saveValue(EMAStorage storage s, uint96 value) internal {
s.lastEmaValue = uint96(emaValue(s));
s.lastValue = value;
s.lastTime = uint40(block.timestamp);
}
/// @dev Return the current ema value.
/// @param s The EMA storage.
function emaValue(EMAStorage storage s) internal view returns (uint256) {
unchecked {
if (uint256(s.lastTime) < block.timestamp) {
uint256 dt = block.timestamp - uint256(s.lastTime);
uint256 e = (dt * PRECISION) / s.sampleInterval;
if (e > 41e18) {
return s.lastValue;
} else {
uint256 alpha = uint256(LogExpMathV8.exp(-int256(e)));
return (s.lastValue * (PRECISION - alpha) + s.lastEmaValue * alpha) / PRECISION;
}
} else {
return s.lastEmaValue;
}
}
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { Currency, CurrencyLibrary } from "../types/Currency.sol";
import { Address, AddressLibrary } from "../types/Address.sol";
import { FxStableMath } from "../libs/math/FxStableMath.sol";
import { ExponentialMovingAverageV8 } from "../libs/math/ExponentialMovingAverageV8.sol";
import { IERC20Upgradeable } from "@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol";
import { IPriceOracle } from "../interfaces/IPriceOracle.sol";
import { DTreasury } from "../declarations/DTreasury.sol";
import { GroupStateHelper, GroupSettings } from "../types/GroupStateHelper.sol";
import { IAToken } from "../interfaces/IAToken.sol";
import { CustomRevert } from "../libs/CustomRevert.sol";
import { IDEXRouter } from "../interfaces/IDEXRouter.sol";
import { IStrategy } from "../interfaces/IStrategy.sol";
import { GroupState } from "../types/CommonTypes.sol";
import { FullMath } from "../libs/math/FullMath.sol";
/**
* @title TreasuryStateLibrary
* @notice Library for managing treasury state and calculations.
*/
library TreasuryStateLibrary {
using CurrencyLibrary for Currency;
using AddressLibrary for Address;
using FxStableMath for FxStableMath.SwapState;
using ExponentialMovingAverageV8 for ExponentialMovingAverageV8.EMAStorage;
using GroupStateHelper for GroupSettings;
using CustomRevert for bytes4;
error ErrorInvalidTwapPrice();
error ErrorSwapFailed();
error ErrorWithdrawFromStrategy();
error StrategyUnderflow();
event BaseTokenCapsUpdated(uint256 newCaps);
event BaseTokenPriceUpdated(uint256 newPrice);
event SwapExecuted(address indexed fromToken, address indexed toToken, uint256 amountIn, uint256 amountOut);
event BaseTokenTransferred(address indexed recipient, uint256 amount);
event StrategyWithdrawal(address indexed strategy, uint256 amount);
event EMALeverageRatioUpdated(uint256 newEMAValue);
event GroupEMALeverageRatioInitialized(uint256 initialValue);
uint256 private constant PRECISION = 1e18;
/**
* @notice Updates the base token caps in the treasury state.
* @param self The treasury state storage.
* @param newCaps The new base token caps.
*/
function updateBaseTokenCaps(DTreasury.TreasuryState storage self, uint256 newCaps) internal {
self.baseTokenCaps = newCaps;
emit BaseTokenCapsUpdated(newCaps);
}
/**
* @notice Updates the base token price in the treasury state.
* @param self The treasury state storage.
* @param newPrice The new base token price.
*/
function updateBaseTokenPrice(DTreasury.TreasuryState storage self, uint256 newPrice) internal {
self.baseTokenPrice = newPrice;
emit BaseTokenPriceUpdated(newPrice);
}
/**
* @notice Loads the swap state with proper precision handling.
* @param self The treasury state storage.
* @param group The group state.
* @return _state The loaded swap state.
*/
function loadSwapState(
DTreasury.TreasuryState storage self,
GroupState memory group
) internal view returns (FxStableMath.SwapState memory _state) {
// Fetch decimals
uint8 baseTokenDecimals = GroupSettings.wrap(group.groupSettings).getBaseTokenDecimals();
uint8 aTokenDecimals = GroupSettings.wrap(group.groupSettings).getATokenDecimals();
uint8 xTokenDecimals = GroupSettings.wrap(group.groupSettings).getXTokenDecimals();
// Normalize baseSupply to 18 decimals
_state.baseSupply = normalizeDecimals(self.totalBaseTokens, baseTokenDecimals);
// Fetch base token price (already in 18 decimals)
(_state.baseTwapNav, _state.baseNav) = fetchBaseTokenPrice(self, group);
if (_state.baseSupply == 0) {
_state.aNav = PRECISION;
_state.xNav = PRECISION;
} else {
// Fetch token supplies and normalize to 18 decimals
uint256 aSupplyRaw = IERC20Upgradeable(group.core.aToken.toAddress()).totalSupply();
uint256 xSupplyRaw = IERC20Upgradeable(group.core.xToken.toAddress()).totalSupply();
_state.aSupply = normalizeDecimals(aSupplyRaw, aTokenDecimals);
_state.xSupply = normalizeDecimals(xSupplyRaw, xTokenDecimals);
// Fetch aNav (assuming it's already in 18 decimals)
_state.beta = IAToken(group.core.aToken.toAddress()).beta();
if (_state.beta) {
_state.aNav = IAToken(group.core.aToken.toAddress()).nav();
} else {
_state.aNav = PRECISION;
}
if (_state.xSupply == 0) {
// No xToken, treat the nav of xToken as 1.0
_state.xNav = PRECISION;
} else {
uint256 _baseVal = _state.baseSupply * _state.baseNav;
uint256 _aVal = _state.aSupply * _state.aNav;
if (_baseVal >= _aVal) {
_state.xNav = (_baseVal - _aVal) / _state.xSupply;
} else {
// Under-collateralized
_state.xNav = 0;
}
}
}
}
/**
* @notice Fetches the base token price from the price oracle.
* @param group The group state.
* @return _twap The time-weighted average price.
* @return _price The selected price
*/
function fetchBaseTokenPrice(
DTreasury.TreasuryState storage /*self*/,
GroupState memory group
) internal view returns (uint256 _twap, uint256 _price) {
bool isValid;
address priceOracle = group.extended.priceOracle.toAddress();
(isValid, _price) = IPriceOracle(priceOracle).getPrice(group.core.baseToken.toAddress());
if (!isValid || _price == 0) ErrorInvalidTwapPrice.selector.revertWith();
// Prices are already in 18 decimals
return (_price, _price);
}
/**
* @notice Checks if the treasury is under-collateralized.
* @param self The treasury state storage.
* @param group The group state.
* @return True if under-collateralized, false otherwise.
*/
function isUnderCollateral(DTreasury.TreasuryState storage self, GroupState memory group) internal view returns (bool) {
FxStableMath.SwapState memory _state = loadSwapState(self, group);
return _state.xNav == 0;
}
/**
* @notice Gets the collateral ratio of the treasury.
* @param self The treasury state storage.
* @param group The group state.
* @return The collateral ratio.
*/
function getCollateralRatio(DTreasury.TreasuryState storage self, GroupState memory group) internal view returns (uint256) {
FxStableMath.SwapState memory _state = loadSwapState(self, group);
if (_state.baseSupply == 0) return PRECISION;
if (_state.aSupply == 0 || _state.aNav == 0) return PRECISION * PRECISION;
return FullMath.mulDiv(_state.baseSupply * _state.baseNav, PRECISION, _state.aSupply * _state.aNav);
}
/**
* @notice Calculates the maximum mintable AToken based on the new collateral ratio.
* @param self The treasury state storage.
* @param group The group state.
* @param _newCollateralRatio The new desired collateral ratio.
* @return _maxBaseIn The maximum base tokens that can be input.
* @return _maxATokenMintable The maximum AToken that can be minted.
*/
function maxMintableAToken(
DTreasury.TreasuryState storage self,
GroupState memory group,
uint256 _newCollateralRatio
) internal view returns (uint256 _maxBaseIn, uint256 _maxATokenMintable) {
FxStableMath.SwapState memory _state = loadSwapState(self, group);
(_maxBaseIn, _maxATokenMintable) = _state.maxMintableAToken(_newCollateralRatio);
}
/**
* @notice Calculates the maximum redeemable AToken based on the new collateral ratio.
* @param self The treasury state storage.
* @param group The group state.
* @param _newCollateralRatio The new desired collateral ratio.
* @return _maxBaseOut The maximum base tokens that can be output.
* @return _maxATokenRedeemable The maximum AToken that can be redeemed.
*/
function maxRedeemableAToken(
DTreasury.TreasuryState storage self,
GroupState memory group,
uint256 _newCollateralRatio
) internal view returns (uint256 _maxBaseOut, uint256 _maxATokenRedeemable) {
FxStableMath.SwapState memory _state = loadSwapState(self, group);
(_maxBaseOut, _maxATokenRedeemable) = _state.maxRedeemableAToken(_newCollateralRatio);
}
/**
* @notice Updates the Exponential Moving Average (EMA) of the leverage ratio.
* @param self The treasury state storage.
* @param _state The current swap state.
*/
function updateEMALeverageRatio(DTreasury.TreasuryState storage self, FxStableMath.SwapState memory _state) internal {
uint256 _ratio = _state.leverageRatio();
self.emaLeverageRatio.saveValue(uint96(_ratio));
emit EMALeverageRatioUpdated(_ratio);
}
function getEMAValue(DTreasury.TreasuryState storage self) internal view returns (uint256) {
return self.emaLeverageRatio.emaValue();
}
function initializeGroupEMALeverageRatio(DTreasury.TreasuryState storage self) internal {
self.emaLeverageRatio.lastTime = uint40(block.timestamp);
self.emaLeverageRatio.lastValue = uint96(PRECISION * 2);
self.emaLeverageRatio.lastEmaValue = uint96(PRECISION * 2);
self.emaLeverageRatio.sampleInterval = 1 days;
emit GroupEMALeverageRatioInitialized(PRECISION * 2);
}
/**
* @dev Normalizes a token amount to 18 decimals.
* @param amount The amount to normalize.
* @param tokenDecimals The decimals of the token.
* @return The normalized amount.
*/
function normalizeDecimals(uint256 amount, uint8 tokenDecimals) internal pure returns (uint256) {
if (tokenDecimals == 18) {
return amount;
}
uint256 factor;
if (tokenDecimals < 18) {
factor = 10 ** (18 - tokenDecimals);
return amount * factor;
} else {
factor = 10 ** (tokenDecimals - 18);
return amount / factor;
}
}
/**
* @dev Denormalizes a token amount from 18 decimals to the token's decimals.
* @param amount The amount to denormalize.
* @param tokenDecimals The decimals of the token.
* @return The denormalized amount.
*/
function denormalizeDecimals(uint256 amount, uint8 tokenDecimals) internal pure returns (uint256) {
if (tokenDecimals == 18) {
return amount;
}
uint256 factor;
if (tokenDecimals < 18) {
factor = 10 ** (18 - tokenDecimals);
return amount / factor;
} else {
factor = 10 ** (tokenDecimals - 18);
return amount * factor;
}
}
/**
* @notice Transfers base tokens to the recipient, withdrawing from strategy if necessary.
* @param baseToken The base token.
* @param strategyAddress The strategy address.
* @param treasuryState The treasury state.
* @param amount The amount of base tokens to transfer.
* @param recipient The recipient address.
*/
function transferBaseToken(
address self,
uint8 baseTokenDecimals,
Currency baseToken,
address strategyAddress,
DTreasury.TreasuryState storage treasuryState,
uint256 amount,
address recipient
) internal {
uint256 amountNormalized = normalizeDecimals(amount, baseTokenDecimals);
uint256 balance = baseToken.balanceOf(self);
uint256 balanceNormalized = normalizeDecimals(balance, baseTokenDecimals);
if (balanceNormalized < amountNormalized) {
uint256 diff = amountNormalized - balanceNormalized;
if (diff == 0) return;
IStrategy strategy = IStrategy(strategyAddress);
bool success = strategy.withdrawToTreasury(diff);
if (!success) ErrorWithdrawFromStrategy.selector.revertWith();
if (treasuryState.strategyUnderlying < diff) StrategyUnderflow.selector.revertWith();
treasuryState.strategyUnderlying -= diff;
balance = baseToken.balanceOf(self);
balanceNormalized = normalizeDecimals(balance, baseTokenDecimals);
if (amountNormalized > balanceNormalized) {
amountNormalized = balanceNormalized;
amount = denormalizeDecimals(amountNormalized, baseTokenDecimals);
}
}
treasuryState.totalBaseTokens -= amountNormalized;
baseToken.safeTransfer(recipient, amount);
emit BaseTokenTransferred(recipient, amount);
}
/**
* @notice Swaps tokens using the specified router.
* @param self The address of the contract using this library
* @param swapRouter The address of the swap router.
* @param from The address of the token to swap from.
* @param to The address of the token to swap to.
* @param amount The amount of tokens to swap.
* @param minOutAmount The minimum acceptable amount of tokens to receive.
* @return swappedAmount The amount of tokens received after the swap.
*/
function swapTokens(
address self,
address swapRouter,
address from,
address to,
uint256 amount,
uint256 minOutAmount
) internal returns (uint256 swappedAmount) {
IDEXRouter router = IDEXRouter(swapRouter);
uint256 deadline = block.timestamp + 30;
address[] memory path = new address[](2);
path[0] = from;
path[1] = to;
Currency fromToken = Currency.wrap(from);
uint256 currentAllowance = fromToken.allowance(self, swapRouter);
if (currentAllowance < amount) {
fromToken.safeIncreaseAllowance(swapRouter, amount - currentAllowance);
}
swappedAmount = _executeSwap(router, amount, minOutAmount, path, self, deadline);
if (swappedAmount < minOutAmount) {
ErrorSwapFailed.selector.revertWith();
}
emit SwapExecuted(from, to, amount, swappedAmount);
return swappedAmount;
}
function _executeSwap(
IDEXRouter router,
uint256 amount,
uint256 minOutAmount,
address[] memory path,
address to,
uint256 deadline
) private returns (uint256) {
try router.swapExactTokensForTokens(amount, minOutAmount, path, to, deadline) returns (uint256[] memory amounts) {
return amounts[amounts.length - 1];
} catch {
ErrorSwapFailed.selector.revertWith();
}
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { IERC20Upgradeable } from "@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol";
import { FeeModel } from "../types/CommonTypes.sol";
interface IDXToken is IERC20Upgradeable {
/**********
* Events *
**********/
/// @notice Emitted when the cooling-off period is updated.
/// @param oldValue The value of the previous cooling-off period.
/// @param newValue The value of the current cooling-off period.
event UpdateCoolingOffPeriod(uint256 indexed oldValue, uint256 indexed newValue);
/// @notice Emitted when the management fee rate is updated.
/// @param oldRate The previous management fee rate.
/// @param newRate The new management fee rate.
event UpdateManagementFeeRate(uint256 indexed oldRate, uint256 indexed newRate);
/// @notice Emitted when the funding fee rate is updated.
/// @param oldRate The previous funding fee rate.
/// @param newRate The new funding fee rate.
event UpdateFundingFeeRate(uint256 indexed oldRate, uint256 indexed newRate);
/// @notice Emitted when the fixed yield amount is updated.
/// @param oldRate The previous fixed yield amount.
/// @param newRate The new fixed yield amount.
event UpdateVariableFundingFeeRate(uint256 indexed oldRate, uint256 indexed newRate);
/// @notice Emitted when fees are collected.
/// @param amount The amount of fees collected.
event FeesCollected(uint256 indexed amount);
/// @notice Emitted when tokens are minted.
/// @param to The address receiving the minted tokens.
/// @param amount The amount of tokens minted.
event Minted(address indexed to, uint256 indexed amount);
/// @notice Emitted when tokens are burned.
/// @param from The address whose tokens are burned.
/// @param amount The amount of tokens burned.
event Burned(address indexed from, uint256 indexed amount);
/// @notice Emitted when the cooling-off period is triggered.
/// @param account The account triggering the cooling-off period.
/// @param timestamp The timestamp when triggered.
event CoolingOffPeriodTriggered(address indexed account, uint256 indexed timestamp);
/// @notice Emitted when the associated group is updated.
/// @param groupId The new associated group ID.
event UpdateGroup(bytes32 indexed groupId);
/// @notice Emitted when the treasury address is updated.
/// @param oldTreasury The old treasury address.
/// @param newTreasury The new treasury address.
event UpdateTreasuryAddress(address indexed oldTreasury, address indexed newTreasury);
/// @notice Emitted when the max supply is updated.
/// @param oldMaxSupply The old max supply.
/// @param newMaxSupply The new max supply.
event UpdateMaxSupply(uint256 indexed oldMaxSupply, uint256 indexed newMaxSupply);
/**********
* Errors *
**********/
/// @dev Thrown when an address is zero.
error ZeroAddress();
/// @dev Thrown when decimals are invalid.
error InvalidDecimals();
/// @dev Thrown when the cooling-off period is too large.
error CoolingOffPeriodTooLarge();
/// @dev Thrown when trying to mint or burn an amount below the minimum.
error ErrorMinimumMintingAmount();
/// @dev Thrown when trying to burn an amount below the minimum.
error ErrorMinimumBurningAmount();
/// @dev Thrown when minting exceeds the max supply.
error ExceedsMaxSupply();
/// @dev Thrown when cooling-off period is active.
error CoolingOffPeriodActive();
/// @dev Thrown when parameter is unchanged.
error ParameterUnchanged();
/// @dev Thrown when management fee is invalid.
error InvalidManagementFee();
/// @dev Thrown when funding fee rate is invalid.
error InvalidFundingFeeRate();
/// @dev Thrown when fixed yield amount is invalid.
error InvalidFixedYieldAmount();
/// @dev Thrown when fee model is invalid.
error InvalidFeeModel();
/// @dev Thrown when base supply is zero.
error BaseSupplyZero();
/// @dev Thrown when transfer amount exceeds balance.
error TransferAmountExceedsBalance();
/// @dev Thrown when burn amount exceeds balance.
error BurnAmountExceedsBalance();
/// @dev Thrown cannot recover token.
error CannotRecoverToken();
/// @dev Thrown when invalid base token price.
error InvalidBaseTokenPrice();
/// @dev Thrown when an operation is not permitted.
error NotPermitted();
/*************************
* Public View Functions *
*************************/
/// @notice Returns the NAV of the token.
function nav() external view returns (uint256);
/// @notice Converts DXToken amount to base token amount.
function dxTokenToBaseToken(uint256 dxTokenAmount) external view returns (uint256 baseTokenAmount);
/// @notice Returns the fee model.
function getFeeModel() external view returns (FeeModel);
/****************************
* Public Mutated Functions *
****************************/
/// @notice Mints tokens to a specified address.
function mint(address _to, uint256 _amount) external;
/// @notice Burns tokens from a specified address.
function burn(address _from, uint256 _amount) external;
/// @notice Collects accumulated fees.
function collectFees() external;
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { IERC20Upgradeable } from "@openzeppelin/contracts-upgradeable/token/ERC20/utils/SafeERC20Upgradeable.sol";
import { GroupId } from "../types/GroupId.sol";
/**
* @title IAToken
* @notice Interface for AToken, an ERC20-compatible token with minting, burning, and
* additional functionality for treasury and rebalance management.
*/
interface IAToken is IERC20Upgradeable {
/**
* @dev Emitted when the treasury address is updated.
* @param oldTreasury The previous treasury address.
* @param newTreasury The new treasury address.
*/
event UpdateTreasuryAddress(address indexed oldTreasury, address indexed newTreasury);
/**
* @dev Emitted when the rebalance pool address is updated.
* @param oldRebalancePool The previous rebalance pool address.
* @param newRebalancePool The new rebalance pool address.
*/
event UpdateRebalancePool(address indexed oldRebalancePool, address indexed newRebalancePool);
/**
* @dev Emitted when tokens are minted to an address.
* @param to The address receiving the minted tokens.
* @param amount The amount of tokens minted.
*/
event Minted(address indexed to, uint256 indexed amount);
/**
* @dev Emitted when tokens are burned from an address.
* @param from The address from which tokens are burned.
* @param amount The amount of tokens burned.
*/
event Burned(address indexed from, uint256 indexed amount);
/**
* @dev Emitted when a group is updated.
* @param groupId The identifier of the updated group.
*/
event UpdateGroup(GroupId indexed groupId);
/**
* @dev Emitted when the max supply is updated.
* @param oldMaxSupply The previous max supply.
* @param newMaxSupply The new max supply.
*/
event UpdateMaxSupply(uint256 indexed oldMaxSupply, uint256 indexed newMaxSupply);
// Custom Errors
error InvalidDecimals();
error ErrorNotPermitted();
error ErrorZeroAddress();
error ErrorZeroAmount();
error ErrorExceedsMaxSupply();
error ErrorCannotRecoverToken();
error ErrorParameterUnchanged();
/**
* @notice Returns the beta status of the token.
*/
function beta() external view returns (bool);
/**
* @notice Returns the Net Asset Value (NAV) of the token.
* @return The current NAV as an unsigned integer.
*/
function nav() external view returns (uint256);
/**
* @notice Returns the number of decimals used by the token.
* @return The number of decimals.
*/
function decimals() external view returns (uint8);
/**
* @notice Mints tokens to a specified address.
* @dev Emits a {Minted} event.
* @param _to The address to receive the minted tokens.
* @param _amount The amount of tokens to mint.
*/
function mint(address _to, uint256 _amount) external;
/**
* @notice Burns tokens from a specified address.
* @dev Emits a {Burned} event.
* @param _from The address from which tokens are burned.
* @param _amount The amount of tokens to burn.
*/
function burn(address _from, uint256 _amount) external;
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
import { GroupId } from "../types/GroupId.sol";
interface IProtocolMinimum {
function stabilityRatio(GroupId groupId) external view returns (uint96);
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/ERC20.sol)
pragma solidity ^0.8.0;
import "./IERC20Upgradeable.sol";
import "./extensions/IERC20MetadataUpgradeable.sol";
import "../../utils/ContextUpgradeable.sol";
import {Initializable} from "../../proxy/utils/Initializable.sol";
/**
* @dev Implementation of the {IERC20} interface.
*
* This implementation is agnostic to the way tokens are created. This means
* that a supply mechanism has to be added in a derived contract using {_mint}.
* For a generic mechanism see {ERC20PresetMinterPauser}.
*
* TIP: For a detailed writeup see our guide
* https://forum.openzeppelin.com/t/how-to-implement-erc20-supply-mechanisms/226[How
* to implement supply mechanisms].
*
* The default value of {decimals} is 18. To change this, you should override
* this function so it returns a different value.
*
* We have followed general OpenZeppelin Contracts guidelines: functions revert
* instead returning `false` on failure. This behavior is nonetheless
* conventional and does not conflict with the expectations of ERC20
* applications.
*
* Additionally, an {Approval} event is emitted on calls to {transferFrom}.
* This allows applications to reconstruct the allowance for all accounts just
* by listening to said events. Other implementations of the EIP may not emit
* these events, as it isn't required by the specification.
*
* Finally, the non-standard {decreaseAllowance} and {increaseAllowance}
* functions have been added to mitigate the well-known issues around setting
* allowances. See {IERC20-approve}.
*/
contract ERC20Upgradeable is Initializable, ContextUpgradeable, IERC20Upgradeable, IERC20MetadataUpgradeable {
mapping(address => uint256) private _balances;
mapping(address => mapping(address => uint256)) private _allowances;
uint256 private _totalSupply;
string private _name;
string private _symbol;
/**
* @dev Sets the values for {name} and {symbol}.
*
* All two of these values are immutable: they can only be set once during
* construction.
*/
function __ERC20_init(string memory name_, string memory symbol_) internal onlyInitializing {
__ERC20_init_unchained(name_, symbol_);
}
function __ERC20_init_unchained(string memory name_, string memory symbol_) internal onlyInitializing {
_name = name_;
_symbol = symbol_;
}
/**
* @dev Returns the name of the token.
*/
function name() public view virtual override returns (string memory) {
return _name;
}
/**
* @dev Returns the symbol of the token, usually a shorter version of the
* name.
*/
function symbol() public view virtual override returns (string memory) {
return _symbol;
}
/**
* @dev Returns the number of decimals used to get its user representation.
* For example, if `decimals` equals `2`, a balance of `505` tokens should
* be displayed to a user as `5.05` (`505 / 10 ** 2`).
*
* Tokens usually opt for a value of 18, imitating the relationship between
* Ether and Wei. This is the default value returned by this function, unless
* it's overridden.
*
* NOTE: This information is only used for _display_ purposes: it in
* no way affects any of the arithmetic of the contract, including
* {IERC20-balanceOf} and {IERC20-transfer}.
*/
function decimals() public view virtual override returns (uint8) {
return 18;
}
/**
* @dev See {IERC20-totalSupply}.
*/
function totalSupply() public view virtual override returns (uint256) {
return _totalSupply;
}
/**
* @dev See {IERC20-balanceOf}.
*/
function balanceOf(address account) public view virtual override returns (uint256) {
return _balances[account];
}
/**
* @dev See {IERC20-transfer}.
*
* Requirements:
*
* - `to` cannot be the zero address.
* - the caller must have a balance of at least `amount`.
*/
function transfer(address to, uint256 amount) public virtual override returns (bool) {
address owner = _msgSender();
_transfer(owner, to, amount);
return true;
}
/**
* @dev See {IERC20-allowance}.
*/
function allowance(address owner, address spender) public view virtual override returns (uint256) {
return _allowances[owner][spender];
}
/**
* @dev See {IERC20-approve}.
*
* NOTE: If `amount` is the maximum `uint256`, the allowance is not updated on
* `transferFrom`. This is semantically equivalent to an infinite approval.
*
* Requirements:
*
* - `spender` cannot be the zero address.
*/
function approve(address spender, uint256 amount) public virtual override returns (bool) {
address owner = _msgSender();
_approve(owner, spender, amount);
return true;
}
/**
* @dev See {IERC20-transferFrom}.
*
* Emits an {Approval} event indicating the updated allowance. This is not
* required by the EIP. See the note at the beginning of {ERC20}.
*
* NOTE: Does not update the allowance if the current allowance
* is the maximum `uint256`.
*
* Requirements:
*
* - `from` and `to` cannot be the zero address.
* - `from` must have a balance of at least `amount`.
* - the caller must have allowance for ``from``'s tokens of at least
* `amount`.
*/
function transferFrom(address from, address to, uint256 amount) public virtual override returns (bool) {
address spender = _msgSender();
_spendAllowance(from, spender, amount);
_transfer(from, to, amount);
return true;
}
/**
* @dev Atomically increases the allowance granted to `spender` by the caller.
*
* This is an alternative to {approve} that can be used as a mitigation for
* problems described in {IERC20-approve}.
*
* Emits an {Approval} event indicating the updated allowance.
*
* Requirements:
*
* - `spender` cannot be the zero address.
*/
function increaseAllowance(address spender, uint256 addedValue) public virtual returns (bool) {
address owner = _msgSender();
_approve(owner, spender, allowance(owner, spender) + addedValue);
return true;
}
/**
* @dev Atomically decreases the allowance granted to `spender` by the caller.
*
* This is an alternative to {approve} that can be used as a mitigation for
* problems described in {IERC20-approve}.
*
* Emits an {Approval} event indicating the updated allowance.
*
* Requirements:
*
* - `spender` cannot be the zero address.
* - `spender` must have allowance for the caller of at least
* `subtractedValue`.
*/
function decreaseAllowance(address spender, uint256 subtractedValue) public virtual returns (bool) {
address owner = _msgSender();
uint256 currentAllowance = allowance(owner, spender);
require(currentAllowance >= subtractedValue, "ERC20: decreased allowance below zero");
unchecked {
_approve(owner, spender, currentAllowance - subtractedValue);
}
return true;
}
/**
* @dev Moves `amount` of tokens from `from` to `to`.
*
* This internal function is equivalent to {transfer}, and can be used to
* e.g. implement automatic token fees, slashing mechanisms, etc.
*
* Emits a {Transfer} event.
*
* Requirements:
*
* - `from` cannot be the zero address.
* - `to` cannot be the zero address.
* - `from` must have a balance of at least `amount`.
*/
function _transfer(address from, address to, uint256 amount) internal virtual {
require(from != address(0), "ERC20: transfer from the zero address");
require(to != address(0), "ERC20: transfer to the zero address");
_beforeTokenTransfer(from, to, amount);
uint256 fromBalance = _balances[from];
require(fromBalance >= amount, "ERC20: transfer amount exceeds balance");
unchecked {
_balances[from] = fromBalance - amount;
// Overflow not possible: the sum of all balances is capped by totalSupply, and the sum is preserved by
// decrementing then incrementing.
_balances[to] += amount;
}
emit Transfer(from, to, amount);
_afterTokenTransfer(from, to, amount);
}
/** @dev Creates `amount` tokens and assigns them to `account`, increasing
* the total supply.
*
* Emits a {Transfer} event with `from` set to the zero address.
*
* Requirements:
*
* - `account` cannot be the zero address.
*/
function _mint(address account, uint256 amount) internal virtual {
require(account != address(0), "ERC20: mint to the zero address");
_beforeTokenTransfer(address(0), account, amount);
_totalSupply += amount;
unchecked {
// Overflow not possible: balance + amount is at most totalSupply + amount, which is checked above.
_balances[account] += amount;
}
emit Transfer(address(0), account, amount);
_afterTokenTransfer(address(0), account, amount);
}
/**
* @dev Destroys `amount` tokens from `account`, reducing the
* total supply.
*
* Emits a {Transfer} event with `to` set to the zero address.
*
* Requirements:
*
* - `account` cannot be the zero address.
* - `account` must have at least `amount` tokens.
*/
function _burn(address account, uint256 amount) internal virtual {
require(account != address(0), "ERC20: burn from the zero address");
_beforeTokenTransfer(account, address(0), amount);
uint256 accountBalance = _balances[account];
require(accountBalance >= amount, "ERC20: burn amount exceeds balance");
unchecked {
_balances[account] = accountBalance - amount;
// Overflow not possible: amount <= accountBalance <= totalSupply.
_totalSupply -= amount;
}
emit Transfer(account, address(0), amount);
_afterTokenTransfer(account, address(0), amount);
}
/**
* @dev Sets `amount` as the allowance of `spender` over the `owner` s tokens.
*
* This internal function is equivalent to `approve`, and can be used to
* e.g. set automatic allowances for certain subsystems, etc.
*
* Emits an {Approval} event.
*
* Requirements:
*
* - `owner` cannot be the zero address.
* - `spender` cannot be the zero address.
*/
function _approve(address owner, address spender, uint256 amount) internal virtual {
require(owner != address(0), "ERC20: approve from the zero address");
require(spender != address(0), "ERC20: approve to the zero address");
_allowances[owner][spender] = amount;
emit Approval(owner, spender, amount);
}
/**
* @dev Updates `owner` s allowance for `spender` based on spent `amount`.
*
* Does not update the allowance amount in case of infinite allowance.
* Revert if not enough allowance is available.
*
* Might emit an {Approval} event.
*/
function _spendAllowance(address owner, address spender, uint256 amount) internal virtual {
uint256 currentAllowance = allowance(owner, spender);
if (currentAllowance != type(uint256).max) {
require(currentAllowance >= amount, "ERC20: insufficient allowance");
unchecked {
_approve(owner, spender, currentAllowance - amount);
}
}
}
/**
* @dev Hook that is called before any transfer of tokens. This includes
* minting and burning.
*
* Calling conditions:
*
* - when `from` and `to` are both non-zero, `amount` of ``from``'s tokens
* will be transferred to `to`.
* - when `from` is zero, `amount` tokens will be minted for `to`.
* - when `to` is zero, `amount` of ``from``'s tokens will be burned.
* - `from` and `to` are never both zero.
*
* To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
*/
function _beforeTokenTransfer(address from, address to, uint256 amount) internal virtual {}
/**
* @dev Hook that is called after any transfer of tokens. This includes
* minting and burning.
*
* Calling conditions:
*
* - when `from` and `to` are both non-zero, `amount` of ``from``'s tokens
* has been transferred to `to`.
* - when `from` is zero, `amount` tokens have been minted for `to`.
* - when `to` is zero, `amount` of ``from``'s tokens have been burned.
* - `from` and `to` are never both zero.
*
* To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
*/
function _afterTokenTransfer(address from, address to, uint256 amount) internal virtual {}
/**
* @dev This empty reserved space is put in place to allow future versions to add new
* variables without shifting down storage in the inheritance chain.
* See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
*/
uint256[45] private __gap;
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.9.0) (interfaces/IERC4626.sol)
pragma solidity ^0.8.0;
import "../token/ERC20/IERC20Upgradeable.sol";
import "../token/ERC20/extensions/IERC20MetadataUpgradeable.sol";
/**
* @dev Interface of the ERC4626 "Tokenized Vault Standard", as defined in
* https://eips.ethereum.org/EIPS/eip-4626[ERC-4626].
*
* _Available since v4.7._
*/
interface IERC4626Upgradeable is IERC20Upgradeable, IERC20MetadataUpgradeable {
event Deposit(address indexed sender, address indexed owner, uint256 assets, uint256 shares);
event Withdraw(
address indexed sender,
address indexed receiver,
address indexed owner,
uint256 assets,
uint256 shares
);
/**
* @dev Returns the address of the underlying token used for the Vault for accounting, depositing, and withdrawing.
*
* - MUST be an ERC-20 token contract.
* - MUST NOT revert.
*/
function asset() external view returns (address assetTokenAddress);
/**
* @dev Returns the total amount of the underlying asset that is “managed” by Vault.
*
* - SHOULD include any compounding that occurs from yield.
* - MUST be inclusive of any fees that are charged against assets in the Vault.
* - MUST NOT revert.
*/
function totalAssets() external view returns (uint256 totalManagedAssets);
/**
* @dev Returns the amount of shares that the Vault would exchange for the amount of assets provided, in an ideal
* scenario where all the conditions are met.
*
* - MUST NOT be inclusive of any fees that are charged against assets in the Vault.
* - MUST NOT show any variations depending on the caller.
* - MUST NOT reflect slippage or other on-chain conditions, when performing the actual exchange.
* - MUST NOT revert.
*
* NOTE: This calculation MAY NOT reflect the “per-user” price-per-share, and instead should reflect the
* “average-user’s” price-per-share, meaning what the average user should expect to see when exchanging to and
* from.
*/
function convertToShares(uint256 assets) external view returns (uint256 shares);
/**
* @dev Returns the amount of assets that the Vault would exchange for the amount of shares provided, in an ideal
* scenario where all the conditions are met.
*
* - MUST NOT be inclusive of any fees that are charged against assets in the Vault.
* - MUST NOT show any variations depending on the caller.
* - MUST NOT reflect slippage or other on-chain conditions, when performing the actual exchange.
* - MUST NOT revert.
*
* NOTE: This calculation MAY NOT reflect the “per-user” price-per-share, and instead should reflect the
* “average-user’s” price-per-share, meaning what the average user should expect to see when exchanging to and
* from.
*/
function convertToAssets(uint256 shares) external view returns (uint256 assets);
/**
* @dev Returns the maximum amount of the underlying asset that can be deposited into the Vault for the receiver,
* through a deposit call.
*
* - MUST return a limited value if receiver is subject to some deposit limit.
* - MUST return 2 ** 256 - 1 if there is no limit on the maximum amount of assets that may be deposited.
* - MUST NOT revert.
*/
function maxDeposit(address receiver) external view returns (uint256 maxAssets);
/**
* @dev Allows an on-chain or off-chain user to simulate the effects of their deposit at the current block, given
* current on-chain conditions.
*
* - MUST return as close to and no more than the exact amount of Vault shares that would be minted in a deposit
* call in the same transaction. I.e. deposit should return the same or more shares as previewDeposit if called
* in the same transaction.
* - MUST NOT account for deposit limits like those returned from maxDeposit and should always act as though the
* deposit would be accepted, regardless if the user has enough tokens approved, etc.
* - MUST be inclusive of deposit fees. Integrators should be aware of the existence of deposit fees.
* - MUST NOT revert.
*
* NOTE: any unfavorable discrepancy between convertToShares and previewDeposit SHOULD be considered slippage in
* share price or some other type of condition, meaning the depositor will lose assets by depositing.
*/
function previewDeposit(uint256 assets) external view returns (uint256 shares);
/**
* @dev Mints shares Vault shares to receiver by depositing exactly amount of underlying tokens.
*
* - MUST emit the Deposit event.
* - MAY support an additional flow in which the underlying tokens are owned by the Vault contract before the
* deposit execution, and are accounted for during deposit.
* - MUST revert if all of assets cannot be deposited (due to deposit limit being reached, slippage, the user not
* approving enough underlying tokens to the Vault contract, etc).
*
* NOTE: most implementations will require pre-approval of the Vault with the Vault’s underlying asset token.
*/
function deposit(uint256 assets, address receiver) external returns (uint256 shares);
/**
* @dev Returns the maximum amount of the Vault shares that can be minted for the receiver, through a mint call.
* - MUST return a limited value if receiver is subject to some mint limit.
* - MUST return 2 ** 256 - 1 if there is no limit on the maximum amount of shares that may be minted.
* - MUST NOT revert.
*/
function maxMint(address receiver) external view returns (uint256 maxShares);
/**
* @dev Allows an on-chain or off-chain user to simulate the effects of their mint at the current block, given
* current on-chain conditions.
*
* - MUST return as close to and no fewer than the exact amount of assets that would be deposited in a mint call
* in the same transaction. I.e. mint should return the same or fewer assets as previewMint if called in the
* same transaction.
* - MUST NOT account for mint limits like those returned from maxMint and should always act as though the mint
* would be accepted, regardless if the user has enough tokens approved, etc.
* - MUST be inclusive of deposit fees. Integrators should be aware of the existence of deposit fees.
* - MUST NOT revert.
*
* NOTE: any unfavorable discrepancy between convertToAssets and previewMint SHOULD be considered slippage in
* share price or some other type of condition, meaning the depositor will lose assets by minting.
*/
function previewMint(uint256 shares) external view returns (uint256 assets);
/**
* @dev Mints exactly shares Vault shares to receiver by depositing amount of underlying tokens.
*
* - MUST emit the Deposit event.
* - MAY support an additional flow in which the underlying tokens are owned by the Vault contract before the mint
* execution, and are accounted for during mint.
* - MUST revert if all of shares cannot be minted (due to deposit limit being reached, slippage, the user not
* approving enough underlying tokens to the Vault contract, etc).
*
* NOTE: most implementations will require pre-approval of the Vault with the Vault’s underlying asset token.
*/
function mint(uint256 shares, address receiver) external returns (uint256 assets);
/**
* @dev Returns the maximum amount of the underlying asset that can be withdrawn from the owner balance in the
* Vault, through a withdraw call.
*
* - MUST return a limited value if owner is subject to some withdrawal limit or timelock.
* - MUST NOT revert.
*/
function maxWithdraw(address owner) external view returns (uint256 maxAssets);
/**
* @dev Allows an on-chain or off-chain user to simulate the effects of their withdrawal at the current block,
* given current on-chain conditions.
*
* - MUST return as close to and no fewer than the exact amount of Vault shares that would be burned in a withdraw
* call in the same transaction. I.e. withdraw should return the same or fewer shares as previewWithdraw if
* called
* in the same transaction.
* - MUST NOT account for withdrawal limits like those returned from maxWithdraw and should always act as though
* the withdrawal would be accepted, regardless if the user has enough shares, etc.
* - MUST be inclusive of withdrawal fees. Integrators should be aware of the existence of withdrawal fees.
* - MUST NOT revert.
*
* NOTE: any unfavorable discrepancy between convertToShares and previewWithdraw SHOULD be considered slippage in
* share price or some other type of condition, meaning the depositor will lose assets by depositing.
*/
function previewWithdraw(uint256 assets) external view returns (uint256 shares);
/**
* @dev Burns shares from owner and sends exactly assets of underlying tokens to receiver.
*
* - MUST emit the Withdraw event.
* - MAY support an additional flow in which the underlying tokens are owned by the Vault contract before the
* withdraw execution, and are accounted for during withdraw.
* - MUST revert if all of assets cannot be withdrawn (due to withdrawal limit being reached, slippage, the owner
* not having enough shares, etc).
*
* Note that some implementations will require pre-requesting to the Vault before a withdrawal may be performed.
* Those methods should be performed separately.
*/
function withdraw(uint256 assets, address receiver, address owner) external returns (uint256 shares);
/**
* @dev Returns the maximum amount of Vault shares that can be redeemed from the owner balance in the Vault,
* through a redeem call.
*
* - MUST return a limited value if owner is subject to some withdrawal limit or timelock.
* - MUST return balanceOf(owner) if owner is not subject to any withdrawal limit or timelock.
* - MUST NOT revert.
*/
function maxRedeem(address owner) external view returns (uint256 maxShares);
/**
* @dev Allows an on-chain or off-chain user to simulate the effects of their redeemption at the current block,
* given current on-chain conditions.
*
* - MUST return as close to and no more than the exact amount of assets that would be withdrawn in a redeem call
* in the same transaction. I.e. redeem should return the same or more assets as previewRedeem if called in the
* same transaction.
* - MUST NOT account for redemption limits like those returned from maxRedeem and should always act as though the
* redemption would be accepted, regardless if the user has enough shares, etc.
* - MUST be inclusive of withdrawal fees. Integrators should be aware of the existence of withdrawal fees.
* - MUST NOT revert.
*
* NOTE: any unfavorable discrepancy between convertToAssets and previewRedeem SHOULD be considered slippage in
* share price or some other type of condition, meaning the depositor will lose assets by redeeming.
*/
function previewRedeem(uint256 shares) external view returns (uint256 assets);
/**
* @dev Burns exactly shares from owner and sends assets of underlying tokens to receiver.
*
* - MUST emit the Withdraw event.
* - MAY support an additional flow in which the underlying tokens are owned by the Vault contract before the
* redeem execution, and are accounted for during redeem.
* - MUST revert if all of shares cannot be redeemed (due to withdrawal limit being reached, slippage, the owner
* not having enough shares, etc).
*
* NOTE: some implementations will require pre-requesting to the Vault before a withdrawal may be performed.
* Those methods should be performed separately.
*/
function redeem(uint256 shares, address receiver, address owner) external returns (uint256 assets);
}// SPDX-License-Identifier: MIT
// THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
// WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
// COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
pragma solidity 0.8.28;
/* solhint-disable */
/**
* @dev Copied from https://github.com/balancer/balancer-v2-monorepo/blob/master/pkg/solidity-utils/contracts/math/LogExpMath.sol
*
* Some modifications are made due to compile error.
*
* It is the same as `LogExpMathV8` with `unchecked` scope.
*
* @dev Exponentiation and logarithm functions for 18 decimal fixed point numbers (both base and exponent/argument).
*
* Exponentiation and logarithm with arbitrary bases (x^y and log_x(y)) are implemented by conversion to natural
* exponentiation and logarithm (where the base is Euler's number).
*
* @author Fernando Martinelli - @fernandomartinelli
* @author Sergio Yuhjtman - @sergioyuhjtman
* @author Daniel Fernandez - @dmf7z
*/
library LogExpMathV8 {
// All fixed point multiplications and divisions are inlined. This means we need to divide by ONE when multiplying
// two numbers, and multiply by ONE when dividing them.
// All arguments and return values are 18 decimal fixed point numbers.
int256 constant ONE_18 = 1e18;
// Internally, intermediate values are computed with higher precision as 20 decimal fixed point numbers, and in the
// case of ln36, 36 decimals.
int256 constant ONE_20 = 1e20;
int256 constant ONE_36 = 1e36;
// The domain of natural exponentiation is bound by the word size and number of decimals used.
//
// Because internally the result will be stored using 20 decimals, the largest possible result is
// (2^255 - 1) / 10^20, which makes the largest exponent ln((2^255 - 1) / 10^20) = 130.700829182905140221.
// The smallest possible result is 10^(-18), which makes largest negative argument
// ln(10^(-18)) = -41.446531673892822312.
// We use 130.0 and -41.0 to have some safety margin.
int256 constant MAX_NATURAL_EXPONENT = 130e18;
int256 constant MIN_NATURAL_EXPONENT = -41e18;
// Bounds for ln_36's argument. Both ln(0.9) and ln(1.1) can be represented with 36 decimal places in a fixed point
// 256 bit integer.
int256 constant LN_36_LOWER_BOUND = ONE_18 - 1e17;
int256 constant LN_36_UPPER_BOUND = ONE_18 + 1e17;
uint256 constant MILD_EXPONENT_BOUND = 2 ** 254 / uint256(ONE_20);
// 18 decimal constants
int256 constant x0 = 128000000000000000000; // 2ˆ7
int256 constant a0 = 38877084059945950922200000000000000000000000000000000000; // eˆ(x0) (no decimals)
int256 constant x1 = 64000000000000000000; // 2ˆ6
int256 constant a1 = 6235149080811616882910000000; // eˆ(x1) (no decimals)
// 20 decimal constants
int256 constant x2 = 3200000000000000000000; // 2ˆ5
int256 constant a2 = 7896296018268069516100000000000000; // eˆ(x2)
int256 constant x3 = 1600000000000000000000; // 2ˆ4
int256 constant a3 = 888611052050787263676000000; // eˆ(x3)
int256 constant x4 = 800000000000000000000; // 2ˆ3
int256 constant a4 = 298095798704172827474000; // eˆ(x4)
int256 constant x5 = 400000000000000000000; // 2ˆ2
int256 constant a5 = 5459815003314423907810; // eˆ(x5)
int256 constant x6 = 200000000000000000000; // 2ˆ1
int256 constant a6 = 738905609893065022723; // eˆ(x6)
int256 constant x7 = 100000000000000000000; // 2ˆ0
int256 constant a7 = 271828182845904523536; // eˆ(x7)
int256 constant x8 = 50000000000000000000; // 2ˆ-1
int256 constant a8 = 164872127070012814685; // eˆ(x8)
int256 constant x9 = 25000000000000000000; // 2ˆ-2
int256 constant a9 = 128402541668774148407; // eˆ(x9)
int256 constant x10 = 12500000000000000000; // 2ˆ-3
int256 constant a10 = 113314845306682631683; // eˆ(x10)
int256 constant x11 = 6250000000000000000; // 2ˆ-4
int256 constant a11 = 106449445891785942956; // eˆ(x11)
/**
* @dev Exponentiation (x^y) with unsigned 18 decimal fixed point base and exponent.
*
* Reverts if ln(x) * y is smaller than `MIN_NATURAL_EXPONENT`, or larger than `MAX_NATURAL_EXPONENT`.
*/
function pow(uint256 x, uint256 y) internal pure returns (uint256) {
unchecked {
if (y == 0) {
// We solve the 0^0 indetermination by making it equal one.
return uint256(ONE_18);
}
if (x == 0) {
return 0;
}
// Instead of computing x^y directly, we instead rely on the properties of logarithms and exponentiation to
// arrive at that result. In particular, exp(ln(x)) = x, and ln(x^y) = y * ln(x). This means
// x^y = exp(y * ln(x)).
// The ln function takes a signed value, so we need to make sure x fits in the signed 256 bit range.
require(x >> 255 == 0, "X_OUT_OF_BOUNDS");
int256 x_int256 = int256(x);
// We will compute y * ln(x) in a single step. Depending on the value of x, we can either use ln or ln_36. In
// both cases, we leave the division by ONE_18 (due to fixed point multiplication) to the end.
// This prevents y * ln(x) from overflowing, and at the same time guarantees y fits in the signed 256 bit range.
require(y < MILD_EXPONENT_BOUND, "Y_OUT_OF_BOUNDS");
int256 y_int256 = int256(y);
int256 logx_times_y;
if (LN_36_LOWER_BOUND < x_int256 && x_int256 < LN_36_UPPER_BOUND) {
int256 ln_36_x = _ln_36(x_int256);
// ln_36_x has 36 decimal places, so multiplying by y_int256 isn't as straightforward, since we can't just
// bring y_int256 to 36 decimal places, as it might overflow. Instead, we perform two 18 decimal
// multiplications and add the results: one with the first 18 decimals of ln_36_x, and one with the
// (downscaled) last 18 decimals.
logx_times_y = ((ln_36_x / ONE_18) * y_int256 + ((ln_36_x % ONE_18) * y_int256) / ONE_18);
} else {
logx_times_y = _ln(x_int256) * y_int256;
}
logx_times_y /= ONE_18;
// Finally, we compute exp(y * ln(x)) to arrive at x^y
require(MIN_NATURAL_EXPONENT <= logx_times_y && logx_times_y <= MAX_NATURAL_EXPONENT, "PRODUCT_OUT_OF_BOUNDS");
return uint256(exp(logx_times_y));
}
}
/**
* @dev Natural exponentiation (e^x) with signed 18 decimal fixed point exponent.
*
* Reverts if `x` is smaller than MIN_NATURAL_EXPONENT, or larger than `MAX_NATURAL_EXPONENT`.
*/
function exp(int256 x) internal pure returns (int256) {
unchecked {
require(x >= MIN_NATURAL_EXPONENT && x <= MAX_NATURAL_EXPONENT, "INVALID_EXPONENT");
if (x < 0) {
// We only handle positive exponents: e^(-x) is computed as 1 / e^x. We can safely make x positive since it
// fits in the signed 256 bit range (as it is larger than MIN_NATURAL_EXPONENT).
// Fixed point division requires multiplying by ONE_18.
return ((ONE_18 * ONE_18) / exp(-x));
}
// First, we use the fact that e^(x+y) = e^x * e^y to decompose x into a sum of powers of two, which we call x_n,
// where x_n == 2^(7 - n), and e^x_n = a_n has been precomputed. We choose the first x_n, x0, to equal 2^7
// because all larger powers are larger than MAX_NATURAL_EXPONENT, and therefore not present in the
// decomposition.
// At the end of this process we will have the product of all e^x_n = a_n that apply, and the remainder of this
// decomposition, which will be lower than the smallest x_n.
// exp(x) = k_0 * a_0 * k_1 * a_1 * ... + k_n * a_n * exp(remainder), where each k_n equals either 0 or 1.
// We mutate x by subtracting x_n, making it the remainder of the decomposition.
// The first two a_n (e^(2^7) and e^(2^6)) are too large if stored as 18 decimal numbers, and could cause
// intermediate overflows. Instead we store them as plain integers, with 0 decimals.
// Additionally, x0 + x1 is larger than MAX_NATURAL_EXPONENT, which means they will not both be present in the
// decomposition.
// For each x_n, we test if that term is present in the decomposition (if x is larger than it), and if so deduct
// it and compute the accumulated product.
int256 firstAN;
if (x >= x0) {
x -= x0;
firstAN = a0;
} else if (x >= x1) {
x -= x1;
firstAN = a1;
} else {
firstAN = 1; // One with no decimal places
}
// We now transform x into a 20 decimal fixed point number, to have enhanced precision when computing the
// smaller terms.
x *= 100;
// `product` is the accumulated product of all a_n (except a0 and a1), which starts at 20 decimal fixed point
// one. Recall that fixed point multiplication requires dividing by ONE_20.
int256 product = ONE_20;
if (x >= x2) {
x -= x2;
product = (product * a2) / ONE_20;
}
if (x >= x3) {
x -= x3;
product = (product * a3) / ONE_20;
}
if (x >= x4) {
x -= x4;
product = (product * a4) / ONE_20;
}
if (x >= x5) {
x -= x5;
product = (product * a5) / ONE_20;
}
if (x >= x6) {
x -= x6;
product = (product * a6) / ONE_20;
}
if (x >= x7) {
x -= x7;
product = (product * a7) / ONE_20;
}
if (x >= x8) {
x -= x8;
product = (product * a8) / ONE_20;
}
if (x >= x9) {
x -= x9;
product = (product * a9) / ONE_20;
}
// x10 and x11 are unnecessary here since we have high enough precision already.
// Now we need to compute e^x, where x is small (in particular, it is smaller than x9). We use the Taylor series
// expansion for e^x: 1 + x + (x^2 / 2!) + (x^3 / 3!) + ... + (x^n / n!).
int256 seriesSum = ONE_20; // The initial one in the sum, with 20 decimal places.
int256 term; // Each term in the sum, where the nth term is (x^n / n!).
// The first term is simply x.
term = x;
seriesSum += term;
// Each term (x^n / n!) equals the previous one times x, divided by n. Since x is a fixed point number,
// multiplying by it requires dividing by ONE_20, but dividing by the non-fixed point n values does not.
term = ((term * x) / ONE_20) / 2;
seriesSum += term;
term = ((term * x) / ONE_20) / 3;
seriesSum += term;
term = ((term * x) / ONE_20) / 4;
seriesSum += term;
term = ((term * x) / ONE_20) / 5;
seriesSum += term;
term = ((term * x) / ONE_20) / 6;
seriesSum += term;
term = ((term * x) / ONE_20) / 7;
seriesSum += term;
term = ((term * x) / ONE_20) / 8;
seriesSum += term;
term = ((term * x) / ONE_20) / 9;
seriesSum += term;
term = ((term * x) / ONE_20) / 10;
seriesSum += term;
term = ((term * x) / ONE_20) / 11;
seriesSum += term;
term = ((term * x) / ONE_20) / 12;
seriesSum += term;
// 12 Taylor terms are sufficient for 18 decimal precision.
// We now have the first a_n (with no decimals), and the product of all other a_n present, and the Taylor
// approximation of the exponentiation of the remainder (both with 20 decimals). All that remains is to multiply
// all three (one 20 decimal fixed point multiplication, dividing by ONE_20, and one integer multiplication),
// and then drop two digits to return an 18 decimal value.
return (((product * seriesSum) / ONE_20) * firstAN) / 100;
}
}
/**
* @dev Logarithm (log(arg, base), with signed 18 decimal fixed point base and argument.
*/
function log(int256 arg, int256 base) internal pure returns (int256) {
unchecked {
// This performs a simple base change: log(arg, base) = ln(arg) / ln(base).
// Both logBase and logArg are computed as 36 decimal fixed point numbers, either by using ln_36, or by
// upscaling.
int256 logBase;
if (LN_36_LOWER_BOUND < base && base < LN_36_UPPER_BOUND) {
logBase = _ln_36(base);
} else {
logBase = _ln(base) * ONE_18;
}
int256 logArg;
if (LN_36_LOWER_BOUND < arg && arg < LN_36_UPPER_BOUND) {
logArg = _ln_36(arg);
} else {
logArg = _ln(arg) * ONE_18;
}
// When dividing, we multiply by ONE_18 to arrive at a result with 18 decimal places
return (logArg * ONE_18) / logBase;
}
}
/**
* @dev Natural logarithm (ln(a)) with signed 18 decimal fixed point argument.
*/
function ln(int256 a) internal pure returns (int256) {
unchecked {
// The real natural logarithm is not defined for negative numbers or zero.
require(a > 0, "OUT_OF_BOUNDS");
if (LN_36_LOWER_BOUND < a && a < LN_36_UPPER_BOUND) {
return _ln_36(a) / ONE_18;
} else {
return _ln(a);
}
}
}
/**
* @dev Internal natural logarithm (ln(a)) with signed 18 decimal fixed point argument.
*/
function _ln(int256 a) private pure returns (int256) {
unchecked {
if (a < ONE_18) {
// Since ln(a^k) = k * ln(a), we can compute ln(a) as ln(a) = ln((1/a)^(-1)) = - ln((1/a)). If a is less
// than one, 1/a will be greater than one, and this if statement will not be entered in the recursive call.
// Fixed point division requires multiplying by ONE_18.
return (-_ln((ONE_18 * ONE_18) / a));
}
// First, we use the fact that ln^(a * b) = ln(a) + ln(b) to decompose ln(a) into a sum of powers of two, which
// we call x_n, where x_n == 2^(7 - n), which are the natural logarithm of precomputed quantities a_n (that is,
// ln(a_n) = x_n). We choose the first x_n, x0, to equal 2^7 because the exponential of all larger powers cannot
// be represented as 18 fixed point decimal numbers in 256 bits, and are therefore larger than a.
// At the end of this process we will have the sum of all x_n = ln(a_n) that apply, and the remainder of this
// decomposition, which will be lower than the smallest a_n.
// ln(a) = k_0 * x_0 + k_1 * x_1 + ... + k_n * x_n + ln(remainder), where each k_n equals either 0 or 1.
// We mutate a by subtracting a_n, making it the remainder of the decomposition.
// For reasons related to how `exp` works, the first two a_n (e^(2^7) and e^(2^6)) are not stored as fixed point
// numbers with 18 decimals, but instead as plain integers with 0 decimals, so we need to multiply them by
// ONE_18 to convert them to fixed point.
// For each a_n, we test if that term is present in the decomposition (if a is larger than it), and if so divide
// by it and compute the accumulated sum.
int256 sum = 0;
if (a >= a0 * ONE_18) {
a /= a0; // Integer, not fixed point division
sum += x0;
}
if (a >= a1 * ONE_18) {
a /= a1; // Integer, not fixed point division
sum += x1;
}
// All other a_n and x_n are stored as 20 digit fixed point numbers, so we convert the sum and a to this format.
sum *= 100;
a *= 100;
// Because further a_n are 20 digit fixed point numbers, we multiply by ONE_20 when dividing by them.
if (a >= a2) {
a = (a * ONE_20) / a2;
sum += x2;
}
if (a >= a3) {
a = (a * ONE_20) / a3;
sum += x3;
}
if (a >= a4) {
a = (a * ONE_20) / a4;
sum += x4;
}
if (a >= a5) {
a = (a * ONE_20) / a5;
sum += x5;
}
if (a >= a6) {
a = (a * ONE_20) / a6;
sum += x6;
}
if (a >= a7) {
a = (a * ONE_20) / a7;
sum += x7;
}
if (a >= a8) {
a = (a * ONE_20) / a8;
sum += x8;
}
if (a >= a9) {
a = (a * ONE_20) / a9;
sum += x9;
}
if (a >= a10) {
a = (a * ONE_20) / a10;
sum += x10;
}
if (a >= a11) {
a = (a * ONE_20) / a11;
sum += x11;
}
// a is now a small number (smaller than a_11, which roughly equals 1.06). This means we can use a Taylor series
// that converges rapidly for values of `a` close to one - the same one used in ln_36.
// Let z = (a - 1) / (a + 1).
// ln(a) = 2 * (z + z^3 / 3 + z^5 / 5 + z^7 / 7 + ... + z^(2 * n + 1) / (2 * n + 1))
// Recall that 20 digit fixed point division requires multiplying by ONE_20, and multiplication requires
// division by ONE_20.
int256 z = ((a - ONE_20) * ONE_20) / (a + ONE_20);
int256 z_squared = (z * z) / ONE_20;
// num is the numerator of the series: the z^(2 * n + 1) term
int256 num = z;
// seriesSum holds the accumulated sum of each term in the series, starting with the initial z
int256 seriesSum = num;
// In each step, the numerator is multiplied by z^2
num = (num * z_squared) / ONE_20;
seriesSum += num / 3;
num = (num * z_squared) / ONE_20;
seriesSum += num / 5;
num = (num * z_squared) / ONE_20;
seriesSum += num / 7;
num = (num * z_squared) / ONE_20;
seriesSum += num / 9;
num = (num * z_squared) / ONE_20;
seriesSum += num / 11;
// 6 Taylor terms are sufficient for 36 decimal precision.
// Finally, we multiply by 2 (non fixed point) to compute ln(remainder)
seriesSum *= 2;
// We now have the sum of all x_n present, and the Taylor approximation of the logarithm of the remainder (both
// with 20 decimals). All that remains is to sum these two, and then drop two digits to return a 18 decimal
// value.
return (sum + seriesSum) / 100;
}
}
/**
* @dev Intrnal high precision (36 decimal places) natural logarithm (ln(x)) with signed 18 decimal fixed point argument,
* for x close to one.
*
* Should only be used if x is between LN_36_LOWER_BOUND and LN_36_UPPER_BOUND.
*/
function _ln_36(int256 x) private pure returns (int256) {
unchecked {
// Since ln(1) = 0, a value of x close to one will yield a very small result, which makes using 36 digits
// worthwhile.
// First, we transform x to a 36 digit fixed point value.
x *= ONE_18;
// We will use the following Taylor expansion, which converges very rapidly. Let z = (x - 1) / (x + 1).
// ln(x) = 2 * (z + z^3 / 3 + z^5 / 5 + z^7 / 7 + ... + z^(2 * n + 1) / (2 * n + 1))
// Recall that 36 digit fixed point division requires multiplying by ONE_36, and multiplication requires
// division by ONE_36.
int256 z = ((x - ONE_36) * ONE_36) / (x + ONE_36);
int256 z_squared = (z * z) / ONE_36;
// num is the numerator of the series: the z^(2 * n + 1) term
int256 num = z;
// seriesSum holds the accumulated sum of each term in the series, starting with the initial z
int256 seriesSum = num;
// In each step, the numerator is multiplied by z^2
num = (num * z_squared) / ONE_36;
seriesSum += num / 3;
num = (num * z_squared) / ONE_36;
seriesSum += num / 5;
num = (num * z_squared) / ONE_36;
seriesSum += num / 7;
num = (num * z_squared) / ONE_36;
seriesSum += num / 9;
num = (num * z_squared) / ONE_36;
seriesSum += num / 11;
num = (num * z_squared) / ONE_36;
seriesSum += num / 13;
num = (num * z_squared) / ONE_36;
seriesSum += num / 15;
// 8 Taylor terms are sufficient for 36 decimal precision.
// All that remains is multiplying by 2 (non fixed point).
return seriesSum * 2;
}
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
/// @dev This library implements formulas for minting and redeeming stable coin token (aToken) and leveraged (xToken)
/// tokens in a system where:
/// - There is a base token with a certain Net Asset Value (NAV).
/// - A stable coin token (aToken) with its own NAV.
/// - A leveraged token (xToken) with its own NAV.
/// All values are scaled by 1e18 before being passed to these functions.
///
/// The core equation governing the system is:
/// n * v = nf * vf + nx * vx
///
/// Where:
/// n = current total base supply
/// v = baseNav (NAV of base token)
/// nf = aSupply (supply of stable coin token)
/// vf = aNav (NAV of stable coin token)
/// nx = xSupply (supply of leveraged token)
/// vx = xNav (NAV of leveraged token)
///
/// The "collateral ratio" (cr) or "target collateral ratio" (ncr) is defined as:
/// cr = (total base value) / (stable coin token value) = (n * v) / (nf * vf)
///
/// Operations involve adjusting supplies (minting or redeeming tokens) to reach a new collateral ratio (ncr).
/// We define:
/// dn = change in base supply (how many base tokens are added or removed)
/// df = change in stable coin token supply (how many stable coin tokens are added or removed)
///
/// By setting:
/// ((n + dn) * v) / ((nf + df)*vf) = ncr
///
/// and using the base equation n * v = nf * vf + nx * vx, we can derive formulas for dn and df
/// depending on whether we are minting or redeeming under a desired collateral ratio.
library FxStableMath {
/*************
* Constants *
*************/
/// @dev The precision (scaling factor) used for calculations.
uint256 internal constant PRECISION = 1e18;
/// @dev The maximum leverage ratio allowed.
uint256 internal constant MAX_LEVERAGE_RATIO = 100e18;
/***********
* Structs *
***********/
struct SwapState {
// Current supply of the base token
uint256 baseSupply;
// Current NAV of the base token (scaled by 1e18)
uint256 baseNav;
// Current TWAP NAV of the base token (scaled by 1e18), used for stable calculations
uint256 baseTwapNav;
// Current supply of the stable coin token
uint256 aSupply;
// Current NAV of the stable coin token (scaled by 1e18)
uint256 aNav;
// Current supply of the leveraged token
uint256 xSupply;
// Current NAV of the leveraged token (scaled by 1e18)
uint256 xNav;
// A boolean parameter `beta` that may adjust behavior in some calculations
bool beta;
}
/**
* @notice Compute how much base token (dn) and stable coin tokens (df) can be minted to achieve a new collateral ratio,
* if the current collateral ratio is less than the new target.
*
* Variables (for reference):
* n: current total base supply
* v: baseNav
* nf: aSupply
* vf: aNav
* nx: xSupply
* vx: xNav
* ncr: newCollateralRatio
*
* Core equations:
* Initially:
* n * v = nf * vf + nx * vx
*
* After adding some amount of base tokens (dn) and stable coin tokens (df):
* (n + dn) * v = (nf + df) * vf + nx * vx
*
* Define collateral ratio as:
* ((n + dn) * v) / ((nf + df) * vf) = ncr
*
* From the above, solving for df and dn:
* df = ((n * v) - (ncr * nf * vf)) / ((ncr - 1) * vf)
* dn = ((n * v) - (ncr * nf * vf)) / ((ncr - 1) * v)
*
* Here, df and dn tell us how much stable coin token and base token must be added to achieve ncr.
* If dn > 0, we need to add that many base tokens; if df > 0, we can mint that many stable coin tokens.
*
* @dev If the current collateral ratio is already >= ncr, then we return 0 because no minting is needed.
*
* @param state Current state of the system.
* @param _newCollateralRatio The desired new collateral ratio (scaled by 1e18).
* @return _maxBaseIn The amount of base token required (corresponds to dn).
* @return _maxATokenMintable The amount of stable coin token (aToken) that can be minted (corresponds to df).
*/
function maxMintableAToken(
SwapState memory state,
uint256 _newCollateralRatio
) internal pure returns (uint256 _maxBaseIn, uint256 _maxATokenMintable) {
// Calculate scaled values
uint256 _baseVal = state.baseSupply * (state.baseNav) * (PRECISION);
uint256 _aVal = _newCollateralRatio * (state.aSupply) * state.aNav;
// If baseVal > aVal, we can mint
if (_baseVal > _aVal) {
// Adjust ncr to (ncr - 1)*PRECISION
_newCollateralRatio = _newCollateralRatio - (PRECISION);
uint256 _delta = _baseVal - _aVal;
// dn = delta / ((ncr - 1)*v)
_maxBaseIn = _delta / (state.baseNav * (_newCollateralRatio));
// df = delta / ((ncr - 1)*vf)
_maxATokenMintable = _delta / (state.aNav * (_newCollateralRatio));
}
}
/**
* @notice Compute how much base token (dn) and leveraged tokens (xToken) can be minted to achieve a new collateral ratio,
* if the current collateral ratio is less than the new target.
*
* Equations:
* n * v = nf * vf + nx * vx
*
* After adding dn base tokens and dx leveraged tokens:
* (n + dn)*v = nf*vf + (nx + dx)*vx
*
* The new collateral ratio condition:
* ((n + dn)*v) / (nf*vf) = ncr
*
* From this, solving for dn and dx:
* dn = (ncr * nf * vf - n * v) / v
* dx = (ncr * nf * vf - n * v) / vx
*
* @dev If the current collateral ratio >= ncr, we return 0 because no minting is needed.
*
* @param state The current state.
* @param _newCollateralRatio The desired new collateral ratio (scaled by 1e18).
* @return _maxBaseIn The amount of base token needed (dn).
* @return _maxXTokenMintable The amount of leveraged token (xToken) that can be minted (dx).
*/
function maxMintableXToken(
SwapState memory state,
uint256 _newCollateralRatio
) internal pure returns (uint256 _maxBaseIn, uint256 _maxXTokenMintable) {
uint256 _baseVal = state.baseSupply * state.baseNav * PRECISION;
uint256 _aVal = _newCollateralRatio * state.aSupply * state.aNav;
if (_aVal > _baseVal) {
uint256 _delta = _aVal - _baseVal;
// dn = delta / (v * PRECISION)
_maxBaseIn = _delta / (state.baseNav * (PRECISION));
// dx = delta / (vx * PRECISION)
_maxXTokenMintable = _delta / (state.xNav * (PRECISION));
}
}
/**
* @notice Compute how many stable coin tokens (aToken) can be redeemed and how much base token (dn) is released
* to reach the new collateral ratio if the current ratio is greater than the target.
*
* Equations:
* Initially:
* n * v = nf * vf + nx * vx
*
* After removing dn base tokens and df stable coin tokens:
* (n - dn)*v = (nf - df)*vf + nx*vx
*
* The new collateral ratio:
* ((n - dn)*v) / ((nf - df)*vf) = ncr
*
* Solve these for df and dn:
* df = (ncr * nf * vf - n * v) / ((ncr - 1)*vf)
* dn = (ncr * nf * vf - n * v) / ((ncr - 1)*v)
*
* Here, df and dn now represent how many stable coin tokens and base tokens must be removed (redeemed)
* to achieve ncr. If df > 0, it means we should redeem that many aTokens; if dn > 0, that many base tokens
* can be released.
*
* @dev If the current collateral ratio <= ncr, no redemption is needed, so return 0.
*
* @param state Current state.
* @param _newCollateralRatio Desired collateral ratio (scaled by 1e18).
* @return _maxBaseOut The amount of base token released (dn).
* @return _maxATokenRedeemable The amount of stable coin token that can be redeemed (df).
*/
function maxRedeemableAToken(
SwapState memory state,
uint256 _newCollateralRatio
) internal pure returns (uint256 _maxBaseOut, uint256 _maxATokenRedeemable) {
uint256 _baseVal = state.baseSupply * (state.baseNav) * (PRECISION);
uint256 _aVal = _newCollateralRatio * (state.aSupply) * (PRECISION);
if (_aVal > _baseVal) {
uint256 _delta = _aVal - _baseVal;
// Adjust ncr to (ncr - 1)*PRECISION
_newCollateralRatio = _newCollateralRatio - (PRECISION);
// df = delta / ((ncr - 1)*vf)
_maxATokenRedeemable = _delta / (_newCollateralRatio * (state.aNav));
// dn = delta / ((ncr - 1)*v)
_maxBaseOut = _delta / (_newCollateralRatio * (state.baseNav));
} else {
_maxBaseOut = 0;
_maxATokenRedeemable = 0;
}
}
/**
* @notice Compute how many leveraged tokens (xToken) can be redeemed and how much base token is released
* if the current collateral ratio is greater than the target.
*
* Equations:
* n * v = nf * vf + nx * vx
*
* After removing dn base tokens and dx leveraged tokens:
* (n - dn)*v = nf*vf + (nx - dx)*vx
*
* The new collateral ratio:
* ((n - dn)*v) / (nf*vf) = ncr
*
* Solve for dn and dx:
* dn = (n * v - ncr * nf * vf) / v
* dx = (n * v - ncr * nf * vf) / vx
*
* If dn > 0, that means base tokens can be redeemed; if dx > 0, that many xTokens can be redeemed.
*
* @dev If current collateral ratio <= ncr, return 0.
*
* @param state Current state.
* @param _newCollateralRatio Desired collateral ratio (scaled by 1e18).
* @return _maxBaseOut The base token released (dn).
* @return _maxXTokenRedeemable The leveraged tokens (xToken) redeemable (dx).
*/
function maxRedeemableXToken(
SwapState memory state,
uint256 _newCollateralRatio
) internal pure returns (uint256 _maxBaseOut, uint256 _maxXTokenRedeemable) {
uint256 _baseVal = state.baseSupply * (state.baseNav) * (PRECISION);
uint256 _aVal = _newCollateralRatio * (state.aSupply) * (state.aNav);
if (_baseVal > _aVal) {
uint256 _delta = _baseVal - _aVal;
// dx = delta / (vx * PRECISION)
_maxXTokenRedeemable = _delta / (state.xNav * (PRECISION));
// dn = delta / (v * PRECISION)
_maxBaseOut = _delta / (state.baseNav * (PRECISION));
} else {
_maxBaseOut = 0;
_maxXTokenRedeemable = 0;
}
}
/**
* @notice Mint stable coin tokens (aToken) given a certain amount of base tokens (dn).
*
* Equations:
* n * v = nf * vf + nx * vx
* After adding dn base and df fraction tokens:
* (n + dn)*v = (nf + df)*vf + nx*vx
*
* Solve for df given dn:
* df = (dn * v) / vf
*
* @param state Current state.
* @param _baseIn Amount of base token supplied.
* @return _aTokenOut Amount of stable coin token minted (df).
*/
function mintAToken(SwapState memory state, uint256 _baseIn) internal pure returns (uint256 _aTokenOut) {
// df = (dn * v) / vf
_aTokenOut = (_baseIn * state.baseNav) / state.aNav;
}
/**
* @notice Mint leveraged tokens (xToken) given a certain amount of base tokens (dn).
*
* Equations:
* n * v = nf * vf + nx * vx
* After adding dn base tokens and dx leveraged tokens:
* (n + dn)*v = nf*vf + (nx + dx)*vx
*
* Solve for dx:
* dx = (dn * v * nx) / (n * v - nf * vf)
*
* @param state Current state.
* @param _baseIn Amount of base token supplied.
* @return _xTokenOut Amount of leveraged token minted (dx).
*/
function mintXToken(SwapState memory state, uint256 _baseIn) internal pure returns (uint256 _xTokenOut) {
// dx = (dn * v * nx) / (n * v - nf * vf)
_xTokenOut = _baseIn * state.baseNav * state.xSupply;
_xTokenOut = _xTokenOut / (state.baseSupply * state.baseNav - state.aSupply * state.aNav);
}
/**
* @notice Redeem base tokens by supplying stable coin tokens (aToken) and/or leveraged tokens (xToken).
*
* Equations:
* n * v = nf * vf + nx * vx
* After removing df fraction tokens and dx leveraged tokens:
* (n - dn)*v = (nf - df)*vf + (nx - dx)*vx
*
* The amount of baseOut (dn*v) depends on how many fraction and leveraged tokens are provided.
*
* If xSupply = 0 (no leveraged tokens):
* baseOut = (aTokenIn * vf) / v
*
* If xSupply > 0:
* baseOut = [ (aTokenIn * vf) + (xTokenIn * (n*v - nf*vf) / nx ) ] / v
*
* @param state Current state.
* @param _aTokenIn stable coin tokens supplied.
* @param _xTokenIn Leveraged tokens supplied.
* @return _baseOut Amount of base token redeemed.
*/
function redeem(SwapState memory state, uint256 _aTokenIn, uint256 _xTokenIn) internal pure returns (uint256 _baseOut) {
uint256 _xVal = state.baseSupply * state.baseNav - state.aSupply * state.aNav;
if (state.xSupply == 0) {
_baseOut = (_aTokenIn * state.aNav) / state.baseNav;
} else {
_baseOut = _aTokenIn * state.aNav;
_baseOut += (_xTokenIn * _xVal) / state.xSupply;
_baseOut /= state.baseNav;
}
}
/**
* @notice Compute the current leverage ratio for the xToken.
*
* Define:
* rho = (aSupply * aNav) / (baseSupply * baseTwapNav)
*
* When beta = false, leverage ratio = 1 / (1 - rho)
* If under-collateralized (rho >= 1), leverage ratio = MAX_LEVERAGE_RATIO
*
* @param state Current state.
* @return ratio Current leverage ratio.
*/
function leverageRatio(SwapState memory state) internal pure returns (uint256 ratio) {
if (state.beta) return PRECISION;
if(state.baseSupply == 0 || state.baseNav == 0 || state.baseTwapNav == 0) return 0;
// rho = (aSupply * aNav * PRECISION) / (baseSupply * baseTwapNav)
uint256 rho = (state.aSupply * state.aNav * PRECISION) / (state.baseSupply * state.baseTwapNav);
if (rho >= PRECISION) {
// Under-collateralized
ratio = MAX_LEVERAGE_RATIO;
} else {
// ratio = 1 / (1 - rho)
ratio = (PRECISION * PRECISION) / (PRECISION - rho);
if (ratio > MAX_LEVERAGE_RATIO) ratio = MAX_LEVERAGE_RATIO;
}
}
}// SPDX-License-Identifier: MIT
pragma solidity 0.8.28;
interface IStrategy {
error ErrorNotPermitted();
error ErrorZeroAmount();
error ErrorZeroAddress();
error ErrorCannotRecoverToken();
error ZeroGroupId();
function withdrawToTreasury(uint256 _diff) external returns (bool);
function emergencyWithdraw() external returns (bool);
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (token/ERC20/extensions/IERC20Metadata.sol)
pragma solidity ^0.8.0;
import "../IERC20Upgradeable.sol";
/**
* @dev Interface for the optional metadata functions from the ERC20 standard.
*
* _Available since v4.1._
*/
interface IERC20MetadataUpgradeable is IERC20Upgradeable {
/**
* @dev Returns the name of the token.
*/
function name() external view returns (string memory);
/**
* @dev Returns the symbol of the token.
*/
function symbol() external view returns (string memory);
/**
* @dev Returns the decimals places of the token.
*/
function decimals() external view returns (uint8);
}{
"remappings": [
"@aave/=node_modules/@aave/",
"@account-abstraction/=node_modules/@account-abstraction/",
"@chainlink/=node_modules/@chainlink/",
"@eth-optimism/=node_modules/@chainlink/contracts/node_modules/@eth-optimism/",
"@openzeppelin/=node_modules/@openzeppelin/",
"@uniswap/=node_modules/@uniswap/",
"base64-sol/=node_modules/base64-sol/",
"ds-test/=lib/ds-test/",
"eth-gas-reporter/=node_modules/eth-gas-reporter/",
"forge-std/=lib/forge-std/src/",
"hardhat/=node_modules/hardhat/",
"solidity-bytes-utils/=node_modules/solidity-bytes-utils/",
"erc4626-tests/=lib/openzeppelin-contracts-upgradeable/lib/erc4626-tests/",
"openzeppelin-contracts-upgradeable/=lib/openzeppelin-contracts-upgradeable/",
"openzeppelin/=lib/openzeppelin-contracts-upgradeable/contracts/",
"solmate/=lib/solmate/src/",
"abdk-libraries-solidity/=node_modules/abdk-libraries-solidity/"
],
"optimizer": {
"enabled": true,
"runs": 50
},
"metadata": {
"useLiteralContent": false,
"bytecodeHash": "ipfs",
"appendCBOR": true
},
"outputSelection": {
"*": {
"*": [
"evm.bytecode",
"evm.deployedBytecode",
"devdoc",
"userdoc",
"metadata",
"abi"
]
}
},
"evmVersion": "shanghai",
"viaIR": true,
"libraries": {
"src/libs/ValidationLibrary.sol": {
"ValidationLibrary": "0xf703CAb4acf850C3B3e08DD9aa585D120016d297"
}
}
}Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
Contract ABI
API[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"},{"internalType":"address","name":"recipient","type":"address"},{"internalType":"address","name":"sender","type":"address"}],"name":"ArbitraryCallsNotAllowed","type":"error"},{"inputs":[],"name":"ConfigNotReady","type":"error"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"}],"name":"EarlyExit","type":"error"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"}],"name":"EmptyCollateralListOnUpdate","type":"error"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"}],"name":"ErrorATokenMintingPausedInStabilityMode","type":"error"},{"inputs":[],"name":"ErrorInsufficientBaseOutput","type":"error"},{"inputs":[],"name":"ErrorInsufficientTokenOutput","type":"error"},{"inputs":[],"name":"ErrorMinTokenAmount","type":"error"},{"inputs":[],"name":"ErrorMintPaused","type":"error"},{"inputs":[],"name":"ErrorRedeemPaused","type":"error"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"},{"internalType":"uint8","name":"operationType","type":"uint8"},{"internalType":"uint8","name":"feeModel","type":"uint8"},{"internalType":"bool","name":"isInStabilityMode","type":"bool"}],"name":"ErrorStabilityModeMintNotAllowed","type":"error"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"},{"internalType":"uint8","name":"operationType","type":"uint8"},{"internalType":"uint8","name":"feeModel","type":"uint8"},{"internalType":"bool","name":"isInStabilityMode","type":"bool"}],"name":"ErrorStabilityModeRedeemNotAllowed","type":"error"},{"inputs":[],"name":"InsufficientAllowance","type":"error"},{"inputs":[{"internalType":"address","name":"token","type":"address"}],"name":"InsufficientBalance","type":"error"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"}],"name":"InsufficientPreparedAmount","type":"error"},{"inputs":[{"internalType":"uint24","name":"fee","type":"uint24"}],"name":"InvalidFee","type":"error"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"}],"name":"InvalidGroupConfiguration","type":"error"},{"inputs":[],"name":"InvalidMinMaxCollateralAmount","type":"error"},{"inputs":[{"internalType":"uint256","name":"minimumAmount","type":"uint256"},{"internalType":"uint256","name":"amountReceived","type":"uint256"}],"name":"InvalidMinimumAmount","type":"error"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"},{"internalType":"uint256","name":"minimumAmount","type":"uint256"},{"internalType":"uint256","name":"amountReceived","type":"uint256"}],"name":"InvalidMinimumAmount","type":"error"},{"inputs":[],"name":"InvalidOperationType","type":"error"},{"inputs":[{"internalType":"uint8","name":"operationType","type":"uint8"}],"name":"InvalidOperationType","type":"error"},{"inputs":[],"name":"InvalidPackedData","type":"error"},{"inputs":[],"name":"InvalidPaymentAmount","type":"error"},{"inputs":[],"name":"InvalidRatios","type":"error"},{"inputs":[],"name":"InvalidSlippageRequested","type":"error"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"},{"internalType":"uint256","name":"minAmount1","type":"uint256"},{"internalType":"uint256","name":"amount0","type":"uint256"}],"name":"InvalidSlippageTolerance","type":"error"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"},{"internalType":"uint256","name":"minOutAmount","type":"uint256"},{"internalType":"uint256","name":"amountReceived","type":"uint256"}],"name":"InvalidSlippageTolerance","type":"error"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"},{"internalType":"uint256","name":"preparedAmount","type":"uint256"},{"internalType":"uint256","name":"maxAmount","type":"uint256"}],"name":"MaximumAmountExceeded","type":"error"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"NativeTokenWithdrawnFailed","type":"error"},{"inputs":[],"name":"NoFeesOwed","type":"error"},{"inputs":[],"name":"NotPermitted","type":"error"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"}],"name":"RedeemLocked","type":"error"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"},{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"TransferFailed","type":"error"},{"inputs":[],"name":"TreasuryGroupUpdateFailed","type":"error"},{"inputs":[],"name":"UnsupportedCollateralType","type":"error"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"}],"name":"UnwrappedAmountIsZero","type":"error"},{"inputs":[],"name":"ZeroAddress","type":"error"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"}],"name":"ZeroAddress","type":"error"},{"inputs":[],"name":"ZeroAmount","type":"error"},{"inputs":[],"name":"ZeroGroupId","type":"error"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"amountIn","type":"uint256"},{"indexed":true,"internalType":"uint256","name":"amountOut","type":"uint256"},{"indexed":true,"internalType":"address","name":"recipient","type":"address"}],"name":"AutoCompounded","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"GroupId","name":"groupId","type":"bytes32"}],"name":"CacheUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"Currency","name":"tokenAddress","type":"address"},{"indexed":true,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"ERC20Recovered","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"feeCollector","type":"address"},{"indexed":true,"internalType":"address","name":"token","type":"address"},{"indexed":true,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"FeeCollected","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"GroupId","name":"groupId","type":"bytes32"},{"indexed":true,"internalType":"address","name":"feeCollector","type":"address"},{"indexed":true,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"FeeCollected","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"feeCollector","type":"address"},{"indexed":false,"internalType":"uint256","name":"protocolFeeAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"hookFeeAmount","type":"uint256"}],"name":"FeeCollected","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"hookContract","type":"address"},{"indexed":true,"internalType":"uint256","name":"protocolFeeAmount","type":"uint256"},{"indexed":true,"internalType":"uint256","name":"hookRemainingFee","type":"uint256"}],"name":"FeeDelegated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"GroupId","name":"groupId","type":"bytes32"},{"indexed":true,"internalType":"address","name":"hookContract","type":"address"},{"indexed":false,"internalType":"uint256","name":"directFeeAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"hookFeeAmount","type":"uint256"}],"name":"FeeDelegated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"hookContract","type":"address"},{"indexed":true,"internalType":"address","name":"token","type":"address"},{"indexed":true,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"FeesSettled","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"GroupId","name":"groupId","type":"bytes32"},{"indexed":true,"internalType":"address","name":"hookContract","type":"address"},{"indexed":true,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"FeesSettled","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint8","name":"version","type":"uint8"}],"name":"Initialized","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"GroupId","name":"groupId","type":"bytes32"},{"indexed":true,"internalType":"address","name":"token","type":"address"},{"indexed":true,"internalType":"address","name":"recipient","type":"address"},{"indexed":false,"internalType":"address","name":"sender","type":"address"},{"indexed":false,"internalType":"uint256","name":"ytMinted","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"vtMinted","type":"uint256"}],"name":"MintToken","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"NativeTokenWithdrawn","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"GroupId","name":"groupId","type":"bytes32"},{"indexed":true,"internalType":"bool","name":"enabled","type":"bool"}],"name":"RedeemLockUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"GroupId","name":"groupId","type":"bytes32"},{"indexed":true,"internalType":"address","name":"token","type":"address"},{"indexed":true,"internalType":"address","name":"recipient","type":"address"},{"indexed":false,"internalType":"address","name":"sender","type":"address"},{"indexed":false,"internalType":"uint256","name":"baseOut","type":"uint256"}],"name":"RedeemToken","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"account","type":"address"},{"indexed":true,"internalType":"uint256","name":"safeRefundAmount","type":"uint256"}],"name":"Refund","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"role","type":"bytes32"},{"indexed":true,"internalType":"bytes32","name":"previousAdminRole","type":"bytes32"},{"indexed":true,"internalType":"bytes32","name":"newAdminRole","type":"bytes32"}],"name":"RoleAdminChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"role","type":"bytes32"},{"indexed":true,"internalType":"address","name":"account","type":"address"},{"indexed":true,"internalType":"address","name":"sender","type":"address"}],"name":"RoleGranted","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"role","type":"bytes32"},{"indexed":true,"internalType":"address","name":"account","type":"address"},{"indexed":true,"internalType":"address","name":"sender","type":"address"}],"name":"RoleRevoked","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"fromToken","type":"address"},{"indexed":true,"internalType":"address","name":"toToken","type":"address"},{"indexed":false,"internalType":"uint256","name":"amountIn","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amountOut","type":"uint256"}],"name":"TokensSwapped","type":"event"},{"stateMutability":"payable","type":"fallback"},{"inputs":[],"name":"DEFAULT_ADMIN_ROLE","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"adminAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"tokenRegistry","type":"address"},{"internalType":"GroupId","name":"groupId","type":"bytes32"}],"name":"forceUpdateGroupCache","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"getGlobalMintFlag","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getGlobalRedeemFlag","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"}],"name":"getGroupMintFlag","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"}],"name":"getGroupRedeemFlag","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"}],"name":"getGroupStabilityFlag","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"},{"internalType":"address","name":"hookContract","type":"address"},{"internalType":"address","name":"token","type":"address"}],"name":"getOwedFees","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"}],"name":"getRoleAdmin","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"},{"internalType":"address","name":"user","type":"address"}],"name":"getUserOperationTime","outputs":[{"internalType":"uint256","name":"lastVTMintTime","type":"uint256"},{"internalType":"uint256","name":"lastYTMintTime","type":"uint256"},{"internalType":"uint256","name":"lastVTRedeemTime","type":"uint256"},{"internalType":"uint256","name":"lastYTRedeemTime","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"account","type":"address"}],"name":"grantRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"account","type":"address"}],"name":"hasRole","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_admin","type":"address"}],"name":"initialize","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"components":[{"components":[{"internalType":"Currency","name":"aToken","type":"address"},{"internalType":"Currency","name":"xToken","type":"address"},{"internalType":"Currency","name":"baseToken","type":"address"},{"internalType":"Currency","name":"yieldBearingToken","type":"address"},{"internalType":"Currency","name":"wethToken","type":"address"}],"internalType":"struct DTokenRegistry.GroupCore","name":"core","type":"tuple"}],"internalType":"struct GroupKey","name":"groupKey","type":"tuple"},{"components":[{"internalType":"uint8","name":"operationType","type":"uint8"},{"internalType":"uint256","name":"baseIn","type":"uint256"},{"internalType":"uint24","name":"slippage","type":"uint24"},{"internalType":"address","name":"paymentToken","type":"address"},{"internalType":"bytes","name":"hookData","type":"bytes"}],"internalType":"struct DProtocol.MintParams","name":"params","type":"tuple"}],"name":"mintToken","outputs":[{"components":[{"internalType":"uint256","name":"ytMinted","type":"uint256"},{"internalType":"uint256","name":"vtMinted","type":"uint256"}],"internalType":"struct DProtocol.MintResult","name":"result","type":"tuple"}],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"tokenAddress","type":"address"},{"internalType":"uint256","name":"tokenAmount","type":"uint256"}],"name":"recoverERC20","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"recoversEther","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"components":[{"components":[{"internalType":"Currency","name":"aToken","type":"address"},{"internalType":"Currency","name":"xToken","type":"address"},{"internalType":"Currency","name":"baseToken","type":"address"},{"internalType":"Currency","name":"yieldBearingToken","type":"address"},{"internalType":"Currency","name":"wethToken","type":"address"}],"internalType":"struct DTokenRegistry.GroupCore","name":"core","type":"tuple"}],"internalType":"struct GroupKey","name":"groupKey","type":"tuple"},{"components":[{"internalType":"uint8","name":"operationType","type":"uint8"},{"internalType":"uint256","name":"baseOut","type":"uint256"},{"internalType":"uint24","name":"slippage","type":"uint24"},{"internalType":"address","name":"desiredCollateral","type":"address"},{"internalType":"bytes","name":"hookData","type":"bytes"}],"internalType":"struct DProtocol.RedeemParams","name":"params","type":"tuple"}],"name":"redeemToken","outputs":[{"internalType":"uint256","name":"baseOut","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"","type":"bytes32"},{"internalType":"address","name":"","type":"address"}],"name":"renounceRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"account","type":"address"}],"name":"revokeRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"value","type":"bool"}],"name":"setGlobalMintFlag","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"value","type":"bool"}],"name":"setGlobalRedeemFlag","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"},{"internalType":"bool","name":"value","type":"bool"}],"name":"setGroupMintFlag","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"},{"internalType":"bool","name":"value","type":"bool"}],"name":"setGroupRedeemFlag","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"},{"internalType":"bool","name":"value","type":"bool"}],"name":"setGroupStabilityFlag","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"},{"internalType":"address","name":"token","type":"address"}],"name":"settleOwedFees","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"GroupId","name":"groupId","type":"bytes32"}],"name":"stabilityRatio","outputs":[{"internalType":"uint96","name":"","type":"uint96"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes4","name":"interfaceId","type":"bytes4"}],"name":"supportsInterface","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"stateMutability":"payable","type":"receive"}]Contract Creation Code
60808060405234610117575f5460ff8160081c16159182809361010a575b80156100f3575b1561009a575060ff1981166001175f5581610088575b5061004f575b604051615ad2908161011c8239f35b61ff00195f54165f557f7f26b83ff96e1f2b6a682f133852f6798a09c465da95921460cefb3847402498602060405160018152a1610040565b61ffff1916610101175f90815561003a565b62461bcd60e51b815260206004820152602e60248201527f496e697469616c697a61626c653a20636f6e747261637420697320616c72656160448201526d191e481a5b9a5d1a585b1a5e995960921b6064820152608490fd5b50303b1580156100245750600160ff831614610024565b50600160ff83161061001d565b5f80fdfe6080806040526004361015610027575b503615610025576339218f3b60e01b5f5260045ffd5b005b5f905f3560e01c90816301ffc9a71461168e575080630587bcf3146116525780630795a7ba146115b657806309ae48411461143a578063248a9ca3146114075780632f2ff15d1461134e57806336568abe146112f4578063386d81fa1461128c578063487523f41461123c578063795c1af9146111ec5780638980f11f1461114b5780638d390d311461111457806391d14854146110cb578063942a13ef14611096578063a217fddf1461107c578063a935508714611035578063b06576e814611005578063bb5e558c14610fd1578063c4d66de814610ded578063cccc6f4c14610dc9578063d1d67e1814610bb5578063d547741f14610afb578063e3dab550146102a3578063eb9915a714610280578063f0e8cb1614610219578063f9de6313146101bd578063fa7f0c4e1461018e5763fc6f94680361000f573461018b578060031936011261018b5760ce546040516001600160a01b039091168152602090f35b80fd5b503461018b5760206101b06101a2366116fb565b906101ab61261d565b61235d565b6001609755604051908152f35b503461018b57604036600319011261018b576004356101da61177a565b6101e2612de2565b811561020a5781835260cd6020526001196040842054161790825260cd602052604082205580f35b633960608d60e11b8352600483fd5b503461018b57604036600319011261018b57604061023561173a565b91600435815260cb602052209060018060a01b03165f52602052608060405f208054906001810154906003600282015491015491604051938452602084015260408301526060820152f35b503461018b578060031936011261018b576020600160cc54166040519015158152f35b503461063a57604036600319011261063a576102bd611750565b90602435916102ca61261d565b6102d2612de2565b6001600160a01b03168015610ae857825f5260c960205260405f2060405190635ab3ea5d60e11b82528460048301525f82602481865afa918215610add575f926108e9575b508151516001600160a01b0316156106fa576060820180515180156108d657600d83019182545f845580610883575b505f5b8281106107c9575050835180519092506001600160a01b031615905080156107b4575b801561079f575b801561078a575b61066357805160208201516001600160a01b03918216911681811492918315610773575b831561075c575b508215610745575b821561072e575b50811561070d575b506106fa5760208201805180519193916001600160a01b0316159081156106e4575b81156106ce575b81156106b8575b81156106a2575b811561068c575b8115610676575b50610663578051805183546001600160a01b03199081166001600160a01b0392831617855560208084015160018781018054851692861692909217909155604080860151600289018054861691871691909117905560608087015160038a018054871691881691909117905560809687015160048a01805487169188169190911790559851805160058a018054871691881691909117905592830151600689018054861691871691909117905582810151600789018054861691871691909117905597820151600888018054851691861691909117905581850151600988018054851691861691909117905560a082810151600a89018054861691871691909117905560c090920151600b88018054851691861691909117905596850151600c870155840151600f8601559190920151600e840180549092169216919091179055436010820155601101805460ff19169091179055827f67795b7df97de6699e0f17226d6111be2ddc0ebe76fb110c2e8ab6df9312579f5f80a2825f5260c960205260ff601160405f2001541615610650575f83815260c960205260409020600801546001600160a01b031692831561063e57833b1561063a5760405191630e3dab5560e41b8352600483015260248201525f8160448183875af19081610625575b5061061d573d8091604051936308c99ae160e21b8552600485015260406024850152816044850152606484013e601f801991011660640190fd5b600160975580f35b6106329192505f90611856565b5f905f6105e3565b5f80fd5b6308c99ae160e21b5f5260045260245ffd5b826337e1342d60e21b5f5260045260245ffd5b85635dbad7a160e01b5f5260045260245ffd5b60c001516001600160a01b03161590505f610401565b60a08101516001600160a01b03161591506103fa565b60808101516001600160a01b03161591506103f3565b60608101516001600160a01b03161591506103ec565b60408101516001600160a01b03161591506103e5565b60208101516001600160a01b03161591506103de565b84635dbad7a160e01b5f5260045260245ffd5b60408101516060909101516001600160a01b0390811691161490505f6103bc565b60608201516001600160a01b03161491505f6103b4565b60408201516001600160a01b0316811492506103ad565b60608301516001600160a01b03161492505f6103a5565b60408301516001600160a01b03168114935061039e565b5060608101516001600160a01b03161561037a565b5060408101516001600160a01b031615610373565b5060208101516001600160a01b03161561036c565b6107d4818351614195565b5184549190600160401b83101561086f576001830180875583101561085b575f86815260209081902082516003909502018054918301516001600160a81b03199092166001600160a01b03959095169490941760a09190911b60ff60a01b161783556040810151600184810191909155606091909101516002939093019290925501610349565b634e487b7160e01b5f52603260045260245ffd5b634e487b7160e01b5f52604160045260245ffd5b806003029060038204036108c257835f5260205f20908101905b8181106108aa5750610346565b805f600392555f60018201555f60028201550161089d565b634e487b7160e01b5f52601160045260245ffd5b8663f4ac45b160e01b5f5260045260245ffd5b9091503d805f833e6108fb8183611856565b81019060208183031261063a578051906001600160401b03821161063a570180820391610200831261063a576040519261093484611798565b60a0811261063a5760e09060405161094b816117b3565b61095485611938565b815261096260208601611938565b602082015261097360408601611938565b604082015261098460608601611938565b606082015261099560808601611938565b60808201528552609f19011261063a576040516109b1816117ce565b6109bd60a08401611938565b81526109cb60c08401611938565b60208201526109dc60e08401611938565b60408201526109ee6101008401611938565b6060820152610a006101208401611938565b6080820152610a126101408401611938565b60a0820152610a246101608401611938565b60c0820152602084015261018082015160408401526101a08201516001600160401b03811161063a57820181601f8201121561063a578051610a65816126fa565b92610a736040519485611856565b81845260208085019260071b8401019281841161063a57602001915b838310610ac3575050505060608301526101e090610ab06101c08201611938565b6080840152015160a0820152905f610317565b6020608091610ad2848661194c565b815201920191610a8f565b6040513d5f823e3d90fd5b826324df413360e21b5f5260045260245ffd5b3461063a57604036600319011261063a57600435610b1761173a565b90610b36610b31825f526065602052600160405f20015490565b612ec1565b805f52606560205260405f2060018060a01b0383165f5260205260ff60405f205416610b5e57005b5f8181526065602090815260408083206001600160a01b0395909516808452949091528120805460ff19169055339291907ff6391f5c32d9c69d2a47ea670b442974b53935d1edc7fd64eb21e047a839171b9080a4005b3461063a57602036600319011261063a576004355f60a0604051610bd881611798565b610be061269a565b8152610bea6126c4565b60208201528260408201526060808201528260808201520152805f5260c960205260405f209060ff60118301541615610db75750604051610c2a81611798565b604051610c36816117b3565b82546001600160a01b039081168252600184015481166020830152600284015481166040808401919091526003850154821660608401526004850154909116608083015290825251610c87816117ce565b60058301546001600160a01b0390811682526006840154811660208084019190915260078501548216604080850191909152600886015483166060850152600986015483166080850152600a860154831660a0850152600b86015490921660c0840152830191909152600c83015490820152600d82018054610d08816126fa565b91610d166040519384611856565b81835260208301905f5260205f205f915b838310610d6d5760608601859052600e8701546001600160a01b03166080870152600f87015460a08701819052604051603b9190911c6001600160601b03168152602090f35b60036020600192604051610d8081611804565b60ff8654868060a01b038116835260a01c168382015284860154604082015260028601546060820152815201920192019190610d27565b6337e1342d60e21b5f5260045260245ffd5b3461063a575f36600319011261063a576020600160cc54811c166040519015158152f35b3461063a57602036600319011261063a57610e06611750565b5f549060ff8260081c161591828093610fc4575b8015610fad575b15610f515760ff1981166001175f5582610f40575b506001600160a01b03168015610f3157610e6860ff5f5460081c16610e5a81612f5b565b610e6381612f5b565b612f5b565b60016097555f8181525f516020615a7d5f395f51905f52602052604090205460ff1615610ee0575b60ce80546001600160a01b031916919091179055610eaa57005b61ff00195f54165f557f7f26b83ff96e1f2b6a682f133852f6798a09c465da95921460cefb3847402498602060405160018152a1005b5f8181525f516020615a7d5f395f51905f5260205260408120805460ff19166001179055339082907f2f8788117e7eff1d82e926ec794901d17c78024a50270940304540a733656f0d8180a4610e90565b63d92e233d60e01b5f5260045ffd5b61ffff1916610101175f5582610e36565b60405162461bcd60e51b815260206004820152602e60248201527f496e697469616c697a61626c653a20636f6e747261637420697320616c72656160448201526d191e481a5b9a5d1a585b1a5e995960921b6064820152608490fd5b50303b158015610e215750600160ff821614610e21565b50600160ff821610610e1a565b3461063a57602036600319011261063a57610fea611789565b610ff2612de2565b60021960cc54169060011b1760cc555f80f35b3461063a57602036600319011261063a5761101e611789565b611026612de2565b60011960cc54161760cc555f80f35b3461063a57602036600319011261063a57600435801561106d575f5260cd6020526020600160405f205460021c166040519015158152f35b633960608d60e11b5f5260045ffd5b3461063a575f36600319011261063a5760206040515f8152f35b3461063a57602036600319011261063a57600435801561106d575f5260cd6020526020600160405f2054166040519015158152f35b3461063a57604036600319011261063a576110e461173a565b6004355f52606560205260405f209060018060a01b03165f52602052602060ff60405f2054166040519015158152f35b3461063a57602036600319011261063a57600435801561106d575f5260cd6020526020600160405f2054811c166040519015158152f35b3461063a57604036600319011261063a57611164611750565b6024359061117061261d565b611178612de2565b6001600160a01b03168015610f315761119130826128ed565b82116111da5760ce546111af9083906001600160a01b031683612f00565b7f505b28e6941631badc363841ecbf8e1214b9379c643936458e87be718e1579995f80a36001609755005b63112fed8b60e31b5f5260045260245ffd5b3461063a57604036600319011261063a5760043561120861177a565b611210612de2565b811561106d57815f5260cd60205260041960405f205416915f5260cd60205260021b1760405f20555f80f35b3461063a57604036600319011261063a5760043561125861177a565b611260612de2565b811561106d57815f5260cd60205260021960405f205416915f5260cd60205260011b1760405f20555f80f35b3461063a57606036600319011261063a576112a561173a565b6044356001600160a01b038116919082900361063a576004355f90815260ca602090815260408083206001600160a01b0394851684528252808320949093168252928352819020549051908152f35b3461063a57604036600319011261063a5761130d61173a565b5060405162461bcd60e51b8152602060048201526018602482015277149bdb195cc818d85b89dd081899481c995b9bdd5b98d95960421b6044820152606490fd5b3461063a57604036600319011261063a5760043561136a61173a565b90611384610b31825f526065602052600160405f20015490565b805f52606560205260405f2060018060a01b0383165f5260205260ff60405f205416156113ad57005b5f8181526065602090815260408083206001600160a01b0395909516808452949091528120805460ff19166001179055339291907f2f8788117e7eff1d82e926ec794901d17c78024a50270940304540a733656f0d9080a4005b3461063a57602036600319011261063a5760206114326004355f526065602052600160405f20015490565b604051908152f35b3461063a57604036600319011261063a5760043561145661173a565b61145e61261d565b815f5260c960205260ff601160405f20015416156115a3575f82815260c960205260409020600e01546001600160a01b03169033829003611594576001600160a01b0316918215611582575f81815260ca602090815260408083206001600160a01b0386811685529083528184209087168452909152902054801561156f575f82815260ca602090815260408083206001600160a01b03878116855290835281842090881684529091528120558061151630866128ed565b106115585760208161154a7fd3093803bade32f00a1623cb5300e4caed82036accaa6aafadc9f55827a83c08938688612f00565b604051908152a46001609755005b8382630c29241b60e21b5f5260045260245260445ffd5b506393aea85b60e01b5f5260045260245ffd5b6383cf4a2b60e01b5f5260045260245ffd5b6339218f3b60e01b5f5260045ffd5b506337e1342d60e21b5f5260045260245ffd5b3461063a57602036600319011261063a576004356115d261261d565b6115da612de2565b47811161163f57807f7bcc4dd1d0cf712ca981d90a999e6ea7d3da3e11844d63b289bd1c15c2ec5e8d5f80a25f8080808460018060a01b0360ce54165af161162061232e565b501561162d576001609755005b631b1eb84f60e31b5f5260045260245ffd5b63112fed8b60e31b5f525f60045260245ffd5b6040611677611660366116fb565b90611669611877565b5061167261261d565b611c96565b600160975560208251918051835201516020820152f35b3461063a57602036600319011261063a576004359063ffffffff60e01b821680920361063a57602091632e2ecea560e01b81149081156116d0575b5015158152f35b637965db0b60e01b8114915081156116ea575b50836116c9565b6301ffc9a760e01b149050836116e3565b90600319820160c0811261063a5760a01361063a5760049160a435906001600160401b03821161063a5760a090829003600319011261063a5760040190565b602435906001600160a01b038216820361063a57565b600435906001600160a01b038216820361063a57565b35906001600160a01b038216820361063a57565b60243590811515820361063a57565b60043590811515820361063a57565b60c081019081106001600160401b0382111761086f57604052565b60a081019081106001600160401b0382111761086f57604052565b60e081019081106001600160401b0382111761086f57604052565b604081019081106001600160401b0382111761086f57604052565b608081019081106001600160401b0382111761086f57604052565b61010081019081106001600160401b0382111761086f57604052565b606081019081106001600160401b0382111761086f57604052565b90601f801991011681019081106001600160401b0382111761086f57604052565b60405190611884826117e9565b5f6020838281520152565b91908260a091031261063a5760405191602083018381106001600160401b0382111761086f576040528261190f6080604051936118cb856117b3565b6118d481611766565b85526118e260208201611766565b60208601526118f360408201611766565b604086015261190460608201611766565b606086015201611766565b608083015252565b356001600160a01b038116810361063a5790565b5190811515820361063a57565b51906001600160a01b038216820361063a57565b919082608091031261063a5760405161196481611804565b809261196f81611938565b8252602081015160ff8116810361063a5760609182916020850152604081015160408501520151910152565b919060a08382031261063a576119bd9060206119b68561192b565b940161194c565b90565b80516001600160a01b0316825260208082015160ff169083015260408082015190830152606090810151910152565b90602080835192838152019201905f5b818110611a0c5750505090565b9091926020608082611a2160019488516119c0565b0194019291016119ff565b3560ff8116810361063a5790565b3562ffffff8116810361063a5790565b903590601e198136030182121561063a57018035906001600160401b03821161063a5760200191813603831361063a57565b6001600160401b03811161086f57601f01601f191660200190565b929192611aa382611a7c565b91611ab16040519384611856565b82948184528183011161063a578281602093845f960137010152565b5f5b838110611ade5750505f910152565b8181015183820152602001611acf565b90602091611b0781518092818552858086019101611acd565b601f01601f1916010190565b80516001600160a01b03908116835260208083015182169084015260408083015182169084015260608083015182169084015260809182015116910152565b80516001600160a01b03908116835260208083015182169084015260408083015182169084015260608083015182169084015260808083015182169084015260a08083015182169084015260c09182015116910152565b9060cc825260208201526060604082015281516060820152610380610100611bf5602080860151610340608087015280516103a0870152015160406103c08601526103e0850190611aee565b936040810151611c0960a086018251611b13565b611c1c6020820151610140870190611b52565b60408101516102208601526060808201516102408701526080918201516001600160a01b0390811661026088015290830151610280870152908201516102a086015260a082015162ffffff166102c086015260c0820151166102e085015260e0810151611c8e906103008601906119c0565b015191015290565b611ca4611ca991369061188f565b612673565b611cb281612711565b6060808201515f9473f703cab4acf850c3b3e08dd9aa585d120016d297939092602084013590840160a0611d146001600160a01b03611cf084611917565b1695604051968792839263196896eb60e11b845285600485015260a48401906119ef565b908660248401526044830152600160648301523460848301520381895af4938415610add575f905f956122fa575b50156122eb575f198214806122d1575b6122ac575b5080611d7b5750505050505060405190611d70826117e9565b808252602082015290565b611e0793611dfd9260405192611d90846117e9565b83525f6020840152611da182611a2c565b9462ffffff611dc0611db560408601611a3a565b946080810190611a4a565b94909560ff60405199611dd28b61181f565b168952602089015216604087015233606087015260808601528760a086015260c08501523691611a97565b60e0820152612a3c565b90803b1561063a575f604051809263469c699160e11b82528180611e2f878960048401611ba9565b03915af48015610add57612297575b50611e47611877565b50611e5181612fd6565b6040820151608001516001600160a01b031692908315159081612280575b611e7d606085015185613336565b611e8b879993949299612fd6565b9860405199611e998b6117e9565b5f808c5260208c01526001600160a01b03611eb38a614a2e565b60408b01516020015160c001516001600160a01b0316939116919082158015612278575b61226457611ee5908b614a46565b60ff60208c01515116611ef78c614b72565b916060611f05863086614ba1565b91019285845180841061224e575b5050505080155f146120635750611f4c935090602091519283888d516040519788958694859363095a7a1b60e21b855260048501614ca8565b03925af1918215612058578592612024575b50811561200a57508692611fba7fac903ad8fe2e7a6d229683df0f465af874089964b0ea74e096f95d106cd333b79360609360208e01525b60208d0180519097508015612002575b60808c0152611fb48b613f12565b8b613f27565b8a5194516040805133815260208101979097528601526001600160a01b03908116951693a4611fea575b50505090565b338314611fe457611ffa9261409c565b5f8080611fe4565b508c51611fa6565b88516323dc081560e01b8652600452602452604452606483fd5b9091506020813d602011612050575b8161204060209383611856565b8101031261063a5751905f611f5e565b3d9150612033565b6040513d87823e3d90fd5b91929160010361223a5790602061209988949351928d51604051968780948193637ef0584160e11b835288309160048501614ca8565b03925af192831561222f5787936121fb575b5082156121df575060408a01515151869392916020916001600160a01b03166120d5853083614ba1565b90858583106121c4575b505050604460405180968193636e553f6560e01b835286600484015260018060a01b0316968760248401525af19283156121b957908995949392918793612179575b509282611fba927fac903ad8fe2e7a6d229683df0f465af874089964b0ea74e096f95d106cd333b796946060967f491ac6e8661dcfb6555d94e6fd4b55961faf240fb9fad5774ba6a93532f330e38b80a48d52611f96565b919395509391506020813d6020116121b1575b8161219960209383611856565b8101031261063a575188949193919290916060612121565b3d915061218c565b6040513d88823e3d90fd5b6121d16121d79387613f05565b91614bf0565b5f80856120df565b8a516323dc081560e01b88526004526024526044829052606486fd5b9092506020813d602011612227575b8161221760209383611856565b8101031261063a5751915f6120ab565b3d915061220a565b6040513d89823e3d90fd5b8a5163a78a5d1160e01b8852600452602487fd5b61225b936121d191613f05565b5f808581611f13565b8a516383cf4a2b60e01b8852600452602487fd5b508315611ed7565b338514611e6f576122928484876131c6565b611e6f565b6122a49193505f90611856565b5f915f611e3e565b6122cb91506001600160a01b03906122c390611917565b1633906128ed565b5f611d57565b506001600160a01b036122e382611917565b161515611d52565b63436cb04160e01b5f5260045ffd5b905061231f91945060a03d60a011612327575b6123178183611856565b81019061199b565b93905f611d42565b503d61230d565b3d15612358573d9061233f82611a7c565b9161234d6040519384611856565b82523d5f602084013e565b606090565b611ca461236e91939293369061188f565b9161237883612711565b9273f703cab4acf850c3b3e08dd9aa585d120016d2979160608501519060208101359560a06123d6600180831b036123b260608601611917565b1694604051958692839263196896eb60e11b845285600485015260a48401906119ef565b908b602484015260448301525f60648301525f60848301520381885af4928315610add575f905f946125f9575b50156122eb575f1987146125a7575b8661241e575050505050565b95611dfd916124999495969760405192612437846117e9565b83525f602084015261244882611a2c565b9462ffffff61245c611db560408601611a3a565b94909560ff6040519961246e8b61181f565b168952602089015216604087015233606087015260808601528660a086015260c08501523691611a97565b91803b1561063a575f604051809263469c699160e11b825281806124c1888860048401611ba9565b03915af48015610add57612597575b506124da82612fd6565b60408301516080015190926001600160a01b039091169182151580612580575b61251a6125208661250f606087015187613a57565b828996929396613f27565b86614313565b60405190966001600160a01b0390811692169084907fdd8b8dbb53fec7033579b7466dc8fd28b088f5b31bca605ca8fa24a570e366ca90806125638b3383612fbb565b0390a46125705750505090565b338314611fe457611ffa92614641565b3384146124fa576125928383866141a9565b6124fa565b5f6125a191611856565b5f6124d0565b9550600260ff6125b683611a2c565b16036125dc578551602001516125d69033906001600160a01b03166128ed565b95612412565b602086015160c001516125d69033906001600160a01b03166128ed565b905061261591935060a03d60a011612327576123178183611856565b92905f612403565b60026097541461262e576002609755565b60405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152606490fd5b60405161268560208201809351611b13565b60a0815261269460c082611856565b51902090565b604051906126a7826117b3565b5f6080838281528260208201528260408201528260608201520152565b604051906126d1826117ce565b5f60c0838281528260208201528260408201528260608201528260808201528260a08201520152565b6001600160401b03811161086f5760051b60200190565b905f60a060405161272181611798565b61272961269a565b81526127336126c4565b60208201528260408201526060808201528260808201520152815f5260c960205260405f209160ff60118401541615610db7575060405161277381611798565b60405161277f816117b3565b83546001600160a01b0390811682526001850154811660208301526002850154811660408084019190915260038601548216606084015260048601549091166080830152908252516127d0816117ce565b60058401546001600160a01b0390811682526006850154811660208084019190915260078601548216604080850191909152600887015483166060850152600987015483166080850152600a870154831660a0850152600b87015490921660c0840152830191909152600c84015490820152600d83018054909390612854816126fa565b906128626040519283611856565b80825260208201955f5260205f20955f905b8282106128a3575050506060830152600e8101546001600160a01b03166080830152600f015460a08201529150565b600360206001926040516128b681611804565b60ff8c54868060a01b038116835260a01c1683820152848c0154604082015260028c01546060820152815201980191019096612874565b6001600160a01b03168061290057503190565b6040516370a0823160e01b81526001600160a01b039092166004830152602090829060249082905afa908115610add575f9161293a575090565b90506020813d602011612961575b8161295560209383611856565b8101031261063a575190565b3d9150612948565b60405190612976826117e9565b60606020835f81520152565b6040519061298f82611804565b5f6060838281528260208201528260408201520152565b6040519061012082018281106001600160401b0382111761086f576040525f610100838281526129d4612969565b60208201526040516129e5816117b3565b6129ed61269a565b81526129f76126c4565b602082015283604082015283606082015283608082015260408201528260608201528260808201528260a08201528260c0820152612a33612982565b60e08201520152565b612a446129a6565b506060810180519092906001600160a01b031615612dcb57602082019283518051159081612dbe575b50612da7576040830162ffffff8151166103e88111612d875750612a8f6129a6565b60a0850180518252865151606083015295516020015160808083019190915292516001600160a01b031660c08201529184015160e08301528451905162ffffff16906103e88211612d6c575060a082015260c0830192835180519060208101519060a0810151604082015191608060018060a01b03910151169260405194612b16866117b3565b855260208501526040840152606083015260808201526040830152819360ff8251169080519160a08301805160018060601b0390603b1c169051936060602060018060a01b039201510151169060208a5160246040518095819363f3c4799760e01b835260048301525afa5f9281612d38575b50612ba2578951637055368360e11b5f5260045260245ffd5b81999293949596979899911115612d245761010087018190528290609b86901c6001600160601b03161115612d0d57612be2915160408551015190614757565b915b60a08151015192612bf58194615442565b908151151580612d01575b612cf3575b5060200151612ce6575b604090510151928360f01c612c22612969565b5060038311612cd3576127108111612cc15750612710620fffff841611612ca9579062ffff0062ffffff60201b92600160501b600160b01b0390604b1c169460e81c16179160201b16171791612c7783615472565b15612c9a5760e0602092015160405193612c90856117e9565b8452828401520152565b6338bfed7360e11b5f5260045ffd5b62ffffff83634d72eb2f60e11b5f521660045260245ffd5b634d72eb2f60e11b5f5260045260245ffd5b82631ac6d2b760e21b5f5260045260245ffd5b6220000090921791612c0f565b628000001793506020612c05565b50602082015115612c00565b612d1e9151604085510151906146f1565b91612be4565b505163ef6f777560e01b5f5260045260245ffd5b9092506020813d602011612d64575b81612d5460209383611856565b8101031261063a5751915f612b89565b3d9150612d47565b63751419d760e11b5f526004526024526103e860445260645ffd5b60a085015163751419d760e11b5f526004526024526103e860445260645ffd5b60a0830151630180267160e01b5f5260045260245ffd5b602091500151155f612a6d565b60a08201516383cf4a2b60e01b5f5260045260245ffd5b335f9081525f516020615a7d5f395f51905f52602052604090205460ff1615612e0757565b612ebd612e1333614cf2565b612e9d6011612e215f614dd8565b92603760405194859276020b1b1b2b9b9a1b7b73a3937b61d1030b1b1b7bab73a1604d1b6020850152612e5d8151809260208688019101611acd565b83017001034b99036b4b9b9b4b733903937b6329607d1b83820152612e8c825180936020604885019101611acd565b01010301601f198101835282611856565b60405162461bcd60e51b8152602060048201529182916024830190611aee565b0390fd5b5f81815260656020908152604080832033845290915290205460ff1615612ee55750565b612ebd90612e9d6011612e21612efa33614cf2565b93614dd8565b9091906001600160a01b031680612f2057630c5d6fad60e21b5f5260045ffd5b612f54612f5993612f4660405194859263a9059cbb60e01b602085015260248401612fbb565b03601f198101845283611856565b614e7a565b565b15612f6257565b60405162461bcd60e51b815260206004820152602b60248201527f496e697469616c697a61626c653a20636f6e7472616374206973206e6f74206960448201526a6e697469616c697a696e6760a81b6064820152608490fd5b6001600160a01b039091168152602081019190915260400190565b60c081019060018060a01b0382511690338203612ffc575050516001600160a01b031690565b516364e9666360e11b5f526004526024523360445260645ffd5b90916119bd949260018060a01b031682526020820152608060408201528251608082015261010061306b60208086015161034060a086015280516103c0860152015160406103e0850152610400840190611aee565b93604081015161307f60c085018251611b13565b6130926020820151610160860190611b52565b60408101516102408501526060808201516102608601526080918201516001600160a01b03908116610280870152908301516102a0860152908201516102c085015260a082015162ffffff166102e085015260c08201511661030084015260e0810151613104906103208501906119c0565b01516103a08201526060818403910152611aee565b919082018092116108c257565b51906001600160e01b03198216820361063a57565b81601f8201121561063a57805161315181611a7c565b9261315f6040519485611856565b8184526020828401011161063a576119bd9160208085019101611acd565b909160608284031261063a5761319282613126565b9260208301516001600160401b03811161063a576040916131b491850161313b565b92015162ffffff8116810361063a5790565b909160016131e08260408061ffff920151015160201c1690565b16156133315760208101916132328284519261322d613204602086015195516154cf565b97612f46628000008a16159660405195869363e3b7f38160e01b60208601523360248601613016565b6147a7565b8051801561332957608011801561331d575b613309578060208061325b9351830101910161317d565b9095916001600160e01b031916631c480c7f60e01b016132f55761327e90614823565b929091826132e6575b826132c5575b50506132b5575b5060400151608001516001600160a01b03166132ae575050565b5160200152565b6132bf9082614850565b5f613294565b81159250906132d7575b505f8061328d565b6220000091501615155f6132cf565b62ffffff841615159250613287565b8351632014ceb360e21b5f5260045260245ffd5b8251632014ceb360e21b5f5260045260245ffd5b50805161044010613244565b505050505050565b505050565b9061333f612982565b508015613a4357815160ff602084015151169061335b84612fd6565b9180613a0a57505f5260cb60205260405f209060018060a01b03165f5260205260405f204290555b60e082018051519093906001600160a01b031615613a02575b6133a4612982565b508251936133b184614b72565b906133bb85613f12565b8151516001600160a01b03908116959192919084161580156139f1575b6139de57604087015160200151518251516001600160a01b0390811699911661340089613f12565b99604051906341976e0960e01b82526004820152604081602481855afa9a8b15610add575f915f9c6139ba575b50604080516341976e0960e01b81526001600160a01b0390921660048301529092839060249082905afa918215610add575f905f93613986575b508b1591821561397d575b8215613974575b50811561396b575b506139575760e08901516020015160ff9081161692670de0b6b3a76400008102818104670de0b6b3a764000014821517156108c257826134d68d6134d16134db946134cb8a614305565b906142e7565b614165565b6142e7565b60606134e7828d614a46565b0151908115613944578181111561393c57670de0b6b3a7640000820291808304670de0b6b3a7640000036108c25761352b61353292670de0b6b3a7640000946142e7565b9093614165565b50505b801561392957613545908a6155a8565b9890801580613920575b61390d57908a85846135658f96958e5190613119565b9586670de0b6b3a7640000613590613581846134d68786614165565b61358a87614305565b90614165565b045f958715155f146138d2576135b5826135ba98996135ae84612fd6565b30916156e1565b61575d565b9b5f966064831161384d575b505050506065810290808204606514901517156108c25760649004891161383b5750613641670de0b6b3a76400006136378161362a8b9c8b604081019d8e805180613818575b505190811580159061380b575b6137ba575b50505061358a87614305565b049361358a8a5191614305565b0480885282613f05565b905f602061364e8b614b72565b6136578c613f12565b6001600160a01b039091169061366e823083614ba1565b90828783106137a5575b505050602460405180948193630ea598cb60e41b83528860048401525af15f9181613771575b506136b757895163381f209760e21b5f5260045260245ffd5b989394959697891561375d57916136d76136dc926136e195945190613f05565b613f05565b613119565b606481116136f2575b505050929190565b6064810290808204606414901517156108c25761271061371e9104809361371884612fd6565b90612f00565b6001600160a01b039061373090612fd6565b167fbb28353e4598c3b9199101a66e0989549b659a59a54d2c27fbb183f1932c8e6d5f80a35f80806136ea565b8451637fbbb20d60e11b5f5260045260245ffd5b9091506020813d60201161379d575b8161378d60209383611856565b8101031261063a5751905f61369e565b3d9150613780565b6121d16137b29389613f05565b5f8082613678565b7f108516ddcf5ba43cea6bb2cd5ff6d59ac196c1c86ccb9178332b9dd72d1ca561916040916020906001600160a01b03906137f490614cc9565b1694015182519182526020820152a28b5f8f61361e565b5060208301511515613619565b613835908d6001600160a01b0361382e86614cc9565b1690612f00565b5f61360c565b6319b811a560e31b5f5260045260245ffd5b5151939650919290916001600160a01b039091169061386b8c613f12565b93670de0b6b3a76400008402848104670de0b6b3a764000014851517156108c2576138a0926134d16134d6926134cb8b614305565b9161232883029280840461232814901517156108c2576138c89361271083159404928d615058565b925f8080806135c6565b50955050505050508034106138fa576135ba85838e868f956138f48134613f05565b9661575d565b8363ea9dce7560e01b5f5260045260245ffd5b83630180267160e01b5f5260045260245ffd5b5089511561354f565b82630180267160e01b5f5260045260245ffd5b915050613535565b84630180267160e01b5f5260045260245ffd5b8851633203e4dd60e21b5f5260045260245ffd5b9050155f613481565b1591505f613479565b83159250613472565b90506139ab91925060403d6040116139b3575b6139a38183611856565b8101906142ca565b91905f613467565b503d613999565b6139d6919c5060408093503d84116139b3576139a38183611856565b9b909161342d565b876383cf4a2b60e01b5f5260045260245ffd5b506001600160a01b038316156133d8565b34915061339c565b600114613a19575b5050613383565b5f5260cb60205260405f209060018060a01b03165f5260205260405f206001429101555f80613a12565b5051630180267160e01b5f5260045260245ffd5b9190613a61612982565b508015613ef157825190602084019160ff8351511690613a8086612fd6565b9160028103613eb857505f5260cb60205260405f209060018060a01b03165f5260205260405f206002429101555b613ac2613aba85612fd6565b8560cb61499b565b90613e42575b50613ad284612fd6565b90613adb612982565b506001600160a01b03613aed86614b72565b1615613e1a576060613b0160ff9287614a46565b015192515116916001600160a01b03613b1986614a2e565b169260028103613c8d5750604085015151602001516001600160a01b03168015613c7957613b4c826020943090846156e1565b613b57843083614ba1565b9084838310613c64575b505050608485515f6040519586948593635435d29560e11b8552600485015282602485015260448401523060648401525af1908115610add575f91613c30575b50613bb0613bb6915b846154e9565b836155a8565b919092613bcc613bc582613f12565b30906128ed565b6040840190613bdc868351613119565b11613c1c5790613bf3915180613bf8575b50613f12565b929190565b613c1690613c0583613f12565b6001600160a01b0361382e85614cc9565b5f613bed565b505163ea9dce7560e01b5f5260045260245ffd5b90506020813d602011613c5c575b81613c4b60209383611856565b8101031261063a5751613bb0613ba1565b3d9150613c3e565b6121d1613c719385613f05565b5f8084613b61565b85516383cf4a2b60e01b5f5260045260245ffd5b91925090600303613e2e5760408401516020015160c001516001600160a01b0316918215613e1a57604051635d043b2960e11b815260048101929092523060248301526001600160a01b0316604482015290602090829060649082905f905af1908115610add575f91613de8575b508015613dd45760206001600160a01b03613d1585614a2e565b60408601515151911692906001600160a01b0316613d34843083614ba1565b9084838310613dbf575b505050608485515f6040519586948593635435d29560e11b8552600485015260248401528160448401523060648401525af1908115610add575f91613d8b575b50613bb0613bb691613baa565b90506020813d602011613db7575b81613da660209383611856565b8101031261063a5751613bb0613d7e565b3d9150613d99565b6121d1613dcc9385613f05565b5f8084613d3e565b825163ea9dce7560e01b5f5260045260245ffd5b90506020813d602011613e12575b81613e0360209383611856565b8101031261063a57515f613cfb565b3d9150613df6565b84516383cf4a2b60e01b5f5260045260245ffd5b835163a78a5d1160e01b5f5260045260245ffd5b62ffffff80613e528551516154cf565b1691160162ffffff81116108c25782515190612710620fffff821611613ea05762ffffff60201b9060201b169062ffffff60201b191617613e9281615472565b15612c9a578251525f613ac8565b62ffffff90634d72eb2f60e11b5f521660045260245ffd5b600314613ec7575b5050613aae565b5f5260cb60205260405f209060018060a01b03165f5260205260405f206003429101555f80613ec0565b8251630180267160e01b5f5260045260245ffd5b919082039182116108c257565b6040015151606001516001600160a01b031690565b9180511580159061405b575b613f96575b519182613f4457505050565b8051907fad7560be13f82820274d285adf47dbc099a4c411d68af52bfd62ef13f0a2ba23906020906001600160a01b0390613f7e90614cc9565b6040519687526001600160a01b0390951695941693a4565b6040830151608001516001600160a01b03166020820190815180151580614049575b613fc5575b505050613f38565b85515f5260ca60205260405f2060018060a01b0383165f5260205260405f2060018060a01b0386165f5260205261400160405f20918254613119565b90557f785f14a7745852cfc4c745b3453427cb39e6d606d6431e8d8f5dcebe459d6e286040865192855194518251958652602086015260018060a01b031693a35f8080613fbd565b506001600160a01b0382161515613fb8565b5060408101511515613f33565b91909160408184031261063a5761407e81613126565b9260208201516001600160401b03811161063a576119bd920161313b565b604083810151015160201c6002161561333157826140e29161322d6020830194612f46602087510151604051958693632c7df23760e01b60208601523360248601613016565b8051801561415f5760408110908115614153575b50613309578060208061410e93518301019101614068565b92906001600160e01b03191663d3820dc960e01b016141405760400151608001516001600160a01b03166132ae575050565b51632014ceb360e21b5f5260045260245ffd5b9050610440105f6140f6565b50505050565b818102929181159184041417156108c257565b80511561085b5760200190565b80516001101561085b5760400190565b805182101561085b5760209160051b010190565b909160046141c38260408061ffff920151015160201c1690565b16156133315760208101916142108284519261322d6141e7602086015195516154cf565b97612f46628000008a161596604051958693637d7f232760e01b60208601523360248601613016565b805180156133295760801180156142be575b61330957806020806142399351830101910161317d565b9095916001600160e01b031916638280dcd960e01b016132f55761425c90614823565b929091826142af575b8261428f57826132c55750506132b5575060400151608001516001600160a01b03166132ae575050565b915062ffffff6142a08651516154cf565b1662ffffff8416141591613287565b62ffffff841615159250614265565b50805161044010614222565b919082604091031261063a5760206142e18361192b565b92015190565b81156142f1570490565b634e487b7160e01b5f52601260045260245ffd5b604d81116108c257600a0a90565b90801561462d5761432382614b72565b9061432d83613f12565b60e08401515190926001600160a01b0391821691168114614619576001600160a01b03831681811461460a5781159162ffffff60a0870151166103e881116145ed57604087810151602001515181516341976e0960e01b815260048101959095526001600160a01b0316959084602481895afa958615610add575f945f976145c3575b5060406024918151928380926341976e0960e01b82528860048301525afa948515610add575f915f966145a0575b5015908115614597575b50801561458f575b8015614587575b6145735760ff6040808a0151015160101c169660ff61441f8a602060e060ff92015101511690565b1697670de0b6b3a76400008302838104670de0b6b3a7640000036108c257604d82116108c2576134d6896134d1899361445b95600a0a906142e7565b92612710039161271083116108c25784670de0b6b3a76400006144926127106144888b9761449a99614165565b0461358a8d614305565b04928b615058565b92831561454f57156145345750670de0b6b3a7640000820291808304670de0b6b3a7640000036108c2576144df5f95946134d68796946134d188976134cb8998614305565b965af16144ea61232e565b50156144f95760808291015290565b60405162461bcd60e51b8152602060048201526013602482015272115512081d1c985b9cd9995c8819985a5b1959606a1b6044820152606490fd5b608094508296908396945061454a939250612f00565b015290565b838888516348cd2abf60e11b5f5260045260018060a01b031660245260445260645ffd5b8751633203e4dd60e21b5f5260045260245ffd5b5083156143f7565b5085156143f0565b9050155f6143e8565b9095506145bc915060403d6040116139b3576139a38183611856565b945f6143de565b6024919750604095506145e290863d88116139b3576139a38183611856565b9590959791506143b0565b865163751419d760e11b5f526004526103e860245260445260645ffd5b5050926119bd92508391612f00565b8351633500335960e21b5f5260045260245ffd5b5051632976bf1560e21b5f5260045260245ffd5b604083810151015160201c6008161561333157826146879161322d6020830194612f46602087510151604051958693634bdd816560e01b60208601523360248601613016565b8051801561415f57604081109081156146e5575b5061330957806020806146b393518301019101614068565b92906001600160e01b03191663b4227e9b60e01b016141405760400151608001516001600160a01b03166132ae575050565b9050610440105f61469b565b909160ff168061470857505060981c62ffffff1690565b6001810361471d57505060681c62ffffff1690565b6002810361473257505060801c62ffffff1690565b600303614745575060501c62ffffff1690565b63a78a5d1160e01b5f5260045260245ffd5b909160ff168061476d57505060b01c61ffff1690565b6001810361478157505060c01c61ffff1690565b6002810361479557505060d01c61ffff1690565b600303614745575060e01c61ffff1690565b91905f809160208151910182865af1913d9260405193601f19603f82011685016040528085525f602086013e15806147dd575050565b6147e45750565b6040805163319d54c360e01b81526001600160a01b03909216600483015260248201523d60448201819052805f606484013e601f801991011660640190fd5b9062100000620fffff8316921661483b578115159190565b9061484a576001906210000090565b5f905f90565b60408101906060614865604084510151615442565b925101515f604080516148778161183b565b82815282602082015201526040519261488f8461183b565b61ffff8260201c16845261ffff60406020860195828560101c16875201921682526020810151158061498d575b61497e57511580614970575b6149615762ffffff620fffff85169351168310159081614950575b5080614944575b1561492c57602001916113888351519211613ea05762ffffff60381b1990911660389190911b62ffffff60381b16179061492382615472565b15612c9a575152565b62ffffff8363071015a560e21b5f521660045260245ffd5b506113888211156148ea565b62ffffff915051168211155f6148e3565b633989c69960e21b5f5260045ffd5b5062800000841615156148c8565b6325ecfb7560e11b5f5260045ffd5b5062200000851615156148bc565b81515f908152602091825260408082206001600160a01b039095168252938252929092209101515160ff1660028114818115614a23575b506149e0575b50505f905f90565b600203614a1a57545b8015159081614a07575b506149ff575f806149d8565b600190601990565b614a12915042613f05565b505f5f6149f3565b600101546149e9565b60039150145f6149d2565b6040015160200151606001516001600160a01b031690565b614a4e61269a565b916001600160a01b03614a6083614a2e565b1691604080820151015160018060601b0390603b1c16906101008101518560ff60208401858282515116941060408401525151168015908115614b67575b5015614b595750600103614b445790604482614abc876040956158b2565b5183519586938492637ac6f8ab60e01b8452600484015260248301525afa8015610add575f925f91614b0b575b50602084015281835281811115614b0457505b606082015290565b9050614afc565b9250506040823d604011614b3c575b81614b2760409383611856565b8101031261063a57602082519201515f614ae9565b3d9150614b1a565b50505f19808452602084015260608301525090565b925050614afc929350615857565b60019150145f614a9e565b6040908101515101516001600160a01b031690565b6001600160a01b0391821681529116602082015260400190565b6001600160a01b03169182614bb7575050505f90565b604051636eb1769f60e11b815292602092849283918291614bdb9160048401614b87565b03915afa908115610add575f9161293a575090565b6001600160a01b03168015614c9957604051636eb1769f60e11b81529260208480614c1f863060048401614b87565b0381855afa938415610add575f94614c63575b50614c43612f5491612f5995613119565b60405163095ea7b360e01b6020820152938491612f469160248401612fbb565b93506020843d602011614c91575b81614c7e60209383611856565b8101031261063a57925192614c43614c32565b3d9150614c71565b63220abce160e01b5f5260045ffd5b90815260208101919091526001600160a01b03909116604082015260600190565b6040015160200151608001516001600160a01b031690565b90815181101561085b570160200190565b614cfc602a611a7c565b90614d0a6040519283611856565b602a8252614d18602a611a7c565b6020830190601f190136823782511561085b576030905381516001101561085b576078602183015360295b60018111614d975750614d535790565b606460405162461bcd60e51b815260206004820152602060248201527f537472696e67733a20686578206c656e67746820696e73756666696369656e746044820152fd5b90600f8116601081101561085b576f181899199a1a9b1b9c1cb0b131b232b360811b901a614dc58385614ce1565b5360041c9080156108c2575f1901614d43565b614de26042611a7c565b90614df06040519283611856565b60428252614dfe6042611a7c565b6020830190601f190136823782511561085b576030905381516001101561085b576078602183015360415b60018111614e395750614d535790565b90600f8116601081101561085b576f181899199a1a9b1b9c1cb0b131b232b360811b901a614e678385614ce1565b5360041c9080156108c2575f1901614e29565b90614eda9160018060a01b03165f8060405193614e98604086611856565b602085527f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564602086015260208151910182855af1614ed461232e565b916159e3565b8051908115918215614f48575b505015614ef057565b60405162461bcd60e51b815260206004820152602a60248201527f5361666545524332303a204552433230206f7065726174696f6e20646964206e6044820152691bdd081cdd58d8d9595960b21b6064820152608490fd5b819250906020918101031261063a576020614f63910161192b565b5f80614ee7565b60208183031261063a578051906001600160401b03821161063a57019080601f8301121561063a578151614f9d816126fa565b92614fab6040519485611856565b81845260208085019260051b82010192831161063a57602001905b828210614fd35750505090565b8151815260209182019101614fc6565b90602080835192838152019201905f5b8181106150005750505090565b82516001600160a01b0316845260209384019390920191600101614ff3565b919260809361504492979695978452602084015260a0604084015260a0830190614fe3565b6001600160a01b0390951660608201520152565b94929390604086019060018060a01b03604060208451015101511690811561542e5760054201928342116108c2575151608001516001600160a01b038281169791169188918861524057505050604051906150b4606083611856565b6002825260403660208401376150c982614178565b526001600160a01b038516806150de83614185565b526150e889613f12565b6001600160a01b0316036151e2575f916151279188604051809681958294637ff36ab560e01b84528a6004850152608060248501526084840190614fe3565b90306044840152606483015203925af1908115610add575f916151c0575b5080515f1981019081116108c25761515c91614195565b51945b8186106151a457505060407f25f1d03755df23c30e25db2dbd3891e31ce084bdfbfc46f9fe5e446ee5f9b2d491815194855285602086015260018060a01b031693a390565b859250516323dc081560e01b5f5260045260245260445260645ffd5b6151dc91503d805f833e6151d48183611856565b810190614f6a565b5f615145565b60405162461bcd60e51b815260206004820152603060248201527f4f6e6c79206e617469766520746f207969656c6442656172696e67546f6b656e60448201526f081cddd85c1cc81cdd5c1c1bdc9d195960821b6064820152608490fd5b9394939192911561531c57915f6152bd959281959460405191615264606084611856565b6002835260403660208501378b61527a84614178565b5261528483614185565b52615290853083614ba1565b9085858310615307575b505050604051968795869485936318cbafe560e01b85528b30926004870161501f565b03925af1908115610add575f916152ed575b5080515f1981019081116108c2576152e691614195565b519461515f565b61530191503d805f833e6151d48183611856565b5f6152cf565b6121d16153149387613f05565b5f808561529a565b90808814801561541c575b156153c05750925f9291836152bd9560405190615345606083611856565b600282526040366020840137818b61535c82614178565b526001600160a01b038b169061537190614185565b525b61537e853083614ba1565b90858583106153ab575b505050604051968795869485936338ed173960e01b85528b30926004870161501f565b6121d16153b89387613f05565b5f8085615388565b92915092604051936153d3608086611856565b6003855260603660208701378493886153eb87614178565b526153f586614185565b5284516002101561085b575f80948a9260606152bd980160018060a01b038b169052615373565b506001600160a01b0387168114615327565b8751636438763b60e11b5f5260045260245ffd5b61544a611877565b5060016040519161545a836117e9565b818160311c161515835260301c161515602082015290565b600360ff8216111590816154b9575b816154a3575b81615490575090565b6113889150620fffff9060381c16111590565b9050612710620fffff8260201c16111590615487565b9050612710620fffff8260081c16111590615481565b60201c612710620fffff62ffffff8316921611612cc15790565b9060016040808401510151603a1c16156155a3575f9060209060246001600160a01b0361551586614b72565b16916040519485938492636f074d1f60e11b845260048401525af15f918161556f575b5061555157505163381f209760e21b5f5260045260245ffd5b90811561555c575090565b51632976bf1560e21b5f5260045260245ffd5b9091506020813d60201161559b575b8161558b60209383611856565b8101031261063a5751905f615538565b3d915061557e565b905090565b919060206155b4612982565b9301906155c28251516154cf565b9082515160381c620fffff8116906113888211613ea05762100000166156b65762ffffff906220000084161515936280000081161582151590816156a0575b501561569457505b16926156158483615935565b921561567a57515160081c612710620fffff62ffffff8316921611612cc15761566692919080156156705761564d61565c9184615935565b80604089015283885283613f05565b6020870152613f05565b9060608401529190565b5061565c5f61564d565b5090615666918186528160408701525f6020870152613f05565b620fffff915016615609565b801591506156af575b5f615601565b50846156a9565b5093505050604051916156c883611804565b5f83525f60208401525f60408401525f60608401529190565b9192916001600160a01b0316908161570257631954782960e01b5f5260045ffd5b6001600160a01b03169133830361574e57612f5993604051936323b872dd60e01b6020860152602485015260018060a01b03166044840152606483015260648252612f54608483611856565b63f8fd233360e01b5f5260045ffd5b939091929460018060a01b0360e086015151169061577a86613f12565b926001600160a01b038416831461582a5762ffffff60a088015116976103e8891161580c57670de0b6b3a7640000860290868204670de0b6b3a764000014871517156108c2576134d16134d6926134cb6157d396614305565b956127100361271081116108c2576157f1612710916119bd98614165565b0493808511615804575b50801594615058565b93505f6157fb565b88885163751419d760e11b5f526004526103e860245260445260645ffd5b505050505091505090565b90815260ff918216602082015291166040820152901515606082015260800190565b90604001908151156158ae57600260ff6020830151511614615877575050565b6040818101510151905160329190911c60ff1691612ebd90839051151590604051948594633583df0360e11b865260048601615835565b5050565b90604001908151156158ae576020810190600160ff83515116146158d557505050565b604080820151015160321c60ff169060ff821660048114159081615929575b506158ff5750505050565b9060ff612ebd925193515116935115159060405194859463aaf4f52d60e01b865260048601615835565b6005915014155f6158f4565b808202905f1983820990828083109203918083039283612710111561599e5714615993577fbc01a36e2eb1c432ca57a786c226809d495182a9930be0ded288ce703afb7e9193612710910990828211900360fc1b910360041c170290565b505061271091500490565b60405162461bcd60e51b815260206004820152601f60248201527f46756c6c4d6174683a2064656e6f6d696e61746f7220746f6f20736d616c6c006044820152606490fd5b91929015615a4557508151156159f7575090565b3b15615a005790565b60405162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e74726163740000006044820152606490fd5b825190915015615a585750805190602001fd5b60405162461bcd60e51b815260206004820152908190612ebd906024830190611aee56feffdfc1249c027f9191656349feb0761381bb32c9f557e01f419fd08754bf5a1ba26469706673582212201368bbc7248a31d69ccf7a6365060829cc4638324f5117b789e2268aec5be16564736f6c634300081c0033
Deployed Bytecode
0x6080806040526004361015610027575b503615610025576339218f3b60e01b5f5260045ffd5b005b5f905f3560e01c90816301ffc9a71461168e575080630587bcf3146116525780630795a7ba146115b657806309ae48411461143a578063248a9ca3146114075780632f2ff15d1461134e57806336568abe146112f4578063386d81fa1461128c578063487523f41461123c578063795c1af9146111ec5780638980f11f1461114b5780638d390d311461111457806391d14854146110cb578063942a13ef14611096578063a217fddf1461107c578063a935508714611035578063b06576e814611005578063bb5e558c14610fd1578063c4d66de814610ded578063cccc6f4c14610dc9578063d1d67e1814610bb5578063d547741f14610afb578063e3dab550146102a3578063eb9915a714610280578063f0e8cb1614610219578063f9de6313146101bd578063fa7f0c4e1461018e5763fc6f94680361000f573461018b578060031936011261018b5760ce546040516001600160a01b039091168152602090f35b80fd5b503461018b5760206101b06101a2366116fb565b906101ab61261d565b61235d565b6001609755604051908152f35b503461018b57604036600319011261018b576004356101da61177a565b6101e2612de2565b811561020a5781835260cd6020526001196040842054161790825260cd602052604082205580f35b633960608d60e11b8352600483fd5b503461018b57604036600319011261018b57604061023561173a565b91600435815260cb602052209060018060a01b03165f52602052608060405f208054906001810154906003600282015491015491604051938452602084015260408301526060820152f35b503461018b578060031936011261018b576020600160cc54166040519015158152f35b503461063a57604036600319011261063a576102bd611750565b90602435916102ca61261d565b6102d2612de2565b6001600160a01b03168015610ae857825f5260c960205260405f2060405190635ab3ea5d60e11b82528460048301525f82602481865afa918215610add575f926108e9575b508151516001600160a01b0316156106fa576060820180515180156108d657600d83019182545f845580610883575b505f5b8281106107c9575050835180519092506001600160a01b031615905080156107b4575b801561079f575b801561078a575b61066357805160208201516001600160a01b03918216911681811492918315610773575b831561075c575b508215610745575b821561072e575b50811561070d575b506106fa5760208201805180519193916001600160a01b0316159081156106e4575b81156106ce575b81156106b8575b81156106a2575b811561068c575b8115610676575b50610663578051805183546001600160a01b03199081166001600160a01b0392831617855560208084015160018781018054851692861692909217909155604080860151600289018054861691871691909117905560608087015160038a018054871691881691909117905560809687015160048a01805487169188169190911790559851805160058a018054871691881691909117905592830151600689018054861691871691909117905582810151600789018054861691871691909117905597820151600888018054851691861691909117905581850151600988018054851691861691909117905560a082810151600a89018054861691871691909117905560c090920151600b88018054851691861691909117905596850151600c870155840151600f8601559190920151600e840180549092169216919091179055436010820155601101805460ff19169091179055827f67795b7df97de6699e0f17226d6111be2ddc0ebe76fb110c2e8ab6df9312579f5f80a2825f5260c960205260ff601160405f2001541615610650575f83815260c960205260409020600801546001600160a01b031692831561063e57833b1561063a5760405191630e3dab5560e41b8352600483015260248201525f8160448183875af19081610625575b5061061d573d8091604051936308c99ae160e21b8552600485015260406024850152816044850152606484013e601f801991011660640190fd5b600160975580f35b6106329192505f90611856565b5f905f6105e3565b5f80fd5b6308c99ae160e21b5f5260045260245ffd5b826337e1342d60e21b5f5260045260245ffd5b85635dbad7a160e01b5f5260045260245ffd5b60c001516001600160a01b03161590505f610401565b60a08101516001600160a01b03161591506103fa565b60808101516001600160a01b03161591506103f3565b60608101516001600160a01b03161591506103ec565b60408101516001600160a01b03161591506103e5565b60208101516001600160a01b03161591506103de565b84635dbad7a160e01b5f5260045260245ffd5b60408101516060909101516001600160a01b0390811691161490505f6103bc565b60608201516001600160a01b03161491505f6103b4565b60408201516001600160a01b0316811492506103ad565b60608301516001600160a01b03161492505f6103a5565b60408301516001600160a01b03168114935061039e565b5060608101516001600160a01b03161561037a565b5060408101516001600160a01b031615610373565b5060208101516001600160a01b03161561036c565b6107d4818351614195565b5184549190600160401b83101561086f576001830180875583101561085b575f86815260209081902082516003909502018054918301516001600160a81b03199092166001600160a01b03959095169490941760a09190911b60ff60a01b161783556040810151600184810191909155606091909101516002939093019290925501610349565b634e487b7160e01b5f52603260045260245ffd5b634e487b7160e01b5f52604160045260245ffd5b806003029060038204036108c257835f5260205f20908101905b8181106108aa5750610346565b805f600392555f60018201555f60028201550161089d565b634e487b7160e01b5f52601160045260245ffd5b8663f4ac45b160e01b5f5260045260245ffd5b9091503d805f833e6108fb8183611856565b81019060208183031261063a578051906001600160401b03821161063a570180820391610200831261063a576040519261093484611798565b60a0811261063a5760e09060405161094b816117b3565b61095485611938565b815261096260208601611938565b602082015261097360408601611938565b604082015261098460608601611938565b606082015261099560808601611938565b60808201528552609f19011261063a576040516109b1816117ce565b6109bd60a08401611938565b81526109cb60c08401611938565b60208201526109dc60e08401611938565b60408201526109ee6101008401611938565b6060820152610a006101208401611938565b6080820152610a126101408401611938565b60a0820152610a246101608401611938565b60c0820152602084015261018082015160408401526101a08201516001600160401b03811161063a57820181601f8201121561063a578051610a65816126fa565b92610a736040519485611856565b81845260208085019260071b8401019281841161063a57602001915b838310610ac3575050505060608301526101e090610ab06101c08201611938565b6080840152015160a0820152905f610317565b6020608091610ad2848661194c565b815201920191610a8f565b6040513d5f823e3d90fd5b826324df413360e21b5f5260045260245ffd5b3461063a57604036600319011261063a57600435610b1761173a565b90610b36610b31825f526065602052600160405f20015490565b612ec1565b805f52606560205260405f2060018060a01b0383165f5260205260ff60405f205416610b5e57005b5f8181526065602090815260408083206001600160a01b0395909516808452949091528120805460ff19169055339291907ff6391f5c32d9c69d2a47ea670b442974b53935d1edc7fd64eb21e047a839171b9080a4005b3461063a57602036600319011261063a576004355f60a0604051610bd881611798565b610be061269a565b8152610bea6126c4565b60208201528260408201526060808201528260808201520152805f5260c960205260405f209060ff60118301541615610db75750604051610c2a81611798565b604051610c36816117b3565b82546001600160a01b039081168252600184015481166020830152600284015481166040808401919091526003850154821660608401526004850154909116608083015290825251610c87816117ce565b60058301546001600160a01b0390811682526006840154811660208084019190915260078501548216604080850191909152600886015483166060850152600986015483166080850152600a860154831660a0850152600b86015490921660c0840152830191909152600c83015490820152600d82018054610d08816126fa565b91610d166040519384611856565b81835260208301905f5260205f205f915b838310610d6d5760608601859052600e8701546001600160a01b03166080870152600f87015460a08701819052604051603b9190911c6001600160601b03168152602090f35b60036020600192604051610d8081611804565b60ff8654868060a01b038116835260a01c168382015284860154604082015260028601546060820152815201920192019190610d27565b6337e1342d60e21b5f5260045260245ffd5b3461063a575f36600319011261063a576020600160cc54811c166040519015158152f35b3461063a57602036600319011261063a57610e06611750565b5f549060ff8260081c161591828093610fc4575b8015610fad575b15610f515760ff1981166001175f5582610f40575b506001600160a01b03168015610f3157610e6860ff5f5460081c16610e5a81612f5b565b610e6381612f5b565b612f5b565b60016097555f8181525f516020615a7d5f395f51905f52602052604090205460ff1615610ee0575b60ce80546001600160a01b031916919091179055610eaa57005b61ff00195f54165f557f7f26b83ff96e1f2b6a682f133852f6798a09c465da95921460cefb3847402498602060405160018152a1005b5f8181525f516020615a7d5f395f51905f5260205260408120805460ff19166001179055339082907f2f8788117e7eff1d82e926ec794901d17c78024a50270940304540a733656f0d8180a4610e90565b63d92e233d60e01b5f5260045ffd5b61ffff1916610101175f5582610e36565b60405162461bcd60e51b815260206004820152602e60248201527f496e697469616c697a61626c653a20636f6e747261637420697320616c72656160448201526d191e481a5b9a5d1a585b1a5e995960921b6064820152608490fd5b50303b158015610e215750600160ff821614610e21565b50600160ff821610610e1a565b3461063a57602036600319011261063a57610fea611789565b610ff2612de2565b60021960cc54169060011b1760cc555f80f35b3461063a57602036600319011261063a5761101e611789565b611026612de2565b60011960cc54161760cc555f80f35b3461063a57602036600319011261063a57600435801561106d575f5260cd6020526020600160405f205460021c166040519015158152f35b633960608d60e11b5f5260045ffd5b3461063a575f36600319011261063a5760206040515f8152f35b3461063a57602036600319011261063a57600435801561106d575f5260cd6020526020600160405f2054166040519015158152f35b3461063a57604036600319011261063a576110e461173a565b6004355f52606560205260405f209060018060a01b03165f52602052602060ff60405f2054166040519015158152f35b3461063a57602036600319011261063a57600435801561106d575f5260cd6020526020600160405f2054811c166040519015158152f35b3461063a57604036600319011261063a57611164611750565b6024359061117061261d565b611178612de2565b6001600160a01b03168015610f315761119130826128ed565b82116111da5760ce546111af9083906001600160a01b031683612f00565b7f505b28e6941631badc363841ecbf8e1214b9379c643936458e87be718e1579995f80a36001609755005b63112fed8b60e31b5f5260045260245ffd5b3461063a57604036600319011261063a5760043561120861177a565b611210612de2565b811561106d57815f5260cd60205260041960405f205416915f5260cd60205260021b1760405f20555f80f35b3461063a57604036600319011261063a5760043561125861177a565b611260612de2565b811561106d57815f5260cd60205260021960405f205416915f5260cd60205260011b1760405f20555f80f35b3461063a57606036600319011261063a576112a561173a565b6044356001600160a01b038116919082900361063a576004355f90815260ca602090815260408083206001600160a01b0394851684528252808320949093168252928352819020549051908152f35b3461063a57604036600319011261063a5761130d61173a565b5060405162461bcd60e51b8152602060048201526018602482015277149bdb195cc818d85b89dd081899481c995b9bdd5b98d95960421b6044820152606490fd5b3461063a57604036600319011261063a5760043561136a61173a565b90611384610b31825f526065602052600160405f20015490565b805f52606560205260405f2060018060a01b0383165f5260205260ff60405f205416156113ad57005b5f8181526065602090815260408083206001600160a01b0395909516808452949091528120805460ff19166001179055339291907f2f8788117e7eff1d82e926ec794901d17c78024a50270940304540a733656f0d9080a4005b3461063a57602036600319011261063a5760206114326004355f526065602052600160405f20015490565b604051908152f35b3461063a57604036600319011261063a5760043561145661173a565b61145e61261d565b815f5260c960205260ff601160405f20015416156115a3575f82815260c960205260409020600e01546001600160a01b03169033829003611594576001600160a01b0316918215611582575f81815260ca602090815260408083206001600160a01b0386811685529083528184209087168452909152902054801561156f575f82815260ca602090815260408083206001600160a01b03878116855290835281842090881684529091528120558061151630866128ed565b106115585760208161154a7fd3093803bade32f00a1623cb5300e4caed82036accaa6aafadc9f55827a83c08938688612f00565b604051908152a46001609755005b8382630c29241b60e21b5f5260045260245260445ffd5b506393aea85b60e01b5f5260045260245ffd5b6383cf4a2b60e01b5f5260045260245ffd5b6339218f3b60e01b5f5260045ffd5b506337e1342d60e21b5f5260045260245ffd5b3461063a57602036600319011261063a576004356115d261261d565b6115da612de2565b47811161163f57807f7bcc4dd1d0cf712ca981d90a999e6ea7d3da3e11844d63b289bd1c15c2ec5e8d5f80a25f8080808460018060a01b0360ce54165af161162061232e565b501561162d576001609755005b631b1eb84f60e31b5f5260045260245ffd5b63112fed8b60e31b5f525f60045260245ffd5b6040611677611660366116fb565b90611669611877565b5061167261261d565b611c96565b600160975560208251918051835201516020820152f35b3461063a57602036600319011261063a576004359063ffffffff60e01b821680920361063a57602091632e2ecea560e01b81149081156116d0575b5015158152f35b637965db0b60e01b8114915081156116ea575b50836116c9565b6301ffc9a760e01b149050836116e3565b90600319820160c0811261063a5760a01361063a5760049160a435906001600160401b03821161063a5760a090829003600319011261063a5760040190565b602435906001600160a01b038216820361063a57565b600435906001600160a01b038216820361063a57565b35906001600160a01b038216820361063a57565b60243590811515820361063a57565b60043590811515820361063a57565b60c081019081106001600160401b0382111761086f57604052565b60a081019081106001600160401b0382111761086f57604052565b60e081019081106001600160401b0382111761086f57604052565b604081019081106001600160401b0382111761086f57604052565b608081019081106001600160401b0382111761086f57604052565b61010081019081106001600160401b0382111761086f57604052565b606081019081106001600160401b0382111761086f57604052565b90601f801991011681019081106001600160401b0382111761086f57604052565b60405190611884826117e9565b5f6020838281520152565b91908260a091031261063a5760405191602083018381106001600160401b0382111761086f576040528261190f6080604051936118cb856117b3565b6118d481611766565b85526118e260208201611766565b60208601526118f360408201611766565b604086015261190460608201611766565b606086015201611766565b608083015252565b356001600160a01b038116810361063a5790565b5190811515820361063a57565b51906001600160a01b038216820361063a57565b919082608091031261063a5760405161196481611804565b809261196f81611938565b8252602081015160ff8116810361063a5760609182916020850152604081015160408501520151910152565b919060a08382031261063a576119bd9060206119b68561192b565b940161194c565b90565b80516001600160a01b0316825260208082015160ff169083015260408082015190830152606090810151910152565b90602080835192838152019201905f5b818110611a0c5750505090565b9091926020608082611a2160019488516119c0565b0194019291016119ff565b3560ff8116810361063a5790565b3562ffffff8116810361063a5790565b903590601e198136030182121561063a57018035906001600160401b03821161063a5760200191813603831361063a57565b6001600160401b03811161086f57601f01601f191660200190565b929192611aa382611a7c565b91611ab16040519384611856565b82948184528183011161063a578281602093845f960137010152565b5f5b838110611ade5750505f910152565b8181015183820152602001611acf565b90602091611b0781518092818552858086019101611acd565b601f01601f1916010190565b80516001600160a01b03908116835260208083015182169084015260408083015182169084015260608083015182169084015260809182015116910152565b80516001600160a01b03908116835260208083015182169084015260408083015182169084015260608083015182169084015260808083015182169084015260a08083015182169084015260c09182015116910152565b9060cc825260208201526060604082015281516060820152610380610100611bf5602080860151610340608087015280516103a0870152015160406103c08601526103e0850190611aee565b936040810151611c0960a086018251611b13565b611c1c6020820151610140870190611b52565b60408101516102208601526060808201516102408701526080918201516001600160a01b0390811661026088015290830151610280870152908201516102a086015260a082015162ffffff166102c086015260c0820151166102e085015260e0810151611c8e906103008601906119c0565b015191015290565b611ca4611ca991369061188f565b612673565b611cb281612711565b6060808201515f9473f703cab4acf850c3b3e08dd9aa585d120016d297939092602084013590840160a0611d146001600160a01b03611cf084611917565b1695604051968792839263196896eb60e11b845285600485015260a48401906119ef565b908660248401526044830152600160648301523460848301520381895af4938415610add575f905f956122fa575b50156122eb575f198214806122d1575b6122ac575b5080611d7b5750505050505060405190611d70826117e9565b808252602082015290565b611e0793611dfd9260405192611d90846117e9565b83525f6020840152611da182611a2c565b9462ffffff611dc0611db560408601611a3a565b946080810190611a4a565b94909560ff60405199611dd28b61181f565b168952602089015216604087015233606087015260808601528760a086015260c08501523691611a97565b60e0820152612a3c565b90803b1561063a575f604051809263469c699160e11b82528180611e2f878960048401611ba9565b03915af48015610add57612297575b50611e47611877565b50611e5181612fd6565b6040820151608001516001600160a01b031692908315159081612280575b611e7d606085015185613336565b611e8b879993949299612fd6565b9860405199611e998b6117e9565b5f808c5260208c01526001600160a01b03611eb38a614a2e565b60408b01516020015160c001516001600160a01b0316939116919082158015612278575b61226457611ee5908b614a46565b60ff60208c01515116611ef78c614b72565b916060611f05863086614ba1565b91019285845180841061224e575b5050505080155f146120635750611f4c935090602091519283888d516040519788958694859363095a7a1b60e21b855260048501614ca8565b03925af1918215612058578592612024575b50811561200a57508692611fba7fac903ad8fe2e7a6d229683df0f465af874089964b0ea74e096f95d106cd333b79360609360208e01525b60208d0180519097508015612002575b60808c0152611fb48b613f12565b8b613f27565b8a5194516040805133815260208101979097528601526001600160a01b03908116951693a4611fea575b50505090565b338314611fe457611ffa9261409c565b5f8080611fe4565b508c51611fa6565b88516323dc081560e01b8652600452602452604452606483fd5b9091506020813d602011612050575b8161204060209383611856565b8101031261063a5751905f611f5e565b3d9150612033565b6040513d87823e3d90fd5b91929160010361223a5790602061209988949351928d51604051968780948193637ef0584160e11b835288309160048501614ca8565b03925af192831561222f5787936121fb575b5082156121df575060408a01515151869392916020916001600160a01b03166120d5853083614ba1565b90858583106121c4575b505050604460405180968193636e553f6560e01b835286600484015260018060a01b0316968760248401525af19283156121b957908995949392918793612179575b509282611fba927fac903ad8fe2e7a6d229683df0f465af874089964b0ea74e096f95d106cd333b796946060967f491ac6e8661dcfb6555d94e6fd4b55961faf240fb9fad5774ba6a93532f330e38b80a48d52611f96565b919395509391506020813d6020116121b1575b8161219960209383611856565b8101031261063a575188949193919290916060612121565b3d915061218c565b6040513d88823e3d90fd5b6121d16121d79387613f05565b91614bf0565b5f80856120df565b8a516323dc081560e01b88526004526024526044829052606486fd5b9092506020813d602011612227575b8161221760209383611856565b8101031261063a5751915f6120ab565b3d915061220a565b6040513d89823e3d90fd5b8a5163a78a5d1160e01b8852600452602487fd5b61225b936121d191613f05565b5f808581611f13565b8a516383cf4a2b60e01b8852600452602487fd5b508315611ed7565b338514611e6f576122928484876131c6565b611e6f565b6122a49193505f90611856565b5f915f611e3e565b6122cb91506001600160a01b03906122c390611917565b1633906128ed565b5f611d57565b506001600160a01b036122e382611917565b161515611d52565b63436cb04160e01b5f5260045ffd5b905061231f91945060a03d60a011612327575b6123178183611856565b81019061199b565b93905f611d42565b503d61230d565b3d15612358573d9061233f82611a7c565b9161234d6040519384611856565b82523d5f602084013e565b606090565b611ca461236e91939293369061188f565b9161237883612711565b9273f703cab4acf850c3b3e08dd9aa585d120016d2979160608501519060208101359560a06123d6600180831b036123b260608601611917565b1694604051958692839263196896eb60e11b845285600485015260a48401906119ef565b908b602484015260448301525f60648301525f60848301520381885af4928315610add575f905f946125f9575b50156122eb575f1987146125a7575b8661241e575050505050565b95611dfd916124999495969760405192612437846117e9565b83525f602084015261244882611a2c565b9462ffffff61245c611db560408601611a3a565b94909560ff6040519961246e8b61181f565b168952602089015216604087015233606087015260808601528660a086015260c08501523691611a97565b91803b1561063a575f604051809263469c699160e11b825281806124c1888860048401611ba9565b03915af48015610add57612597575b506124da82612fd6565b60408301516080015190926001600160a01b039091169182151580612580575b61251a6125208661250f606087015187613a57565b828996929396613f27565b86614313565b60405190966001600160a01b0390811692169084907fdd8b8dbb53fec7033579b7466dc8fd28b088f5b31bca605ca8fa24a570e366ca90806125638b3383612fbb565b0390a46125705750505090565b338314611fe457611ffa92614641565b3384146124fa576125928383866141a9565b6124fa565b5f6125a191611856565b5f6124d0565b9550600260ff6125b683611a2c565b16036125dc578551602001516125d69033906001600160a01b03166128ed565b95612412565b602086015160c001516125d69033906001600160a01b03166128ed565b905061261591935060a03d60a011612327576123178183611856565b92905f612403565b60026097541461262e576002609755565b60405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152606490fd5b60405161268560208201809351611b13565b60a0815261269460c082611856565b51902090565b604051906126a7826117b3565b5f6080838281528260208201528260408201528260608201520152565b604051906126d1826117ce565b5f60c0838281528260208201528260408201528260608201528260808201528260a08201520152565b6001600160401b03811161086f5760051b60200190565b905f60a060405161272181611798565b61272961269a565b81526127336126c4565b60208201528260408201526060808201528260808201520152815f5260c960205260405f209160ff60118401541615610db7575060405161277381611798565b60405161277f816117b3565b83546001600160a01b0390811682526001850154811660208301526002850154811660408084019190915260038601548216606084015260048601549091166080830152908252516127d0816117ce565b60058401546001600160a01b0390811682526006850154811660208084019190915260078601548216604080850191909152600887015483166060850152600987015483166080850152600a870154831660a0850152600b87015490921660c0840152830191909152600c84015490820152600d83018054909390612854816126fa565b906128626040519283611856565b80825260208201955f5260205f20955f905b8282106128a3575050506060830152600e8101546001600160a01b03166080830152600f015460a08201529150565b600360206001926040516128b681611804565b60ff8c54868060a01b038116835260a01c1683820152848c0154604082015260028c01546060820152815201980191019096612874565b6001600160a01b03168061290057503190565b6040516370a0823160e01b81526001600160a01b039092166004830152602090829060249082905afa908115610add575f9161293a575090565b90506020813d602011612961575b8161295560209383611856565b8101031261063a575190565b3d9150612948565b60405190612976826117e9565b60606020835f81520152565b6040519061298f82611804565b5f6060838281528260208201528260408201520152565b6040519061012082018281106001600160401b0382111761086f576040525f610100838281526129d4612969565b60208201526040516129e5816117b3565b6129ed61269a565b81526129f76126c4565b602082015283604082015283606082015283608082015260408201528260608201528260808201528260a08201528260c0820152612a33612982565b60e08201520152565b612a446129a6565b506060810180519092906001600160a01b031615612dcb57602082019283518051159081612dbe575b50612da7576040830162ffffff8151166103e88111612d875750612a8f6129a6565b60a0850180518252865151606083015295516020015160808083019190915292516001600160a01b031660c08201529184015160e08301528451905162ffffff16906103e88211612d6c575060a082015260c0830192835180519060208101519060a0810151604082015191608060018060a01b03910151169260405194612b16866117b3565b855260208501526040840152606083015260808201526040830152819360ff8251169080519160a08301805160018060601b0390603b1c169051936060602060018060a01b039201510151169060208a5160246040518095819363f3c4799760e01b835260048301525afa5f9281612d38575b50612ba2578951637055368360e11b5f5260045260245ffd5b81999293949596979899911115612d245761010087018190528290609b86901c6001600160601b03161115612d0d57612be2915160408551015190614757565b915b60a08151015192612bf58194615442565b908151151580612d01575b612cf3575b5060200151612ce6575b604090510151928360f01c612c22612969565b5060038311612cd3576127108111612cc15750612710620fffff841611612ca9579062ffff0062ffffff60201b92600160501b600160b01b0390604b1c169460e81c16179160201b16171791612c7783615472565b15612c9a5760e0602092015160405193612c90856117e9565b8452828401520152565b6338bfed7360e11b5f5260045ffd5b62ffffff83634d72eb2f60e11b5f521660045260245ffd5b634d72eb2f60e11b5f5260045260245ffd5b82631ac6d2b760e21b5f5260045260245ffd5b6220000090921791612c0f565b628000001793506020612c05565b50602082015115612c00565b612d1e9151604085510151906146f1565b91612be4565b505163ef6f777560e01b5f5260045260245ffd5b9092506020813d602011612d64575b81612d5460209383611856565b8101031261063a5751915f612b89565b3d9150612d47565b63751419d760e11b5f526004526024526103e860445260645ffd5b60a085015163751419d760e11b5f526004526024526103e860445260645ffd5b60a0830151630180267160e01b5f5260045260245ffd5b602091500151155f612a6d565b60a08201516383cf4a2b60e01b5f5260045260245ffd5b335f9081525f516020615a7d5f395f51905f52602052604090205460ff1615612e0757565b612ebd612e1333614cf2565b612e9d6011612e215f614dd8565b92603760405194859276020b1b1b2b9b9a1b7b73a3937b61d1030b1b1b7bab73a1604d1b6020850152612e5d8151809260208688019101611acd565b83017001034b99036b4b9b9b4b733903937b6329607d1b83820152612e8c825180936020604885019101611acd565b01010301601f198101835282611856565b60405162461bcd60e51b8152602060048201529182916024830190611aee565b0390fd5b5f81815260656020908152604080832033845290915290205460ff1615612ee55750565b612ebd90612e9d6011612e21612efa33614cf2565b93614dd8565b9091906001600160a01b031680612f2057630c5d6fad60e21b5f5260045ffd5b612f54612f5993612f4660405194859263a9059cbb60e01b602085015260248401612fbb565b03601f198101845283611856565b614e7a565b565b15612f6257565b60405162461bcd60e51b815260206004820152602b60248201527f496e697469616c697a61626c653a20636f6e7472616374206973206e6f74206960448201526a6e697469616c697a696e6760a81b6064820152608490fd5b6001600160a01b039091168152602081019190915260400190565b60c081019060018060a01b0382511690338203612ffc575050516001600160a01b031690565b516364e9666360e11b5f526004526024523360445260645ffd5b90916119bd949260018060a01b031682526020820152608060408201528251608082015261010061306b60208086015161034060a086015280516103c0860152015160406103e0850152610400840190611aee565b93604081015161307f60c085018251611b13565b6130926020820151610160860190611b52565b60408101516102408501526060808201516102608601526080918201516001600160a01b03908116610280870152908301516102a0860152908201516102c085015260a082015162ffffff166102e085015260c08201511661030084015260e0810151613104906103208501906119c0565b01516103a08201526060818403910152611aee565b919082018092116108c257565b51906001600160e01b03198216820361063a57565b81601f8201121561063a57805161315181611a7c565b9261315f6040519485611856565b8184526020828401011161063a576119bd9160208085019101611acd565b909160608284031261063a5761319282613126565b9260208301516001600160401b03811161063a576040916131b491850161313b565b92015162ffffff8116810361063a5790565b909160016131e08260408061ffff920151015160201c1690565b16156133315760208101916132328284519261322d613204602086015195516154cf565b97612f46628000008a16159660405195869363e3b7f38160e01b60208601523360248601613016565b6147a7565b8051801561332957608011801561331d575b613309578060208061325b9351830101910161317d565b9095916001600160e01b031916631c480c7f60e01b016132f55761327e90614823565b929091826132e6575b826132c5575b50506132b5575b5060400151608001516001600160a01b03166132ae575050565b5160200152565b6132bf9082614850565b5f613294565b81159250906132d7575b505f8061328d565b6220000091501615155f6132cf565b62ffffff841615159250613287565b8351632014ceb360e21b5f5260045260245ffd5b8251632014ceb360e21b5f5260045260245ffd5b50805161044010613244565b505050505050565b505050565b9061333f612982565b508015613a4357815160ff602084015151169061335b84612fd6565b9180613a0a57505f5260cb60205260405f209060018060a01b03165f5260205260405f204290555b60e082018051519093906001600160a01b031615613a02575b6133a4612982565b508251936133b184614b72565b906133bb85613f12565b8151516001600160a01b03908116959192919084161580156139f1575b6139de57604087015160200151518251516001600160a01b0390811699911661340089613f12565b99604051906341976e0960e01b82526004820152604081602481855afa9a8b15610add575f915f9c6139ba575b50604080516341976e0960e01b81526001600160a01b0390921660048301529092839060249082905afa918215610add575f905f93613986575b508b1591821561397d575b8215613974575b50811561396b575b506139575760e08901516020015160ff9081161692670de0b6b3a76400008102818104670de0b6b3a764000014821517156108c257826134d68d6134d16134db946134cb8a614305565b906142e7565b614165565b6142e7565b60606134e7828d614a46565b0151908115613944578181111561393c57670de0b6b3a7640000820291808304670de0b6b3a7640000036108c25761352b61353292670de0b6b3a7640000946142e7565b9093614165565b50505b801561392957613545908a6155a8565b9890801580613920575b61390d57908a85846135658f96958e5190613119565b9586670de0b6b3a7640000613590613581846134d68786614165565b61358a87614305565b90614165565b045f958715155f146138d2576135b5826135ba98996135ae84612fd6565b30916156e1565b61575d565b9b5f966064831161384d575b505050506065810290808204606514901517156108c25760649004891161383b5750613641670de0b6b3a76400006136378161362a8b9c8b604081019d8e805180613818575b505190811580159061380b575b6137ba575b50505061358a87614305565b049361358a8a5191614305565b0480885282613f05565b905f602061364e8b614b72565b6136578c613f12565b6001600160a01b039091169061366e823083614ba1565b90828783106137a5575b505050602460405180948193630ea598cb60e41b83528860048401525af15f9181613771575b506136b757895163381f209760e21b5f5260045260245ffd5b989394959697891561375d57916136d76136dc926136e195945190613f05565b613f05565b613119565b606481116136f2575b505050929190565b6064810290808204606414901517156108c25761271061371e9104809361371884612fd6565b90612f00565b6001600160a01b039061373090612fd6565b167fbb28353e4598c3b9199101a66e0989549b659a59a54d2c27fbb183f1932c8e6d5f80a35f80806136ea565b8451637fbbb20d60e11b5f5260045260245ffd5b9091506020813d60201161379d575b8161378d60209383611856565b8101031261063a5751905f61369e565b3d9150613780565b6121d16137b29389613f05565b5f8082613678565b7f108516ddcf5ba43cea6bb2cd5ff6d59ac196c1c86ccb9178332b9dd72d1ca561916040916020906001600160a01b03906137f490614cc9565b1694015182519182526020820152a28b5f8f61361e565b5060208301511515613619565b613835908d6001600160a01b0361382e86614cc9565b1690612f00565b5f61360c565b6319b811a560e31b5f5260045260245ffd5b5151939650919290916001600160a01b039091169061386b8c613f12565b93670de0b6b3a76400008402848104670de0b6b3a764000014851517156108c2576138a0926134d16134d6926134cb8b614305565b9161232883029280840461232814901517156108c2576138c89361271083159404928d615058565b925f8080806135c6565b50955050505050508034106138fa576135ba85838e868f956138f48134613f05565b9661575d565b8363ea9dce7560e01b5f5260045260245ffd5b83630180267160e01b5f5260045260245ffd5b5089511561354f565b82630180267160e01b5f5260045260245ffd5b915050613535565b84630180267160e01b5f5260045260245ffd5b8851633203e4dd60e21b5f5260045260245ffd5b9050155f613481565b1591505f613479565b83159250613472565b90506139ab91925060403d6040116139b3575b6139a38183611856565b8101906142ca565b91905f613467565b503d613999565b6139d6919c5060408093503d84116139b3576139a38183611856565b9b909161342d565b876383cf4a2b60e01b5f5260045260245ffd5b506001600160a01b038316156133d8565b34915061339c565b600114613a19575b5050613383565b5f5260cb60205260405f209060018060a01b03165f5260205260405f206001429101555f80613a12565b5051630180267160e01b5f5260045260245ffd5b9190613a61612982565b508015613ef157825190602084019160ff8351511690613a8086612fd6565b9160028103613eb857505f5260cb60205260405f209060018060a01b03165f5260205260405f206002429101555b613ac2613aba85612fd6565b8560cb61499b565b90613e42575b50613ad284612fd6565b90613adb612982565b506001600160a01b03613aed86614b72565b1615613e1a576060613b0160ff9287614a46565b015192515116916001600160a01b03613b1986614a2e565b169260028103613c8d5750604085015151602001516001600160a01b03168015613c7957613b4c826020943090846156e1565b613b57843083614ba1565b9084838310613c64575b505050608485515f6040519586948593635435d29560e11b8552600485015282602485015260448401523060648401525af1908115610add575f91613c30575b50613bb0613bb6915b846154e9565b836155a8565b919092613bcc613bc582613f12565b30906128ed565b6040840190613bdc868351613119565b11613c1c5790613bf3915180613bf8575b50613f12565b929190565b613c1690613c0583613f12565b6001600160a01b0361382e85614cc9565b5f613bed565b505163ea9dce7560e01b5f5260045260245ffd5b90506020813d602011613c5c575b81613c4b60209383611856565b8101031261063a5751613bb0613ba1565b3d9150613c3e565b6121d1613c719385613f05565b5f8084613b61565b85516383cf4a2b60e01b5f5260045260245ffd5b91925090600303613e2e5760408401516020015160c001516001600160a01b0316918215613e1a57604051635d043b2960e11b815260048101929092523060248301526001600160a01b0316604482015290602090829060649082905f905af1908115610add575f91613de8575b508015613dd45760206001600160a01b03613d1585614a2e565b60408601515151911692906001600160a01b0316613d34843083614ba1565b9084838310613dbf575b505050608485515f6040519586948593635435d29560e11b8552600485015260248401528160448401523060648401525af1908115610add575f91613d8b575b50613bb0613bb691613baa565b90506020813d602011613db7575b81613da660209383611856565b8101031261063a5751613bb0613d7e565b3d9150613d99565b6121d1613dcc9385613f05565b5f8084613d3e565b825163ea9dce7560e01b5f5260045260245ffd5b90506020813d602011613e12575b81613e0360209383611856565b8101031261063a57515f613cfb565b3d9150613df6565b84516383cf4a2b60e01b5f5260045260245ffd5b835163a78a5d1160e01b5f5260045260245ffd5b62ffffff80613e528551516154cf565b1691160162ffffff81116108c25782515190612710620fffff821611613ea05762ffffff60201b9060201b169062ffffff60201b191617613e9281615472565b15612c9a578251525f613ac8565b62ffffff90634d72eb2f60e11b5f521660045260245ffd5b600314613ec7575b5050613aae565b5f5260cb60205260405f209060018060a01b03165f5260205260405f206003429101555f80613ec0565b8251630180267160e01b5f5260045260245ffd5b919082039182116108c257565b6040015151606001516001600160a01b031690565b9180511580159061405b575b613f96575b519182613f4457505050565b8051907fad7560be13f82820274d285adf47dbc099a4c411d68af52bfd62ef13f0a2ba23906020906001600160a01b0390613f7e90614cc9565b6040519687526001600160a01b0390951695941693a4565b6040830151608001516001600160a01b03166020820190815180151580614049575b613fc5575b505050613f38565b85515f5260ca60205260405f2060018060a01b0383165f5260205260405f2060018060a01b0386165f5260205261400160405f20918254613119565b90557f785f14a7745852cfc4c745b3453427cb39e6d606d6431e8d8f5dcebe459d6e286040865192855194518251958652602086015260018060a01b031693a35f8080613fbd565b506001600160a01b0382161515613fb8565b5060408101511515613f33565b91909160408184031261063a5761407e81613126565b9260208201516001600160401b03811161063a576119bd920161313b565b604083810151015160201c6002161561333157826140e29161322d6020830194612f46602087510151604051958693632c7df23760e01b60208601523360248601613016565b8051801561415f5760408110908115614153575b50613309578060208061410e93518301019101614068565b92906001600160e01b03191663d3820dc960e01b016141405760400151608001516001600160a01b03166132ae575050565b51632014ceb360e21b5f5260045260245ffd5b9050610440105f6140f6565b50505050565b818102929181159184041417156108c257565b80511561085b5760200190565b80516001101561085b5760400190565b805182101561085b5760209160051b010190565b909160046141c38260408061ffff920151015160201c1690565b16156133315760208101916142108284519261322d6141e7602086015195516154cf565b97612f46628000008a161596604051958693637d7f232760e01b60208601523360248601613016565b805180156133295760801180156142be575b61330957806020806142399351830101910161317d565b9095916001600160e01b031916638280dcd960e01b016132f55761425c90614823565b929091826142af575b8261428f57826132c55750506132b5575060400151608001516001600160a01b03166132ae575050565b915062ffffff6142a08651516154cf565b1662ffffff8416141591613287565b62ffffff841615159250614265565b50805161044010614222565b919082604091031261063a5760206142e18361192b565b92015190565b81156142f1570490565b634e487b7160e01b5f52601260045260245ffd5b604d81116108c257600a0a90565b90801561462d5761432382614b72565b9061432d83613f12565b60e08401515190926001600160a01b0391821691168114614619576001600160a01b03831681811461460a5781159162ffffff60a0870151166103e881116145ed57604087810151602001515181516341976e0960e01b815260048101959095526001600160a01b0316959084602481895afa958615610add575f945f976145c3575b5060406024918151928380926341976e0960e01b82528860048301525afa948515610add575f915f966145a0575b5015908115614597575b50801561458f575b8015614587575b6145735760ff6040808a0151015160101c169660ff61441f8a602060e060ff92015101511690565b1697670de0b6b3a76400008302838104670de0b6b3a7640000036108c257604d82116108c2576134d6896134d1899361445b95600a0a906142e7565b92612710039161271083116108c25784670de0b6b3a76400006144926127106144888b9761449a99614165565b0461358a8d614305565b04928b615058565b92831561454f57156145345750670de0b6b3a7640000820291808304670de0b6b3a7640000036108c2576144df5f95946134d68796946134d188976134cb8998614305565b965af16144ea61232e565b50156144f95760808291015290565b60405162461bcd60e51b8152602060048201526013602482015272115512081d1c985b9cd9995c8819985a5b1959606a1b6044820152606490fd5b608094508296908396945061454a939250612f00565b015290565b838888516348cd2abf60e11b5f5260045260018060a01b031660245260445260645ffd5b8751633203e4dd60e21b5f5260045260245ffd5b5083156143f7565b5085156143f0565b9050155f6143e8565b9095506145bc915060403d6040116139b3576139a38183611856565b945f6143de565b6024919750604095506145e290863d88116139b3576139a38183611856565b9590959791506143b0565b865163751419d760e11b5f526004526103e860245260445260645ffd5b5050926119bd92508391612f00565b8351633500335960e21b5f5260045260245ffd5b5051632976bf1560e21b5f5260045260245ffd5b604083810151015160201c6008161561333157826146879161322d6020830194612f46602087510151604051958693634bdd816560e01b60208601523360248601613016565b8051801561415f57604081109081156146e5575b5061330957806020806146b393518301019101614068565b92906001600160e01b03191663b4227e9b60e01b016141405760400151608001516001600160a01b03166132ae575050565b9050610440105f61469b565b909160ff168061470857505060981c62ffffff1690565b6001810361471d57505060681c62ffffff1690565b6002810361473257505060801c62ffffff1690565b600303614745575060501c62ffffff1690565b63a78a5d1160e01b5f5260045260245ffd5b909160ff168061476d57505060b01c61ffff1690565b6001810361478157505060c01c61ffff1690565b6002810361479557505060d01c61ffff1690565b600303614745575060e01c61ffff1690565b91905f809160208151910182865af1913d9260405193601f19603f82011685016040528085525f602086013e15806147dd575050565b6147e45750565b6040805163319d54c360e01b81526001600160a01b03909216600483015260248201523d60448201819052805f606484013e601f801991011660640190fd5b9062100000620fffff8316921661483b578115159190565b9061484a576001906210000090565b5f905f90565b60408101906060614865604084510151615442565b925101515f604080516148778161183b565b82815282602082015201526040519261488f8461183b565b61ffff8260201c16845261ffff60406020860195828560101c16875201921682526020810151158061498d575b61497e57511580614970575b6149615762ffffff620fffff85169351168310159081614950575b5080614944575b1561492c57602001916113888351519211613ea05762ffffff60381b1990911660389190911b62ffffff60381b16179061492382615472565b15612c9a575152565b62ffffff8363071015a560e21b5f521660045260245ffd5b506113888211156148ea565b62ffffff915051168211155f6148e3565b633989c69960e21b5f5260045ffd5b5062800000841615156148c8565b6325ecfb7560e11b5f5260045ffd5b5062200000851615156148bc565b81515f908152602091825260408082206001600160a01b039095168252938252929092209101515160ff1660028114818115614a23575b506149e0575b50505f905f90565b600203614a1a57545b8015159081614a07575b506149ff575f806149d8565b600190601990565b614a12915042613f05565b505f5f6149f3565b600101546149e9565b60039150145f6149d2565b6040015160200151606001516001600160a01b031690565b614a4e61269a565b916001600160a01b03614a6083614a2e565b1691604080820151015160018060601b0390603b1c16906101008101518560ff60208401858282515116941060408401525151168015908115614b67575b5015614b595750600103614b445790604482614abc876040956158b2565b5183519586938492637ac6f8ab60e01b8452600484015260248301525afa8015610add575f925f91614b0b575b50602084015281835281811115614b0457505b606082015290565b9050614afc565b9250506040823d604011614b3c575b81614b2760409383611856565b8101031261063a57602082519201515f614ae9565b3d9150614b1a565b50505f19808452602084015260608301525090565b925050614afc929350615857565b60019150145f614a9e565b6040908101515101516001600160a01b031690565b6001600160a01b0391821681529116602082015260400190565b6001600160a01b03169182614bb7575050505f90565b604051636eb1769f60e11b815292602092849283918291614bdb9160048401614b87565b03915afa908115610add575f9161293a575090565b6001600160a01b03168015614c9957604051636eb1769f60e11b81529260208480614c1f863060048401614b87565b0381855afa938415610add575f94614c63575b50614c43612f5491612f5995613119565b60405163095ea7b360e01b6020820152938491612f469160248401612fbb565b93506020843d602011614c91575b81614c7e60209383611856565b8101031261063a57925192614c43614c32565b3d9150614c71565b63220abce160e01b5f5260045ffd5b90815260208101919091526001600160a01b03909116604082015260600190565b6040015160200151608001516001600160a01b031690565b90815181101561085b570160200190565b614cfc602a611a7c565b90614d0a6040519283611856565b602a8252614d18602a611a7c565b6020830190601f190136823782511561085b576030905381516001101561085b576078602183015360295b60018111614d975750614d535790565b606460405162461bcd60e51b815260206004820152602060248201527f537472696e67733a20686578206c656e67746820696e73756666696369656e746044820152fd5b90600f8116601081101561085b576f181899199a1a9b1b9c1cb0b131b232b360811b901a614dc58385614ce1565b5360041c9080156108c2575f1901614d43565b614de26042611a7c565b90614df06040519283611856565b60428252614dfe6042611a7c565b6020830190601f190136823782511561085b576030905381516001101561085b576078602183015360415b60018111614e395750614d535790565b90600f8116601081101561085b576f181899199a1a9b1b9c1cb0b131b232b360811b901a614e678385614ce1565b5360041c9080156108c2575f1901614e29565b90614eda9160018060a01b03165f8060405193614e98604086611856565b602085527f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564602086015260208151910182855af1614ed461232e565b916159e3565b8051908115918215614f48575b505015614ef057565b60405162461bcd60e51b815260206004820152602a60248201527f5361666545524332303a204552433230206f7065726174696f6e20646964206e6044820152691bdd081cdd58d8d9595960b21b6064820152608490fd5b819250906020918101031261063a576020614f63910161192b565b5f80614ee7565b60208183031261063a578051906001600160401b03821161063a57019080601f8301121561063a578151614f9d816126fa565b92614fab6040519485611856565b81845260208085019260051b82010192831161063a57602001905b828210614fd35750505090565b8151815260209182019101614fc6565b90602080835192838152019201905f5b8181106150005750505090565b82516001600160a01b0316845260209384019390920191600101614ff3565b919260809361504492979695978452602084015260a0604084015260a0830190614fe3565b6001600160a01b0390951660608201520152565b94929390604086019060018060a01b03604060208451015101511690811561542e5760054201928342116108c2575151608001516001600160a01b038281169791169188918861524057505050604051906150b4606083611856565b6002825260403660208401376150c982614178565b526001600160a01b038516806150de83614185565b526150e889613f12565b6001600160a01b0316036151e2575f916151279188604051809681958294637ff36ab560e01b84528a6004850152608060248501526084840190614fe3565b90306044840152606483015203925af1908115610add575f916151c0575b5080515f1981019081116108c25761515c91614195565b51945b8186106151a457505060407f25f1d03755df23c30e25db2dbd3891e31ce084bdfbfc46f9fe5e446ee5f9b2d491815194855285602086015260018060a01b031693a390565b859250516323dc081560e01b5f5260045260245260445260645ffd5b6151dc91503d805f833e6151d48183611856565b810190614f6a565b5f615145565b60405162461bcd60e51b815260206004820152603060248201527f4f6e6c79206e617469766520746f207969656c6442656172696e67546f6b656e60448201526f081cddd85c1cc81cdd5c1c1bdc9d195960821b6064820152608490fd5b9394939192911561531c57915f6152bd959281959460405191615264606084611856565b6002835260403660208501378b61527a84614178565b5261528483614185565b52615290853083614ba1565b9085858310615307575b505050604051968795869485936318cbafe560e01b85528b30926004870161501f565b03925af1908115610add575f916152ed575b5080515f1981019081116108c2576152e691614195565b519461515f565b61530191503d805f833e6151d48183611856565b5f6152cf565b6121d16153149387613f05565b5f808561529a565b90808814801561541c575b156153c05750925f9291836152bd9560405190615345606083611856565b600282526040366020840137818b61535c82614178565b526001600160a01b038b169061537190614185565b525b61537e853083614ba1565b90858583106153ab575b505050604051968795869485936338ed173960e01b85528b30926004870161501f565b6121d16153b89387613f05565b5f8085615388565b92915092604051936153d3608086611856565b6003855260603660208701378493886153eb87614178565b526153f586614185565b5284516002101561085b575f80948a9260606152bd980160018060a01b038b169052615373565b506001600160a01b0387168114615327565b8751636438763b60e11b5f5260045260245ffd5b61544a611877565b5060016040519161545a836117e9565b818160311c161515835260301c161515602082015290565b600360ff8216111590816154b9575b816154a3575b81615490575090565b6113889150620fffff9060381c16111590565b9050612710620fffff8260201c16111590615487565b9050612710620fffff8260081c16111590615481565b60201c612710620fffff62ffffff8316921611612cc15790565b9060016040808401510151603a1c16156155a3575f9060209060246001600160a01b0361551586614b72565b16916040519485938492636f074d1f60e11b845260048401525af15f918161556f575b5061555157505163381f209760e21b5f5260045260245ffd5b90811561555c575090565b51632976bf1560e21b5f5260045260245ffd5b9091506020813d60201161559b575b8161558b60209383611856565b8101031261063a5751905f615538565b3d915061557e565b905090565b919060206155b4612982565b9301906155c28251516154cf565b9082515160381c620fffff8116906113888211613ea05762100000166156b65762ffffff906220000084161515936280000081161582151590816156a0575b501561569457505b16926156158483615935565b921561567a57515160081c612710620fffff62ffffff8316921611612cc15761566692919080156156705761564d61565c9184615935565b80604089015283885283613f05565b6020870152613f05565b9060608401529190565b5061565c5f61564d565b5090615666918186528160408701525f6020870152613f05565b620fffff915016615609565b801591506156af575b5f615601565b50846156a9565b5093505050604051916156c883611804565b5f83525f60208401525f60408401525f60608401529190565b9192916001600160a01b0316908161570257631954782960e01b5f5260045ffd5b6001600160a01b03169133830361574e57612f5993604051936323b872dd60e01b6020860152602485015260018060a01b03166044840152606483015260648252612f54608483611856565b63f8fd233360e01b5f5260045ffd5b939091929460018060a01b0360e086015151169061577a86613f12565b926001600160a01b038416831461582a5762ffffff60a088015116976103e8891161580c57670de0b6b3a7640000860290868204670de0b6b3a764000014871517156108c2576134d16134d6926134cb6157d396614305565b956127100361271081116108c2576157f1612710916119bd98614165565b0493808511615804575b50801594615058565b93505f6157fb565b88885163751419d760e11b5f526004526103e860245260445260645ffd5b505050505091505090565b90815260ff918216602082015291166040820152901515606082015260800190565b90604001908151156158ae57600260ff6020830151511614615877575050565b6040818101510151905160329190911c60ff1691612ebd90839051151590604051948594633583df0360e11b865260048601615835565b5050565b90604001908151156158ae576020810190600160ff83515116146158d557505050565b604080820151015160321c60ff169060ff821660048114159081615929575b506158ff5750505050565b9060ff612ebd925193515116935115159060405194859463aaf4f52d60e01b865260048601615835565b6005915014155f6158f4565b808202905f1983820990828083109203918083039283612710111561599e5714615993577fbc01a36e2eb1c432ca57a786c226809d495182a9930be0ded288ce703afb7e9193612710910990828211900360fc1b910360041c170290565b505061271091500490565b60405162461bcd60e51b815260206004820152601f60248201527f46756c6c4d6174683a2064656e6f6d696e61746f7220746f6f20736d616c6c006044820152606490fd5b91929015615a4557508151156159f7575090565b3b15615a005790565b60405162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e74726163740000006044820152606490fd5b825190915015615a585750805190602001fd5b60405162461bcd60e51b815260206004820152908190612ebd906024830190611aee56feffdfc1249c027f9191656349feb0761381bb32c9f557e01f419fd08754bf5a1ba26469706673582212201368bbc7248a31d69ccf7a6365060829cc4638324f5117b789e2268aec5be16564736f6c634300081c0033
Loading...
Loading
Loading...
Loading
Loading...
Loading
Net Worth in USD
$0.00
Net Worth in S
Multichain Portfolio | 35 Chains
| Chain | Token | Portfolio % | Price | Amount | Value |
|---|
Loading...
Loading
Loading...
Loading
Loading...
Loading
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.