Overview
S Balance
0 S
S Value
$0.00More Info
Private Name Tags
ContractCreator
Loading...
Loading
Contract Name:
Core
Compiler Version
v0.8.18+commit.87f61d96
Optimization Enabled:
Yes with 1337 runs
Other Settings:
default evmVersion
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: UNKNOWN pragma solidity 0.8.18; // Contracts/Libraries/Modifiers import { LibCore } from "../../libraries/LibCore.sol"; import { LibUsd } from "../../libraries/LibUsd.sol"; import { LibTokens } from "../../libraries/LibTokens.sol"; import { LibDex } from "../../libraries/LibDex.sol"; import { LibLST } from "../../libraries/LibLST.sol"; import { Diamondable } from "../../Diamondable.sol"; // Facets import { FakePools } from "./LaunchStrategies/FakePools.sol"; import { Launcher } from "./Launcher.sol"; import { Degen } from "./Degen.sol"; // Interfaces import { Token } from "../../../Token.sol"; // Third Party import { DynamicBufferLib } from "solady/src/utils/DynamicBufferLib.sol"; contract Core is Diamondable { event TokenCreated( address token, address creator, LibTokens.LaunchStrategy strategy, LibDex.Dex dex, bytes data, bytes priceData ); event TokenLaunched( address token, address creator, LibTokens.LaunchStrategy strategy, LibDex.Dex dex, address pair ); function create( string calldata name, string calldata symbol, string calldata description, bytes calldata image, string[] calldata links, bytes calldata data, LibTokens.LaunchStrategy strategy, LibDex.Dex dex, uint256 initialBuy ) public payable { Core(address(this))._create(msg.sender, name, symbol, description, image, links, data, strategy, dex, initialBuy, msg.value); } function _create( address creator, string calldata name, string calldata symbol, string calldata description, bytes calldata image, string[] calldata links, bytes calldata data, LibTokens.LaunchStrategy strategy, LibDex.Dex dex, uint256 initialBuy, uint256 eth ) public payable onlyDiamond { require( bytes(name).length <= 18 && bytes(symbol).length <= 18 && bytes(image).length <= 16777217, // 1mb base64 "invalid name/symbol/image" ); require(links.length < 5, "4 links max"); for (uint8 i = 0; i < links.length; i++) { require(bytes(links[i]).length <= 128, "link too long"); } LibCore.Storage storage d = LibCore.store(); Token token = new Token(creator, name, symbol, description, image, links, d.tokenSupply, address(this)); address tokenAddress = address(token); DynamicBufferLib.DynamicBuffer memory buffer = DynamicBufferLib.DynamicBuffer(hex''); if (strategy == LibTokens.LaunchStrategy.FakeLiquidity) { (bytes memory _data) = FakePools(address(this)).fakepool_create(tokenAddress, d.tokenSupply, data); DynamicBufferLib.p(buffer, _data); } else { revert("invalid strategy"); } LibTokens.store().tokens[tokenAddress] = LibTokens.TokenInfo( creator, strategy, dex ); emit TokenCreated(tokenAddress, creator, strategy, dex, data, buffer.data); uint256 creationEth = LibUsd.usdToEth(d.creationPrice); require(eth >= creationEth, "usd price changed"); eth -= creationEth; LibCore.gatherProceeds(creationEth); if (initialBuy > 0) { _buy(creator, tokenAddress, initialBuy, 0); eth -= initialBuy; } if (eth > 0) payable(creator).transfer(eth); // refund dust } function quote(address token, uint256 amount, bool ethOut) public view returns(uint256) { LibTokens.LaunchStrategy strategy = LibTokens.store().tokens[token].strategy; if (strategy == LibTokens.LaunchStrategy.FakeLiquidity) { return FakePools(address(this)).fakepool_quote(token, amount, ethOut); } else { revert("invalid strategy"); } } event Bought(address buyer, address token, uint256 ethIn, uint256 tokensOut, bytes data); function buy(address token, uint256 min) public payable { Core(address(this))._buy(msg.sender, token, msg.value, min); } function _buy(address buyer, address token, uint256 ethIn, uint256 min) public onlyDiamond { LibTokens.LaunchStrategy strategy = LibTokens.store().tokens[token].strategy; (uint256 tokensOut, DynamicBufferLib.DynamicBuffer memory buffer) = (0, DynamicBufferLib.DynamicBuffer(hex'')); if (strategy == LibTokens.LaunchStrategy.FakeLiquidity) { (uint256 tokens, bytes memory data) = FakePools(address(this)).fakepool_buy{ value: ethIn }(token); tokensOut = tokens; DynamicBufferLib.p(buffer, data); } else { revert("invalid strategy"); } if (min != 0) require(tokensOut >= min, "amount out lower than min"); LibLST.addLiquidity(ethIn); Degen(address(this)).attributeXp(buyer, Degen.XpType.Buy, ethIn); Token(token).transfer(buyer, tokensOut); // Transfer tokens to buyer emit Bought(buyer, token, ethIn, tokensOut, buffer.data); } event Sold(address seller, address token, uint256 ethOut, uint256 tokensIn, bytes data); function sell(address token, uint256 amount, uint256 min) public { Core(address(this))._sell(msg.sender, token, amount, min); } function _sell(address seller, address token, uint256 amount, uint256 min) public onlyDiamond { LibTokens.LaunchStrategy strategy = LibTokens.store().tokens[token].strategy; (uint256 ethOut, DynamicBufferLib.DynamicBuffer memory buffer) = (0, DynamicBufferLib.DynamicBuffer(hex'')); if (strategy == LibTokens.LaunchStrategy.FakeLiquidity) { (uint256 eth, bytes memory data) = FakePools(address(this)).fakepool_sell(token, amount); ethOut = eth; DynamicBufferLib.p(buffer, data); } else { revert("invalid strategy"); } if (min != 0) require(ethOut >= min, "amount out lower than min"); LibLST.removeLiquidity(ethOut); Degen(address(this)).attributeXp(seller, Degen.XpType.Sell, ethOut); Token(token).transferFrom(seller, address(this), amount); // Transfer tokens from seller payable(seller).transfer(ethOut); // Transfer eth to seller emit Sold(seller, token, ethOut, amount, buffer.data); } function launch(address token) public onlyDiamond { LibTokens.TokenInfo storage tokenInfo = LibTokens.store().tokens[token]; require(tokenInfo.creator != address(0), "invalid token"); Token(token).unlock(); (address pair, uint256 eth,) = Launcher(address(this)).launch(token, tokenInfo); Degen(address(this)).attributeXp(tokenInfo.creator, Degen.XpType.Launch, eth); emit TokenLaunched(token, tokenInfo.creator, tokenInfo.strategy, tokenInfo.dex, pair); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/ERC20.sol) pragma solidity ^0.8.0; import "./IERC20.sol"; import "./extensions/IERC20Metadata.sol"; import "../../utils/Context.sol"; /** * @dev Implementation of the {IERC20} interface. * * This implementation is agnostic to the way tokens are created. This means * that a supply mechanism has to be added in a derived contract using {_mint}. * For a generic mechanism see {ERC20PresetMinterPauser}. * * TIP: For a detailed writeup see our guide * https://forum.openzeppelin.com/t/how-to-implement-erc20-supply-mechanisms/226[How * to implement supply mechanisms]. * * 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 ERC20 is Context, IERC20, IERC20Metadata { mapping(address => uint256) private _balances; mapping(address => mapping(address => uint256)) private _allowances; uint256 private _totalSupply; string private _name; string private _symbol; /** * @dev Sets the values for {name} and {symbol}. * * All two of these values are immutable: they can only be set once during * construction. */ constructor(string memory name_, string memory symbol_) { _name = name_; _symbol = symbol_; } /** * @dev Returns the name of the token. */ function name() public view virtual override returns (string memory) { return _name; } /** * @dev Returns the symbol of the token, usually a shorter version of the * name. */ function symbol() public view virtual override returns (string memory) { return _symbol; } /** * @dev Returns the number of decimals used to get its user representation. * For example, if `decimals` equals `2`, a balance of `505` tokens should * be displayed to a user as `5.05` (`505 / 10 ** 2`). * * Tokens usually opt for a value of 18, imitating the relationship between * Ether and Wei. This is the 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 {} }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (token/ERC20/extensions/IERC20Metadata.sol) pragma solidity ^0.8.0; import "../IERC20.sol"; /** * @dev Interface for the optional metadata functions from the ERC20 standard. * * _Available since v4.1._ */ interface IERC20Metadata is IERC20 { /** * @dev Returns the name of the token. */ function name() external view returns (string memory); /** * @dev Returns the symbol of the token. */ function symbol() external view returns (string memory); /** * @dev Returns the decimals places of the token. */ function decimals() external view returns (uint8); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/IERC20.sol) pragma solidity ^0.8.0; /** * @dev Interface of the ERC20 standard as defined in the EIP. */ interface IERC20 { /** * @dev Emitted when `value` tokens are moved from one account (`from`) to * another (`to`). * * Note that `value` may be zero. */ event Transfer(address indexed from, address indexed to, uint256 value); /** * @dev Emitted when the allowance of a `spender` for an `owner` is set by * a call to {approve}. `value` is the new allowance. */ event Approval(address indexed owner, address indexed spender, uint256 value); /** * @dev Returns the amount of tokens in existence. */ function totalSupply() external view returns (uint256); /** * @dev Returns the amount of tokens owned by `account`. */ function balanceOf(address account) external view returns (uint256); /** * @dev Moves `amount` tokens from the caller's account to `to`. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transfer(address to, uint256 amount) external returns (bool); /** * @dev Returns the remaining number of tokens that `spender` will be * allowed to spend on behalf of `owner` through {transferFrom}. This is * zero by default. * * This value changes when {approve} or {transferFrom} are called. */ function allowance(address owner, address spender) external view returns (uint256); /** * @dev Sets `amount` as the allowance of `spender` over the caller's tokens. * * Returns a boolean value indicating whether the operation succeeded. * * IMPORTANT: Beware that changing an allowance with this method brings the risk * that someone may use both the old and the new allowance by unfortunate * transaction ordering. One possible solution to mitigate this race * condition is to first reduce the spender's allowance to 0 and set the * desired value afterwards: * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729 * * Emits an {Approval} event. */ function approve(address spender, uint256 amount) external returns (bool); /** * @dev Moves `amount` tokens from `from` to `to` using the * allowance mechanism. `amount` is then deducted from the caller's * allowance. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transferFrom(address from, address to, uint256 amount) external returns (bool); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/Context.sol) pragma solidity ^0.8.0; /** * @dev Provides information about the current execution context, including the * sender of the transaction and its data. While these are generally available * via msg.sender and msg.data, they should not be accessed in such a direct * manner, since when dealing with meta-transactions the account sending and * paying for execution may not be the actual sender (as far as an application * is concerned). * * This contract is only required for intermediate, library-like contracts. */ abstract contract Context { function _msgSender() internal view virtual returns (address) { return msg.sender; } function _msgData() internal view virtual returns (bytes calldata) { return msg.data; } }
// SPDX-License-Identifier: MIT pragma solidity 0.8.18; import { LibDiamond } from "./libraries/LibDiamond.sol"; contract Diamondable { error Unauthorized(address account); modifier onlyDiamond() { LibDiamond.enforceDiamondItself(); _; } function diamond() internal view returns (address diamond_) { diamond_ = LibDiamond.diamondStorage().diamondAddress; } }
// SPDX-License-Identifier: UNKNOWN pragma solidity 0.8.18; // Contracts/Libraries/Modifiers import { LibUsd } from "../../libraries/LibUsd.sol"; import { Diamondable } from "../../Diamondable.sol"; import { ERC20 } from "@openzeppelin/contracts/token/ERC20/ERC20.sol"; import { FixedPointMathLib } from "solady/src/utils/FixedPointMathLib.sol"; contract Degen is Diamondable { struct Storage { mapping(address => string) pfps; mapping(address => uint256) xp; } function store() internal pure returns (Storage storage s) { bytes32 position = keccak256("diamond.degen.storage"); assembly { s.slot := position } } event PfpSet(address degen); function pfp(address degen) public view returns (string memory) { return string.concat("data:image/webp;base64,", store().pfps[degen]); } function setPfp(string memory image) public { require( bytes(image).length <= 16777217, // 1mb base64 "invalid image" ); emit PfpSet(msg.sender); } event XP(address degen, uint256 xp, uint256 totalXp); enum XpType { Buy, Sell, Launch } function attributeXp(address degen, XpType xpType, uint256 amount) public onlyDiamond { amount = FixedPointMathLib.mulWad(amount, LibUsd.getPrice() / 3000); uint256 xp; if (xpType == XpType.Buy) { xp = amount * 800; } else if (xpType == XpType.Sell) { xp = amount * 600; } else if (xpType == XpType.Launch) { xp = amount * 400; } store().xp[degen] += xp; emit XP(degen, xp, store().xp[degen]); } }
// SPDX-License-Identifier: UNKNOWN pragma solidity 0.8.18; // Contracts/Libraries/Modifiers import { Diamondable } from "../../../Diamondable.sol"; // Libraries import { FixedPointMathLib } from "solady/src/utils/FixedPointMathLib.sol"; // Interfaces import { Token } from "../../../../Token.sol"; interface IEqualV2Pair { function totalSupply() external view returns (uint); function getReserves() external view returns (uint _reserve0, uint _reserve1, uint _blockTimestampLast); } interface IEqualV3Router { function weth() external view returns (address); function pairFor(address tokenA, address tokenB, bool stable) external view returns (address pair); function addLiquidityETH( address token, bool stable, uint amountTokenDesired, uint amountTokenMin, uint amountETHMin, address to, uint deadline ) external payable returns (uint amountToken, uint amountETH, uint liquidity); function removeLiquidityETH( address token, bool stable, uint liquidity, uint amountTokenMin, uint amountETHMin, address to, uint deadline ) external returns (uint amountToken, uint amountETH); } contract EqualizerLpHandler is Diamondable { struct Storage { mapping (address => address) positions; } function store() internal pure returns (Storage storage s) { bytes32 position = keccak256("diamond.equallp.storage"); assembly { s.slot := position } } address constant factory = address(0); IEqualV3Router constant router = IEqualV3Router(address(0)); function equal_pairFor(address token) public view returns (address) { return router.pairFor(token, router.weth(), false); } function equal_addLiquidty( address token, uint256 ethAmount, uint256 tokenAmount ) public onlyDiamond { Token(token).approve(address(router), tokenAmount); router.addLiquidityETH{value: ethAmount}( token, false, tokenAmount, 0, 0, address(this), block.timestamp ); store().positions[token] = equal_pairFor(token); } function equal_decreaseLiquidity(address token, uint256 amount) public onlyDiamond { address pair = store().positions[token]; (uint reserve0, uint reserve1,) = IEqualV2Pair(pair).getReserves(); (uint reserveETH,) = token < router.weth() ? (reserve1, reserve0) : (reserve0, reserve1); uint256 lpTokensToBurn = FixedPointMathLib.mulDivUp(amount, IEqualV2Pair(pair).totalSupply(), reserveETH); Token(pair).approve(address(router), lpTokensToBurn); router.removeLiquidityETH( token, false, lpTokensToBurn, 0, amount, address(this), block.timestamp ); } }
// SPDX-License-Identifier: UNKNOWN pragma solidity 0.8.18; // Contracts/Libraries/Modifiers import { LibDex } from "../../libraries/LibDex.sol"; import { LibCore } from "../../libraries/LibCore.sol"; import { LibUtils } from "../../libraries/LibUtils.sol"; import { LibTokens } from "../../libraries/LibTokens.sol"; import { Diamondable } from "../../Diamondable.sol"; // Facets import { FakePools } from "./LaunchStrategies/FakePools.sol"; contract Launcher is Diamondable { function launch(address token, LibTokens.TokenInfo calldata tokenInfo) public onlyDiamond returns (address pair, uint256 eth, uint256 tokens) { if (tokenInfo.strategy == LibTokens.LaunchStrategy.FakeLiquidity) { uint256 fakeEth; (eth, tokens, fakeEth) = FakePools(address(this)).fakepool_close(token); eth = deductLaunchFee(eth); LibDex.addLiquidty(tokenInfo.dex, token, eth + fakeEth, tokens); LibDex.decreaseLiquidity(tokenInfo.dex, token, fakeEth); } else { revert("invalid strategy"); } pair = LibDex.getPair(tokenInfo.dex, token); } function deductLaunchFee(uint256 eth) internal returns (uint256) { uint256 launchFee = LibUtils.calculatePercentage(LibCore.store().launchFee, eth); LibCore.gatherProceeds(launchFee); eth -= launchFee; return eth; } }
// SPDX-License-Identifier: UNKNOWN pragma solidity 0.8.18; // Contracts/Libraries/Modifiers import { LibCore } from "../../../libraries/LibCore.sol"; import { LibFakePools } from "../../../libraries/LibFakePools.sol"; import { LibTokens } from "../../../libraries/LibTokens.sol"; import { LibUsd } from "../../../libraries/LibUsd.sol"; import { Diamondable } from "../../../Diamondable.sol"; // Facets import { Core } from "../Core.sol"; // Third Party import { FixedPointMathLib } from "solady/src/utils/FixedPointMathLib.sol"; contract FakePools is Diamondable { function swapExactTokensForETH(LibFakePools.FakePool storage pool, uint256 tokens) internal returns (uint256) { uint256 out = getAmountOut(tokens, pool.tokenReserve, pool.ethReserve + fakeEther()); pool.tokenReserve += tokens; pool.ethReserve -= out; return out; } function swapExactETHForTokens(LibFakePools.FakePool storage pool, uint256 eth) internal returns (uint256) { uint256 out = getAmountOut(eth, pool.ethReserve + fakeEther(), pool.tokenReserve); pool.tokenReserve -= out; pool.ethReserve += eth; return out; } function getAmountOut(uint256 amountIn, uint256 reserveIn, uint256 reserveOut) internal pure returns (uint256) { uint256 numerator = amountIn * reserveOut; uint256 denominator = reserveIn + amountIn; return numerator / denominator; } function price(LibFakePools.FakePool storage pool, uint256 amount, bool ethOut) internal view returns (uint256) { if (ethOut) { return (amount * (pool.ethReserve + fakeEther())) / pool.tokenReserve; } else { return (amount * pool.tokenReserve) / (pool.ethReserve + fakeEther()); } } function checkMarketCapThreshold(LibFakePools.FakePool storage pool) internal { uint256 ethPrice = price(pool, 1 ether, true); uint256 usdPrice = LibUsd.ethToUsd(ethPrice); uint256 usdMcap = FixedPointMathLib.mulWad(LibCore.store().tokenSupply, usdPrice); if (usdMcap >= LibFakePools.store().usdMcapThreshold) { Core(address(this)).launch(pool.token); } } function fakeEther() internal view returns (uint256) { return LibUsd.usdToEth(LibFakePools.store().fakeUsd); } // PUBLIC function fakepool_quote(address token, uint256 amount, bool ethOut) public view returns (uint256) { LibFakePools.FakePool storage pool = LibFakePools.store().pools[token]; if (ethOut) { // sell uint256 eth = getAmountOut(amount, pool.tokenReserve, (pool.ethReserve + fakeEther())); eth -= LibCore.calculateTradeFee(eth); return eth; } else { // buy uint256 txFee = LibCore.calculateTradeFee(amount); return getAmountOut(amount - txFee, (pool.ethReserve + fakeEther()), pool.tokenReserve); } } function fakepool_create(address token, uint256 supply, bytes calldata data) external onlyDiamond returns (bytes memory) { LibFakePools.FakePool storage pool = LibFakePools.store().pools[token]; pool.token = token; pool.ethReserve = 0; pool.tokenReserve = supply; return abi.encode(LibTokens.LaunchStrategy.FakeLiquidity, pool.ethReserve, pool.tokenReserve, fakeEther()); } function fakepool_close(address token) public onlyDiamond returns (uint256, uint256, uint256) { LibFakePools.Storage storage fp = LibFakePools.store(); LibFakePools.FakePool memory pool = fp.pools[token]; delete fp.pools[token]; return (pool.ethReserve, pool.tokenReserve, fakeEther()); } function fakepool_buy(address token) external onlyDiamond payable returns (uint256, bytes memory) { LibFakePools.FakePool storage pool = LibFakePools.store().pools[token]; require(pool.token != address(0)); uint256 ethIn = LibCore.deductTradeFee(msg.value); uint256 tokensOut = swapExactETHForTokens(pool, ethIn); checkMarketCapThreshold(pool); return (tokensOut, abi.encode(LibTokens.LaunchStrategy.FakeLiquidity, pool.ethReserve, pool.tokenReserve, fakeEther())); } function fakepool_sell(address token, uint256 amount) external onlyDiamond returns (uint256, bytes memory) { LibFakePools.FakePool storage pool = LibFakePools.store().pools[token]; require(pool.token != address(0)); uint256 ethOut = swapExactTokensForETH(pool, amount); ethOut = LibCore.deductTradeFee(ethOut); checkMarketCapThreshold(pool); return (ethOut, abi.encode(LibTokens.LaunchStrategy.FakeLiquidity, pool.ethReserve, pool.tokenReserve, fakeEther())); } }
// SPDX-License-Identifier: UNKNOWN pragma solidity 0.8.18; interface IChainlinkAggregatorV3 { function latestRoundData() external view returns (uint80 roundId, int256 answer, uint256 startedAt, uint256 updatedAt, uint80 answeredInRound); }
// SPDX-License-Identifier: MIT pragma solidity 0.8.18; /******************************************************************************\ * Author: Nick Mudge <[email protected]> (https://twitter.com/mudgen) * EIP-2535 Diamonds: https://eips.ethereum.org/EIPS/eip-2535 /******************************************************************************/ interface IDiamondCut { enum FacetCutAction { Add, Replace, Remove } // Add=0, Replace=1, Remove=2 struct FacetCut { address facetAddress; FacetCutAction action; bytes4[] functionSelectors; } /// @notice Add/replace/remove any number of functions and optionally execute /// a function with delegatecall /// @param _diamondCut Contains the facet addresses and function selectors /// @param _init The address of the contract or facet to execute _calldata /// @param _calldata A function call, including function selector and arguments /// _calldata is executed with delegatecall on _init function diamondCut( FacetCut[] calldata _diamondCut, address _init, bytes calldata _calldata ) external; event DiamondCut(FacetCut[] _diamondCut, address _init, bytes _calldata); }
// SPDX-License-Identifier: UNKNOWN pragma solidity 0.8.18; import { LibUtils } from "./LibUtils.sol"; library LibCore { bytes32 constant STORAGE_POSITION = keccak256("diamond.core.storage"); struct Storage { address proceedsReceiver; uint256 creationPrice; uint16 tradeFee; uint16 launchFee; uint256 tokenSupply; } function store() internal pure returns (Storage storage s) { bytes32 position = STORAGE_POSITION; assembly { s.slot := position } } function gatherProceeds(uint256 amount) internal { payable(store().proceedsReceiver).transfer(amount); } function calculateTradeFee(uint256 eth) internal view returns (uint256) { return LibUtils.calculatePercentage(store().tradeFee, eth); } function deductTradeFee(uint256 eth) internal returns (uint256) { uint256 fee = calculateTradeFee(eth); gatherProceeds(fee); return eth - fee; } }
// SPDX-License-Identifier: UNKNOWN pragma solidity 0.8.18; // Facets import { EqualizerLpHandler } from "../facets/degen/dexes/EqualizerLpHandler.sol"; library LibDex { enum Dex { Shadow, // https://github.com/code-423n4/2024-10-ramses-exchange/blob/main/contracts/CL/periphery/NonfungiblePositionManager.sol, https://github.com/code-423n4/2024-10-ramses-exchange/blob/main/contracts/CL/core/RamsesV3Factory.sol Equalizer // https://ftmscan.com/address/0x2B52294425a9a229322228de659eDE9D146D7c2f#writeContract, https://ftmscan.com/address/0xE6dA85feb3B4E0d6AEd95c41a125fba859bB9d24#writeContract } function getPair(Dex dex, address token) internal view returns (address pair) { if (dex == Dex.Shadow) { // TODO pair = address(0); } else if (dex == Dex.Equalizer) { pair = EqualizerLpHandler(address(this)).equal_pairFor(token); } } function addLiquidty(Dex dex, address token, uint256 ethAmount, uint256 tokenAmount) internal { if (dex == Dex.Shadow) { // TODO } else if (dex == Dex.Equalizer) { EqualizerLpHandler(address(this)).equal_addLiquidty(token, ethAmount, tokenAmount); } } function removeLiquidity(Dex dex, address token) internal { // TODO } function decreaseLiquidity(Dex dex, address token, uint256 amount) internal { if (dex == Dex.Shadow) { // TODO } else if (dex == Dex.Equalizer) { EqualizerLpHandler(address(this)).equal_decreaseLiquidity(token, amount); } } }
// SPDX-License-Identifier: MIT pragma solidity 0.8.18; /******************************************************************************\ * Author: Nick Mudge <[email protected]> (https://twitter.com/mudgen) * EIP-2535 Diamonds: https://eips.ethereum.org/EIPS/eip-2535 /******************************************************************************/ import {IDiamondCut} from "../interfaces/IDiamondCut.sol"; // Remember to add the loupe functions from DiamondLoupeFacet to the diamond. // The loupe functions are required by the EIP2535 Diamonds standard error InitializationFunctionReverted( address _initializationContractAddress, bytes _calldata ); library LibDiamond { bytes32 constant DIAMOND_STORAGE_POSITION = keccak256("diamond.standard.diamond.storage"); struct FacetAddressAndPosition { address facetAddress; uint96 functionSelectorPosition; // position in facetFunctionSelectors.functionSelectors array } struct FacetFunctionSelectors { bytes4[] functionSelectors; uint256 facetAddressPosition; // position of facetAddress in facetAddresses array } struct DiamondStorage { // maps function selector to the facet address and // the position of the selector in the facetFunctionSelectors.selectors array mapping(bytes4 => FacetAddressAndPosition) selectorToFacetAndPosition; // maps facet addresses to function selectors mapping(address => FacetFunctionSelectors) facetFunctionSelectors; // facet addresses address[] facetAddresses; // Used to query if a contract implements an interface. // Used to implement ERC-165. mapping(bytes4 => bool) supportedInterfaces; // owner of the contract address contractOwner; // owner of the diamond itself address diamondAddress; } function diamondStorage() internal pure returns (DiamondStorage storage ds) { bytes32 position = DIAMOND_STORAGE_POSITION; assembly { ds.slot := position } } event OwnershipTransferred( address indexed previousOwner, address indexed newOwner ); function setContractOwner(address _newOwner) internal { DiamondStorage storage ds = diamondStorage(); address previousOwner = ds.contractOwner; ds.contractOwner = _newOwner; emit OwnershipTransferred(previousOwner, _newOwner); } function contractOwner() internal view returns (address contractOwner_) { contractOwner_ = diamondStorage().contractOwner; } function enforceIsContractOwner() internal view { require( msg.sender == diamondStorage().contractOwner, "LibDiamond: Must be contract owner" ); } function enforceDiamondItself() internal view { require( msg.sender == diamondStorage().diamondAddress, "LibDiamond: Must be the diamond" ); } event DiamondCut( IDiamondCut.FacetCut[] _diamondCut, address _init, bytes _calldata ); // Internal function version of diamondCut function diamondCut( IDiamondCut.FacetCut[] memory _diamondCut, address _init, bytes memory _calldata ) internal { for (uint256 facetIndex; facetIndex < _diamondCut.length; facetIndex++) { IDiamondCut.FacetCutAction action = _diamondCut[facetIndex].action; if (action == IDiamondCut.FacetCutAction.Add) { addFunctions( _diamondCut[facetIndex].facetAddress, _diamondCut[facetIndex].functionSelectors ); } else if (action == IDiamondCut.FacetCutAction.Replace) { replaceFunctions( _diamondCut[facetIndex].facetAddress, _diamondCut[facetIndex].functionSelectors ); } else if (action == IDiamondCut.FacetCutAction.Remove) { removeFunctions( _diamondCut[facetIndex].facetAddress, _diamondCut[facetIndex].functionSelectors ); } else { revert("LibDiamondCut: Incorrect FacetCutAction"); } } emit DiamondCut(_diamondCut, _init, _calldata); initializeDiamondCut(_init, _calldata); } function addFunctions( address _facetAddress, bytes4[] memory _functionSelectors ) internal { require( _functionSelectors.length > 0, "LibDiamondCut: No selectors in facet to cut" ); DiamondStorage storage ds = diamondStorage(); require( _facetAddress != address(0), "LibDiamondCut: Add facet can't be address(0)" ); uint96 selectorPosition = uint96( ds.facetFunctionSelectors[_facetAddress].functionSelectors.length ); // add new facet address if it does not exist if (selectorPosition == 0) { addFacet(ds, _facetAddress); } for ( uint256 selectorIndex; selectorIndex < _functionSelectors.length; selectorIndex++ ) { bytes4 selector = _functionSelectors[selectorIndex]; address oldFacetAddress = ds .selectorToFacetAndPosition[selector] .facetAddress; require( oldFacetAddress == address(0), "LibDiamondCut: Can't add function that already exists" ); addFunction(ds, selector, selectorPosition, _facetAddress); selectorPosition++; } } function replaceFunctions( address _facetAddress, bytes4[] memory _functionSelectors ) internal { require( _functionSelectors.length > 0, "LibDiamondCut: No selectors in facet to cut" ); DiamondStorage storage ds = diamondStorage(); require( _facetAddress != address(0), "LibDiamondCut: Add facet can't be address(0)" ); uint96 selectorPosition = uint96( ds.facetFunctionSelectors[_facetAddress].functionSelectors.length ); // add new facet address if it does not exist if (selectorPosition == 0) { addFacet(ds, _facetAddress); } for ( uint256 selectorIndex; selectorIndex < _functionSelectors.length; selectorIndex++ ) { bytes4 selector = _functionSelectors[selectorIndex]; address oldFacetAddress = ds .selectorToFacetAndPosition[selector] .facetAddress; require( oldFacetAddress != _facetAddress, "LibDiamondCut: Can't replace function with same function" ); removeFunction(ds, oldFacetAddress, selector); addFunction(ds, selector, selectorPosition, _facetAddress); selectorPosition++; } } function removeFunctions( address _facetAddress, bytes4[] memory _functionSelectors ) internal { require( _functionSelectors.length > 0, "LibDiamondCut: No selectors in facet to cut" ); DiamondStorage storage ds = diamondStorage(); // if function does not exist then do nothing and return require( _facetAddress == address(0), "LibDiamondCut: Remove facet address must be address(0)" ); for ( uint256 selectorIndex; selectorIndex < _functionSelectors.length; selectorIndex++ ) { bytes4 selector = _functionSelectors[selectorIndex]; address oldFacetAddress = ds .selectorToFacetAndPosition[selector] .facetAddress; removeFunction(ds, oldFacetAddress, selector); } } function addFacet(DiamondStorage storage ds, address _facetAddress) internal { enforceHasContractCode( _facetAddress, "LibDiamondCut: New facet has no code" ); ds.facetFunctionSelectors[_facetAddress].facetAddressPosition = ds .facetAddresses .length; ds.facetAddresses.push(_facetAddress); } function addFunction( DiamondStorage storage ds, bytes4 _selector, uint96 _selectorPosition, address _facetAddress ) internal { ds .selectorToFacetAndPosition[_selector] .functionSelectorPosition = _selectorPosition; ds.facetFunctionSelectors[_facetAddress].functionSelectors.push(_selector); ds.selectorToFacetAndPosition[_selector].facetAddress = _facetAddress; } function removeFunction( DiamondStorage storage ds, address _facetAddress, bytes4 _selector ) internal { require( _facetAddress != address(0), "LibDiamondCut: Can't remove function that doesn't exist" ); // an immutable function is a function defined directly in a diamond require( _facetAddress != address(this), "LibDiamondCut: Can't remove immutable function" ); // replace selector with last selector, then delete last selector uint256 selectorPosition = ds .selectorToFacetAndPosition[_selector] .functionSelectorPosition; uint256 lastSelectorPosition = ds .facetFunctionSelectors[_facetAddress] .functionSelectors .length - 1; // if not the same then replace _selector with lastSelector if (selectorPosition != lastSelectorPosition) { bytes4 lastSelector = ds .facetFunctionSelectors[_facetAddress] .functionSelectors[lastSelectorPosition]; ds.facetFunctionSelectors[_facetAddress].functionSelectors[ selectorPosition ] = lastSelector; ds .selectorToFacetAndPosition[lastSelector] .functionSelectorPosition = uint96(selectorPosition); } // delete the last selector ds.facetFunctionSelectors[_facetAddress].functionSelectors.pop(); delete ds.selectorToFacetAndPosition[_selector]; // if no more selectors for facet address then delete the facet address if (lastSelectorPosition == 0) { // replace facet address with last facet address and delete last facet address uint256 lastFacetAddressPosition = ds.facetAddresses.length - 1; uint256 facetAddressPosition = ds .facetFunctionSelectors[_facetAddress] .facetAddressPosition; if (facetAddressPosition != lastFacetAddressPosition) { address lastFacetAddress = ds.facetAddresses[lastFacetAddressPosition]; ds.facetAddresses[facetAddressPosition] = lastFacetAddress; ds .facetFunctionSelectors[lastFacetAddress] .facetAddressPosition = facetAddressPosition; } ds.facetAddresses.pop(); delete ds.facetFunctionSelectors[_facetAddress].facetAddressPosition; } } function initializeDiamondCut( address _init, bytes memory _calldata ) internal { if (_init == address(0)) { return; } enforceHasContractCode(_init, "LibDiamondCut: _init address has no code"); (bool success, bytes memory error) = _init.delegatecall(_calldata); if (!success) { if (error.length > 0) { // bubble up error /// @solidity memory-safe-assembly assembly { let returndata_size := mload(error) revert(add(32, error), returndata_size) } } else { revert InitializationFunctionReverted(_init, _calldata); } } } function enforceHasContractCode( address _contract, string memory _errorMessage ) internal view { uint256 contractSize; assembly { contractSize := extcodesize(_contract) } require(contractSize > 0, _errorMessage); } }
// SPDX-License-Identifier: UNKNOWN pragma solidity 0.8.18; library LibFakePools { bytes32 constant STORAGE_POSITION = keccak256("diamond.fakepools.storage"); struct FakePool { address token; uint256 ethReserve; uint256 tokenReserve; } struct Storage { uint256 fakeUsd; uint256 usdMcapThreshold; mapping(address => FakePool) pools; } function store() internal pure returns (Storage storage s) { bytes32 position = STORAGE_POSITION; assembly { s.slot := position } } }
// SPDX-License-Identifier: UNKNOWN pragma solidity 0.8.18; library LibLST { bytes32 constant STORAGE_POSITION = keccak256("diamond.lst.storage"); struct Storage { uint256 staked; } function store() internal pure returns (Storage storage s) { bytes32 position = STORAGE_POSITION; assembly { s.slot := position } } function addLiquidity(uint256 ethAmount) internal { store().staked += ethAmount; } function removeLiquidity(uint256 ethAmount) internal { store().staked -= ethAmount; } }
// SPDX-License-Identifier: UNKNOWN pragma solidity 0.8.18; import { LibDex } from "./LibDex.sol"; library LibTokens { bytes32 constant STORAGE_POSITION = keccak256("diamond.tokens.storage"); enum LaunchStrategy { FakeLiquidity } struct TokenInfo { address creator; LaunchStrategy strategy; LibDex.Dex dex; } struct Storage { mapping(address => TokenInfo) tokens; } function store() internal pure returns (Storage storage s) { bytes32 position = STORAGE_POSITION; assembly { s.slot := position } } }
// SPDX-License-Identifier: UNKNOWN pragma solidity 0.8.18; import { IChainlinkAggregatorV3 } from "../interfaces/IChainlinkAggregatorV3.sol"; import { FixedPointMathLib } from "solady/src/utils/FixedPointMathLib.sol"; library LibUsd { bytes32 constant STORAGE_POSITION = keccak256("diamond.usd.storage"); struct ChainlinkOracle { address priceFeed; uint256 heartBeat; } struct Storage { ChainlinkOracle usdOracle; } function store() internal pure returns (Storage storage s) { bytes32 position = STORAGE_POSITION; assembly { s.slot := position } } function ethToUsd(uint256 ethAmount) internal view returns (uint256) { return FixedPointMathLib.mulWad(ethAmount, getPrice()); } function usdToEth(uint256 usdAmount) internal view returns (uint256) { return FixedPointMathLib.divWad(usdAmount, getPrice()); } function getPrice() internal view returns (uint256) { ChainlinkOracle storage oracle = store().usdOracle; (, int256 price, uint256 timeStamp,,) = IChainlinkAggregatorV3(oracle.priceFeed).latestRoundData(); require(uint256(timeStamp) >= block.timestamp - oracle.heartBeat, "stale pricefeed"); // usd oracle returns the price in 8 decimals, we want 18 return uint256(price) * (10 ** 10); } }
// SPDX-License-Identifier: UNKNOWN pragma solidity 0.8.18; library LibUtils { function calculatePercentage(uint16 fee, uint256 amount) internal pure returns (uint256) { return amount * fee / 1000; } }
// SPDX-License-Identifier: UNKNOWN pragma solidity 0.8.18; // Contracts/Libraries/Modifiers import { ERC20 } from "@openzeppelin/contracts/token/ERC20/ERC20.sol"; import { Base64 } from "solady/src/utils/Base64.sol"; import { LibString } from "solady/src/utils/LibString.sol"; contract Token is ERC20 { address internal protocol; bool internal locked = true; address internal creator; string internal descripiton; bytes internal image; string[] internal links; constructor(address _creator, string memory name, string memory symbol, string memory _desc, bytes memory _image, string[] memory _links, uint256 _supply, address _protocol) ERC20(name, symbol) { protocol = _protocol; creator = _creator; descripiton = _desc; image = _image; links = _links; _mint(msg.sender, _supply); } function unlock() external { require(msg.sender == protocol && locked == true); locked = false; } function updateMetadata( string calldata _desc, bytes calldata _image, string[] calldata _links ) external { require(msg.sender == creator); if (bytes(_desc).length > 0) { descripiton = _desc; } if (_image.length > 0) { image = _image; } if (_links.length != links.length) { links = _links; } } function tokenURI( uint256 _unused ) public view returns (string memory) { string memory linksString = ''; for (uint256 i = 0; i < links.length; i++) { linksString = string.concat(linksString, '"', links[i], '"'); if (i < links.length - 1) { linksString = string.concat(linksString, ","); } } return string.concat("data:application/json;base64,", Base64.encode( bytes( string.concat( "{", '"creator":"', LibString.toHexString(creator), '",', '"name":"', name(), '",', '"symbol":"', symbol(), '",', '"supply":"', LibString.toString(totalSupply() / (10 ** 18)), '",', '"description":"', descripiton, '",', '"links":[',linksString,'],', '"image_data":"data:image/webp;base64,', Base64.encode(image), '",', '"background_color":"000000"', "}" ) ) ) ); } function _beforeTokenTransfer(address from, address to, uint256 amount) internal override { if (locked) { require(from == protocol || to == protocol, "transfer not allowed before launch"); } } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.4; /// @notice Library to encode strings in Base64. /// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/Base64.sol) /// @author Modified from Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/Base64.sol) /// @author Modified from (https://github.com/Brechtpd/base64/blob/main/base64.sol) by Brecht Devos - <[email protected]>. library Base64 { /// @dev Encodes `data` using the base64 encoding described in RFC 4648. /// See: https://datatracker.ietf.org/doc/html/rfc4648 /// @param fileSafe Whether to replace '+' with '-' and '/' with '_'. /// @param noPadding Whether to strip away the padding. function encode(bytes memory data, bool fileSafe, bool noPadding) internal pure returns (string memory result) { /// @solidity memory-safe-assembly assembly { let dataLength := mload(data) if dataLength { // Multiply by 4/3 rounded up. // The `shl(2, ...)` is equivalent to multiplying by 4. let encodedLength := shl(2, div(add(dataLength, 2), 3)) // Set `result` to point to the start of the free memory. result := mload(0x40) // Store the table into the scratch space. // Offsetted by -1 byte so that the `mload` will load the character. // We will rewrite the free memory pointer at `0x40` later with // the allocated size. // The magic constant 0x0670 will turn "-_" into "+/". mstore(0x1f, "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdef") mstore(0x3f, xor("ghijklmnopqrstuvwxyz0123456789-_", mul(iszero(fileSafe), 0x0670))) // Skip the first slot, which stores the length. let ptr := add(result, 0x20) let end := add(ptr, encodedLength) let dataEnd := add(add(0x20, data), dataLength) let dataEndValue := mload(dataEnd) // Cache the value at the `dataEnd` slot. mstore(dataEnd, 0x00) // Zeroize the `dataEnd` slot to clear dirty bits. // Run over the input, 3 bytes at a time. for {} 1 {} { data := add(data, 3) // Advance 3 bytes. let input := mload(data) // Write 4 bytes. Optimized for fewer stack operations. mstore8(0, mload(and(shr(18, input), 0x3F))) mstore8(1, mload(and(shr(12, input), 0x3F))) mstore8(2, mload(and(shr(6, input), 0x3F))) mstore8(3, mload(and(input, 0x3F))) mstore(ptr, mload(0x00)) ptr := add(ptr, 4) // Advance 4 bytes. if iszero(lt(ptr, end)) { break } } mstore(dataEnd, dataEndValue) // Restore the cached value at `dataEnd`. mstore(0x40, add(end, 0x20)) // Allocate the memory. // Equivalent to `o = [0, 2, 1][dataLength % 3]`. let o := div(2, mod(dataLength, 3)) // Offset `ptr` and pad with '='. We can simply write over the end. mstore(sub(ptr, o), shl(240, 0x3d3d)) // Set `o` to zero if there is padding. o := mul(iszero(iszero(noPadding)), o) mstore(sub(ptr, o), 0) // Zeroize the slot after the string. mstore(result, sub(encodedLength, o)) // Store the length. } } } /// @dev Encodes `data` using the base64 encoding described in RFC 4648. /// Equivalent to `encode(data, false, false)`. function encode(bytes memory data) internal pure returns (string memory result) { result = encode(data, false, false); } /// @dev Encodes `data` using the base64 encoding described in RFC 4648. /// Equivalent to `encode(data, fileSafe, false)`. function encode(bytes memory data, bool fileSafe) internal pure returns (string memory result) { result = encode(data, fileSafe, false); } /// @dev Decodes base64 encoded `data`. /// /// Supports: /// - RFC 4648 (both standard and file-safe mode). /// - RFC 3501 (63: ','). /// /// Does not support: /// - Line breaks. /// /// Note: For performance reasons, /// this function will NOT revert on invalid `data` inputs. /// Outputs for invalid inputs will simply be undefined behaviour. /// It is the user's responsibility to ensure that the `data` /// is a valid base64 encoded string. function decode(string memory data) internal pure returns (bytes memory result) { /// @solidity memory-safe-assembly assembly { let dataLength := mload(data) if dataLength { let decodedLength := mul(shr(2, dataLength), 3) for {} 1 {} { // If padded. if iszero(and(dataLength, 3)) { let t := xor(mload(add(data, dataLength)), 0x3d3d) // forgefmt: disable-next-item decodedLength := sub( decodedLength, add(iszero(byte(30, t)), iszero(byte(31, t))) ) break } // If non-padded. decodedLength := add(decodedLength, sub(and(dataLength, 3), 1)) break } result := mload(0x40) // Write the length of the bytes. mstore(result, decodedLength) // Skip the first slot, which stores the length. let ptr := add(result, 0x20) let end := add(ptr, decodedLength) // Load the table into the scratch space. // Constants are optimized for smaller bytecode with zero gas overhead. // `m` also doubles as the mask of the upper 6 bits. let m := 0xfc000000fc00686c7074787c8084888c9094989ca0a4a8acb0b4b8bcc0c4c8cc mstore(0x5b, m) mstore(0x3b, 0x04080c1014181c2024282c3034383c4044484c5054585c6064) mstore(0x1a, 0xf8fcf800fcd0d4d8dce0e4e8ecf0f4) for {} 1 {} { // Read 4 bytes. data := add(data, 4) let input := mload(data) // Write 3 bytes. // forgefmt: disable-next-item mstore(ptr, or( and(m, mload(byte(28, input))), shr(6, or( and(m, mload(byte(29, input))), shr(6, or( and(m, mload(byte(30, input))), shr(6, mload(byte(31, input))) )) )) )) ptr := add(ptr, 3) if iszero(lt(ptr, end)) { break } } mstore(0x40, add(end, 0x20)) // Allocate the memory. mstore(end, 0) // Zeroize the slot after the bytes. mstore(0x60, 0) // Restore the zero slot. } } } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.4; /// @notice Library for buffers with automatic capacity resizing. /// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/DynamicBufferLib.sol) /// @author Modified from cozyco (https://github.com/samkingco/cozyco/blob/main/contracts/utils/DynamicBuffer.sol) library DynamicBufferLib { /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* STRUCTS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev Type to represent a dynamic buffer in memory. /// You can directly assign to `data`, and the `p` function will /// take care of the memory allocation. struct DynamicBuffer { bytes data; } /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* OPERATIONS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ // Some of these functions returns the same buffer for function chaining. // `e.g. `buffer.p("1").p("2")`. /// @dev Shorthand for `buffer.data.length`. function length(DynamicBuffer memory buffer) internal pure returns (uint256) { return buffer.data.length; } /// @dev Reserves at least `minimum` amount of contiguous memory. function reserve(DynamicBuffer memory buffer, uint256 minimum) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = buffer; uint256 n = buffer.data.length; if (minimum > n) { uint256 i = 0x40; do {} while ((i <<= 1) < minimum); bytes memory data; /// @solidity memory-safe-assembly assembly { data := 0x01 mstore(data, sub(i, n)) } result = p(result, data); } } /// @dev Clears the buffer without deallocating the memory. function clear(DynamicBuffer memory buffer) internal pure returns (DynamicBuffer memory result) { _deallocate(result); /// @solidity memory-safe-assembly assembly { mstore(mload(buffer), 0) } result = buffer; } /// @dev Returns a string pointing to the underlying bytes data. /// Note: The string WILL change if the buffer is updated. function s(DynamicBuffer memory buffer) internal pure returns (string memory) { return string(buffer.data); } /// @dev Appends `data` to `buffer`. function p(DynamicBuffer memory buffer, bytes memory data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = buffer; if (data.length == uint256(0)) return result; /// @solidity memory-safe-assembly assembly { let w := not(0x1f) let bufData := mload(buffer) let bufDataLen := mload(bufData) let newBufDataLen := add(mload(data), bufDataLen) // Some random prime number to multiply `cap`, so that // we know that the `cap` is for a dynamic buffer. // Selected to be larger than any memory pointer realistically. let prime := 1621250193422201 let cap := mload(add(bufData, w)) // `mload(sub(bufData, 0x20))`. // Extract `cap`, initializing it to zero if it is not a multiple of `prime`. cap := mul(div(cap, prime), iszero(mod(cap, prime))) // Expand / Reallocate memory if required. // Note that we need to allocate an extra word for the length, and // and another extra word as a safety word (giving a total of 0x40 bytes). // Without the safety word, the backwards copying can cause a buffer overflow. for {} iszero(lt(newBufDataLen, cap)) {} { // Approximately more than double the capacity to ensure more than enough space. let newCap := and(add(cap, add(or(cap, newBufDataLen), 0x20)), w) // If the memory is contiguous, we can simply expand it. if iszero(or(xor(mload(0x40), add(bufData, add(0x40, cap))), eq(bufData, 0x60))) { // Store `cap * prime` in the word before the length. mstore(add(bufData, w), mul(prime, newCap)) mstore(0x40, add(bufData, add(0x40, newCap))) // Expand the memory allocation. break } // Set the `newBufData` to point to the word after `cap`. let newBufData := add(mload(0x40), 0x20) mstore(0x40, add(newBufData, add(0x40, newCap))) // Reallocate the memory. mstore(buffer, newBufData) // Store the `newBufData`. // Copy `bufData` one word at a time, backwards. for { let o := and(add(bufDataLen, 0x20), w) } 1 {} { mstore(add(newBufData, o), mload(add(bufData, o))) o := add(o, w) // `sub(o, 0x20)`. if iszero(o) { break } } // Store `cap * prime` in the word before the length. mstore(add(newBufData, w), mul(prime, newCap)) bufData := newBufData // Assign `newBufData` to `bufData`. break } // If it's a reserve operation, set the variables to skip the appending. if eq(data, 0x01) { mstore(data, 0x00) newBufDataLen := bufDataLen } // Copy `data` one word at a time, backwards. for { let o := and(add(mload(data), 0x20), w) } 1 {} { mstore(add(add(bufData, bufDataLen), o), mload(add(data, o))) o := add(o, w) // `sub(o, 0x20)`. if iszero(o) { break } } mstore(add(add(bufData, 0x20), newBufDataLen), 0) // Zeroize the word after the buffer. mstore(bufData, newBufDataLen) // Store the length. } } /// @dev Appends `data0`, `data1` to `buffer`. function p(DynamicBuffer memory buffer, bytes memory data0, bytes memory data1) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(p(buffer, data0), data1); } /// @dev Appends `data0` .. `data2` to `buffer`. function p( DynamicBuffer memory buffer, bytes memory data0, bytes memory data1, bytes memory data2 ) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(p(p(buffer, data0), data1), data2); } /// @dev Appends `data0` .. `data3` to `buffer`. function p( DynamicBuffer memory buffer, bytes memory data0, bytes memory data1, bytes memory data2, bytes memory data3 ) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(p(p(p(buffer, data0), data1), data2), data3); } /// @dev Appends `data0` .. `data4` to `buffer`. function p( DynamicBuffer memory buffer, bytes memory data0, bytes memory data1, bytes memory data2, bytes memory data3, bytes memory data4 ) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(p(p(p(p(buffer, data0), data1), data2), data3), data4); } /// @dev Appends `data0` .. `data5` to `buffer`. function p( DynamicBuffer memory buffer, bytes memory data0, bytes memory data1, bytes memory data2, bytes memory data3, bytes memory data4, bytes memory data5 ) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(p(p(p(p(p(buffer, data0), data1), data2), data3), data4), data5); } /// @dev Appends `data0` .. `data6` to `buffer`. function p( DynamicBuffer memory buffer, bytes memory data0, bytes memory data1, bytes memory data2, bytes memory data3, bytes memory data4, bytes memory data5, bytes memory data6 ) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(p(p(p(p(p(p(buffer, data0), data1), data2), data3), data4), data5), data6); } /// @dev Appends `abi.encodePacked(bool(data))` to buffer. function pBool(DynamicBuffer memory buffer, bool data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); uint256 casted; /// @solidity memory-safe-assembly assembly { casted := iszero(iszero(data)) } result = p(buffer, _single(casted, 1)); } /// @dev Appends `abi.encodePacked(address(data))` to buffer. function pAddress(DynamicBuffer memory buffer, address data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(uint256(uint160(data)), 20)); } /// @dev Appends `abi.encodePacked(uint8(data))` to buffer. function pUint8(DynamicBuffer memory buffer, uint8 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 1)); } /// @dev Appends `abi.encodePacked(uint16(data))` to buffer. function pUint16(DynamicBuffer memory buffer, uint16 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 2)); } /// @dev Appends `abi.encodePacked(uint24(data))` to buffer. function pUint24(DynamicBuffer memory buffer, uint24 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 3)); } /// @dev Appends `abi.encodePacked(uint32(data))` to buffer. function pUint32(DynamicBuffer memory buffer, uint32 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 4)); } /// @dev Appends `abi.encodePacked(uint40(data))` to buffer. function pUint40(DynamicBuffer memory buffer, uint40 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 5)); } /// @dev Appends `abi.encodePacked(uint48(data))` to buffer. function pUint48(DynamicBuffer memory buffer, uint48 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 6)); } /// @dev Appends `abi.encodePacked(uint56(data))` to buffer. function pUint56(DynamicBuffer memory buffer, uint56 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 7)); } /// @dev Appends `abi.encodePacked(uint64(data))` to buffer. function pUint64(DynamicBuffer memory buffer, uint64 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 8)); } /// @dev Appends `abi.encodePacked(uint72(data))` to buffer. function pUint72(DynamicBuffer memory buffer, uint72 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 9)); } /// @dev Appends `abi.encodePacked(uint80(data))` to buffer. function pUint80(DynamicBuffer memory buffer, uint80 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 10)); } /// @dev Appends `abi.encodePacked(uint88(data))` to buffer. function pUint88(DynamicBuffer memory buffer, uint88 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 11)); } /// @dev Appends `abi.encodePacked(uint96(data))` to buffer. function pUint96(DynamicBuffer memory buffer, uint96 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 12)); } /// @dev Appends `abi.encodePacked(uint104(data))` to buffer. function pUint104(DynamicBuffer memory buffer, uint104 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 13)); } /// @dev Appends `abi.encodePacked(uint112(data))` to buffer. function pUint112(DynamicBuffer memory buffer, uint112 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 14)); } /// @dev Appends `abi.encodePacked(uint120(data))` to buffer. function pUint120(DynamicBuffer memory buffer, uint120 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 15)); } /// @dev Appends `abi.encodePacked(uint128(data))` to buffer. function pUint128(DynamicBuffer memory buffer, uint128 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 16)); } /// @dev Appends `abi.encodePacked(uint136(data))` to buffer. function pUint136(DynamicBuffer memory buffer, uint136 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 17)); } /// @dev Appends `abi.encodePacked(uint144(data))` to buffer. function pUint144(DynamicBuffer memory buffer, uint144 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 18)); } /// @dev Appends `abi.encodePacked(uint152(data))` to buffer. function pUint152(DynamicBuffer memory buffer, uint152 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 19)); } /// @dev Appends `abi.encodePacked(uint160(data))` to buffer. function pUint160(DynamicBuffer memory buffer, uint160 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 20)); } /// @dev Appends `abi.encodePacked(uint168(data))` to buffer. function pUint168(DynamicBuffer memory buffer, uint168 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 21)); } /// @dev Appends `abi.encodePacked(uint176(data))` to buffer. function pUint176(DynamicBuffer memory buffer, uint176 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 22)); } /// @dev Appends `abi.encodePacked(uint184(data))` to buffer. function pUint184(DynamicBuffer memory buffer, uint184 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 23)); } /// @dev Appends `abi.encodePacked(uint192(data))` to buffer. function pUint192(DynamicBuffer memory buffer, uint192 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 24)); } /// @dev Appends `abi.encodePacked(uint200(data))` to buffer. function pUint200(DynamicBuffer memory buffer, uint200 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 25)); } /// @dev Appends `abi.encodePacked(uint208(data))` to buffer. function pUint208(DynamicBuffer memory buffer, uint208 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 26)); } /// @dev Appends `abi.encodePacked(uint216(data))` to buffer. function pUint216(DynamicBuffer memory buffer, uint216 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 27)); } /// @dev Appends `abi.encodePacked(uint224(data))` to buffer. function pUint224(DynamicBuffer memory buffer, uint224 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 28)); } /// @dev Appends `abi.encodePacked(uint232(data))` to buffer. function pUint232(DynamicBuffer memory buffer, uint232 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 29)); } /// @dev Appends `abi.encodePacked(uint240(data))` to buffer. function pUint240(DynamicBuffer memory buffer, uint240 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 30)); } /// @dev Appends `abi.encodePacked(uint248(data))` to buffer. function pUint248(DynamicBuffer memory buffer, uint248 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 31)); } /// @dev Appends `abi.encodePacked(uint256(data))` to buffer. function pUint256(DynamicBuffer memory buffer, uint256 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(data, 32)); } /// @dev Appends `abi.encodePacked(bytes1(data))` to buffer. function pBytes1(DynamicBuffer memory buffer, bytes1 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 1)); } /// @dev Appends `abi.encodePacked(bytes2(data))` to buffer. function pBytes2(DynamicBuffer memory buffer, bytes2 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 2)); } /// @dev Appends `abi.encodePacked(bytes3(data))` to buffer. function pBytes3(DynamicBuffer memory buffer, bytes3 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 3)); } /// @dev Appends `abi.encodePacked(bytes4(data))` to buffer. function pBytes4(DynamicBuffer memory buffer, bytes4 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 4)); } /// @dev Appends `abi.encodePacked(bytes5(data))` to buffer. function pBytes5(DynamicBuffer memory buffer, bytes5 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 5)); } /// @dev Appends `abi.encodePacked(bytes6(data))` to buffer. function pBytes6(DynamicBuffer memory buffer, bytes6 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 6)); } /// @dev Appends `abi.encodePacked(bytes7(data))` to buffer. function pBytes7(DynamicBuffer memory buffer, bytes7 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 7)); } /// @dev Appends `abi.encodePacked(bytes8(data))` to buffer. function pBytes8(DynamicBuffer memory buffer, bytes8 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 8)); } /// @dev Appends `abi.encodePacked(bytes9(data))` to buffer. function pBytes9(DynamicBuffer memory buffer, bytes9 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 9)); } /// @dev Appends `abi.encodePacked(bytes10(data))` to buffer. function pBytes10(DynamicBuffer memory buffer, bytes10 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 10)); } /// @dev Appends `abi.encodePacked(bytes11(data))` to buffer. function pBytes11(DynamicBuffer memory buffer, bytes11 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 11)); } /// @dev Appends `abi.encodePacked(bytes12(data))` to buffer. function pBytes12(DynamicBuffer memory buffer, bytes12 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 12)); } /// @dev Appends `abi.encodePacked(bytes13(data))` to buffer. function pBytes13(DynamicBuffer memory buffer, bytes13 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 13)); } /// @dev Appends `abi.encodePacked(bytes14(data))` to buffer. function pBytes14(DynamicBuffer memory buffer, bytes14 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 14)); } /// @dev Appends `abi.encodePacked(bytes15(data))` to buffer. function pBytes15(DynamicBuffer memory buffer, bytes15 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 15)); } /// @dev Appends `abi.encodePacked(bytes16(data))` to buffer. function pBytes16(DynamicBuffer memory buffer, bytes16 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 16)); } /// @dev Appends `abi.encodePacked(bytes17(data))` to buffer. function pBytes17(DynamicBuffer memory buffer, bytes17 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 17)); } /// @dev Appends `abi.encodePacked(bytes18(data))` to buffer. function pBytes18(DynamicBuffer memory buffer, bytes18 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 18)); } /// @dev Appends `abi.encodePacked(bytes19(data))` to buffer. function pBytes19(DynamicBuffer memory buffer, bytes19 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 19)); } /// @dev Appends `abi.encodePacked(bytes20(data))` to buffer. function pBytes20(DynamicBuffer memory buffer, bytes20 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 20)); } /// @dev Appends `abi.encodePacked(bytes21(data))` to buffer. function pBytes21(DynamicBuffer memory buffer, bytes21 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 21)); } /// @dev Appends `abi.encodePacked(bytes22(data))` to buffer. function pBytes22(DynamicBuffer memory buffer, bytes22 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 22)); } /// @dev Appends `abi.encodePacked(bytes23(data))` to buffer. function pBytes23(DynamicBuffer memory buffer, bytes23 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 23)); } /// @dev Appends `abi.encodePacked(bytes24(data))` to buffer. function pBytes24(DynamicBuffer memory buffer, bytes24 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 24)); } /// @dev Appends `abi.encodePacked(bytes25(data))` to buffer. function pBytes25(DynamicBuffer memory buffer, bytes25 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 25)); } /// @dev Appends `abi.encodePacked(bytes26(data))` to buffer. function pBytes26(DynamicBuffer memory buffer, bytes26 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 26)); } /// @dev Appends `abi.encodePacked(bytes27(data))` to buffer. function pBytes27(DynamicBuffer memory buffer, bytes27 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 27)); } /// @dev Appends `abi.encodePacked(bytes28(data))` to buffer. function pBytes28(DynamicBuffer memory buffer, bytes28 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 28)); } /// @dev Appends `abi.encodePacked(bytes29(data))` to buffer. function pBytes29(DynamicBuffer memory buffer, bytes29 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 29)); } /// @dev Appends `abi.encodePacked(bytes30(data))` to buffer. function pBytes30(DynamicBuffer memory buffer, bytes30 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 30)); } /// @dev Appends `abi.encodePacked(bytes31(data))` to buffer. function pBytes31(DynamicBuffer memory buffer, bytes31 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 31)); } /// @dev Appends `abi.encodePacked(bytes32(data))` to buffer. function pBytes32(DynamicBuffer memory buffer, bytes32 data) internal pure returns (DynamicBuffer memory result) { _deallocate(result); result = p(buffer, _single(bytes32(data), 32)); } /// @dev Shorthand for returning a new buffer. function p() internal pure returns (DynamicBuffer memory result) {} /// @dev Shorthand for `p(p(), data)`. function p(bytes memory data) internal pure returns (DynamicBuffer memory result) { p(result, data); } /// @dev Shorthand for `p(p(), data0, data1)`. function p(bytes memory data0, bytes memory data1) internal pure returns (DynamicBuffer memory result) { p(p(result, data0), data1); } /// @dev Shorthand for `p(p(), data0, .., data2)`. function p(bytes memory data0, bytes memory data1, bytes memory data2) internal pure returns (DynamicBuffer memory result) { p(p(p(result, data0), data1), data2); } /// @dev Shorthand for `p(p(), data0, .., data3)`. function p(bytes memory data0, bytes memory data1, bytes memory data2, bytes memory data3) internal pure returns (DynamicBuffer memory result) { p(p(p(p(result, data0), data1), data2), data3); } /// @dev Shorthand for `p(p(), data0, .., data4)`. function p( bytes memory data0, bytes memory data1, bytes memory data2, bytes memory data3, bytes memory data4 ) internal pure returns (DynamicBuffer memory result) { p(p(p(p(p(result, data0), data1), data2), data3), data4); } /// @dev Shorthand for `p(p(), data0, .., data5)`. function p( bytes memory data0, bytes memory data1, bytes memory data2, bytes memory data3, bytes memory data4, bytes memory data5 ) internal pure returns (DynamicBuffer memory result) { p(p(p(p(p(p(result, data0), data1), data2), data3), data4), data5); } /// @dev Shorthand for `p(p(), data0, .., data6)`. function p( bytes memory data0, bytes memory data1, bytes memory data2, bytes memory data3, bytes memory data4, bytes memory data5, bytes memory data6 ) internal pure returns (DynamicBuffer memory result) { p(p(p(p(p(p(p(result, data0), data1), data2), data3), data4), data5), data6); } /// @dev Shorthand for `pBool(p(), data)`. function pBool(bool data) internal pure returns (DynamicBuffer memory result) { pBool(result, data); } /// @dev Shorthand for `pAddress(p(), data)`. function pAddress(address data) internal pure returns (DynamicBuffer memory result) { pAddress(result, data); } /// @dev Shorthand for `pUint8(p(), data)`. function pUint8(uint8 data) internal pure returns (DynamicBuffer memory result) { pUint8(result, data); } /// @dev Shorthand for `pUint16(p(), data)`. function pUint16(uint16 data) internal pure returns (DynamicBuffer memory result) { pUint16(result, data); } /// @dev Shorthand for `pUint24(p(), data)`. function pUint24(uint24 data) internal pure returns (DynamicBuffer memory result) { pUint24(result, data); } /// @dev Shorthand for `pUint32(p(), data)`. function pUint32(uint32 data) internal pure returns (DynamicBuffer memory result) { pUint32(result, data); } /// @dev Shorthand for `pUint40(p(), data)`. function pUint40(uint40 data) internal pure returns (DynamicBuffer memory result) { pUint40(result, data); } /// @dev Shorthand for `pUint48(p(), data)`. function pUint48(uint48 data) internal pure returns (DynamicBuffer memory result) { pUint48(result, data); } /// @dev Shorthand for `pUint56(p(), data)`. function pUint56(uint56 data) internal pure returns (DynamicBuffer memory result) { pUint56(result, data); } /// @dev Shorthand for `pUint64(p(), data)`. function pUint64(uint64 data) internal pure returns (DynamicBuffer memory result) { pUint64(result, data); } /// @dev Shorthand for `pUint72(p(), data)`. function pUint72(uint72 data) internal pure returns (DynamicBuffer memory result) { pUint72(result, data); } /// @dev Shorthand for `pUint80(p(), data)`. function pUint80(uint80 data) internal pure returns (DynamicBuffer memory result) { pUint80(result, data); } /// @dev Shorthand for `pUint88(p(), data)`. function pUint88(uint88 data) internal pure returns (DynamicBuffer memory result) { pUint88(result, data); } /// @dev Shorthand for `pUint96(p(), data)`. function pUint96(uint96 data) internal pure returns (DynamicBuffer memory result) { pUint96(result, data); } /// @dev Shorthand for `pUint104(p(), data)`. function pUint104(uint104 data) internal pure returns (DynamicBuffer memory result) { pUint104(result, data); } /// @dev Shorthand for `pUint112(p(), data)`. function pUint112(uint112 data) internal pure returns (DynamicBuffer memory result) { pUint112(result, data); } /// @dev Shorthand for `pUint120(p(), data)`. function pUint120(uint120 data) internal pure returns (DynamicBuffer memory result) { pUint120(result, data); } /// @dev Shorthand for `pUint128(p(), data)`. function pUint128(uint128 data) internal pure returns (DynamicBuffer memory result) { pUint128(result, data); } /// @dev Shorthand for `pUint136(p(), data)`. function pUint136(uint136 data) internal pure returns (DynamicBuffer memory result) { pUint136(result, data); } /// @dev Shorthand for `pUint144(p(), data)`. function pUint144(uint144 data) internal pure returns (DynamicBuffer memory result) { pUint144(result, data); } /// @dev Shorthand for `pUint152(p(), data)`. function pUint152(uint152 data) internal pure returns (DynamicBuffer memory result) { pUint152(result, data); } /// @dev Shorthand for `pUint160(p(), data)`. function pUint160(uint160 data) internal pure returns (DynamicBuffer memory result) { pUint160(result, data); } /// @dev Shorthand for `pUint168(p(), data)`. function pUint168(uint168 data) internal pure returns (DynamicBuffer memory result) { pUint168(result, data); } /// @dev Shorthand for `pUint176(p(), data)`. function pUint176(uint176 data) internal pure returns (DynamicBuffer memory result) { pUint176(result, data); } /// @dev Shorthand for `pUint184(p(), data)`. function pUint184(uint184 data) internal pure returns (DynamicBuffer memory result) { pUint184(result, data); } /// @dev Shorthand for `pUint192(p(), data)`. function pUint192(uint192 data) internal pure returns (DynamicBuffer memory result) { pUint192(result, data); } /// @dev Shorthand for `pUint200(p(), data)`. function pUint200(uint200 data) internal pure returns (DynamicBuffer memory result) { pUint200(result, data); } /// @dev Shorthand for `pUint208(p(), data)`. function pUint208(uint208 data) internal pure returns (DynamicBuffer memory result) { pUint208(result, data); } /// @dev Shorthand for `pUint216(p(), data)`. function pUint216(uint216 data) internal pure returns (DynamicBuffer memory result) { pUint216(result, data); } /// @dev Shorthand for `pUint224(p(), data)`. function pUint224(uint224 data) internal pure returns (DynamicBuffer memory result) { pUint224(result, data); } /// @dev Shorthand for `pUint232(p(), data)`. function pUint232(uint232 data) internal pure returns (DynamicBuffer memory result) { pUint232(result, data); } /// @dev Shorthand for `pUint240(p(), data)`. function pUint240(uint240 data) internal pure returns (DynamicBuffer memory result) { pUint240(result, data); } /// @dev Shorthand for `pUint248(p(), data)`. function pUint248(uint248 data) internal pure returns (DynamicBuffer memory result) { pUint248(result, data); } /// @dev Shorthand for `pUint256(p(), data)`. function pUint256(uint256 data) internal pure returns (DynamicBuffer memory result) { pUint256(result, data); } /// @dev Shorthand for `pBytes1(p(), data)`. function pBytes1(bytes1 data) internal pure returns (DynamicBuffer memory result) { pBytes1(result, data); } /// @dev Shorthand for `pBytes2(p(), data)`. function pBytes2(bytes2 data) internal pure returns (DynamicBuffer memory result) { pBytes2(result, data); } /// @dev Shorthand for `pBytes3(p(), data)`. function pBytes3(bytes3 data) internal pure returns (DynamicBuffer memory result) { pBytes3(result, data); } /// @dev Shorthand for `pBytes4(p(), data)`. function pBytes4(bytes4 data) internal pure returns (DynamicBuffer memory result) { pBytes4(result, data); } /// @dev Shorthand for `pBytes5(p(), data)`. function pBytes5(bytes5 data) internal pure returns (DynamicBuffer memory result) { pBytes5(result, data); } /// @dev Shorthand for `pBytes6(p(), data)`. function pBytes6(bytes6 data) internal pure returns (DynamicBuffer memory result) { pBytes6(result, data); } /// @dev Shorthand for `pBytes7(p(), data)`. function pBytes7(bytes7 data) internal pure returns (DynamicBuffer memory result) { pBytes7(result, data); } /// @dev Shorthand for `pBytes8(p(), data)`. function pBytes8(bytes8 data) internal pure returns (DynamicBuffer memory result) { pBytes8(result, data); } /// @dev Shorthand for `pBytes9(p(), data)`. function pBytes9(bytes9 data) internal pure returns (DynamicBuffer memory result) { pBytes9(result, data); } /// @dev Shorthand for `pBytes10(p(), data)`. function pBytes10(bytes10 data) internal pure returns (DynamicBuffer memory result) { pBytes10(result, data); } /// @dev Shorthand for `pBytes11(p(), data)`. function pBytes11(bytes11 data) internal pure returns (DynamicBuffer memory result) { pBytes11(result, data); } /// @dev Shorthand for `pBytes12(p(), data)`. function pBytes12(bytes12 data) internal pure returns (DynamicBuffer memory result) { pBytes12(result, data); } /// @dev Shorthand for `pBytes13(p(), data)`. function pBytes13(bytes13 data) internal pure returns (DynamicBuffer memory result) { pBytes13(result, data); } /// @dev Shorthand for `pBytes14(p(), data)`. function pBytes14(bytes14 data) internal pure returns (DynamicBuffer memory result) { pBytes14(result, data); } /// @dev Shorthand for `pBytes15(p(), data)`. function pBytes15(bytes15 data) internal pure returns (DynamicBuffer memory result) { pBytes15(result, data); } /// @dev Shorthand for `pBytes16(p(), data)`. function pBytes16(bytes16 data) internal pure returns (DynamicBuffer memory result) { pBytes16(result, data); } /// @dev Shorthand for `pBytes17(p(), data)`. function pBytes17(bytes17 data) internal pure returns (DynamicBuffer memory result) { pBytes17(result, data); } /// @dev Shorthand for `pBytes18(p(), data)`. function pBytes18(bytes18 data) internal pure returns (DynamicBuffer memory result) { pBytes18(result, data); } /// @dev Shorthand for `pBytes19(p(), data)`. function pBytes19(bytes19 data) internal pure returns (DynamicBuffer memory result) { pBytes19(result, data); } /// @dev Shorthand for `pBytes20(p(), data)`. function pBytes20(bytes20 data) internal pure returns (DynamicBuffer memory result) { pBytes20(result, data); } /// @dev Shorthand for `pBytes21(p(), data)`. function pBytes21(bytes21 data) internal pure returns (DynamicBuffer memory result) { pBytes21(result, data); } /// @dev Shorthand for `pBytes22(p(), data)`. function pBytes22(bytes22 data) internal pure returns (DynamicBuffer memory result) { pBytes22(result, data); } /// @dev Shorthand for `pBytes23(p(), data)`. function pBytes23(bytes23 data) internal pure returns (DynamicBuffer memory result) { pBytes23(result, data); } /// @dev Shorthand for `pBytes24(p(), data)`. function pBytes24(bytes24 data) internal pure returns (DynamicBuffer memory result) { pBytes24(result, data); } /// @dev Shorthand for `pBytes25(p(), data)`. function pBytes25(bytes25 data) internal pure returns (DynamicBuffer memory result) { pBytes25(result, data); } /// @dev Shorthand for `pBytes26(p(), data)`. function pBytes26(bytes26 data) internal pure returns (DynamicBuffer memory result) { pBytes26(result, data); } /// @dev Shorthand for `pBytes27(p(), data)`. function pBytes27(bytes27 data) internal pure returns (DynamicBuffer memory result) { pBytes27(result, data); } /// @dev Shorthand for `pBytes28(p(), data)`. function pBytes28(bytes28 data) internal pure returns (DynamicBuffer memory result) { pBytes28(result, data); } /// @dev Shorthand for `pBytes29(p(), data)`. function pBytes29(bytes29 data) internal pure returns (DynamicBuffer memory result) { pBytes29(result, data); } /// @dev Shorthand for `pBytes30(p(), data)`. function pBytes30(bytes30 data) internal pure returns (DynamicBuffer memory result) { pBytes30(result, data); } /// @dev Shorthand for `pBytes31(p(), data)`. function pBytes31(bytes31 data) internal pure returns (DynamicBuffer memory result) { pBytes31(result, data); } /// @dev Shorthand for `pBytes32(p(), data)`. function pBytes32(bytes32 data) internal pure returns (DynamicBuffer memory result) { pBytes32(result, data); } /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* PRIVATE HELPERS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev Helper for deallocating a automatically allocated `buffer` pointer. function _deallocate(DynamicBuffer memory result) private pure { /// @solidity memory-safe-assembly assembly { mstore(0x40, result) // Deallocate, as we have already allocated. } } /// @dev Returns a temporary bytes string of length `n` for `data`. function _single(uint256 data, uint256 n) private pure returns (bytes memory result) { /// @solidity memory-safe-assembly assembly { result := 0x00 mstore(n, data) mstore(result, n) } } /// @dev Returns a temporary bytes string of length `n` for `data`. function _single(bytes32 data, uint256 n) private pure returns (bytes memory result) { /// @solidity memory-safe-assembly assembly { result := 0x00 mstore(0x20, data) mstore(result, n) } } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.4; /// @notice Arithmetic library with operations for fixed-point numbers. /// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/FixedPointMathLib.sol) /// @author Modified from Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/FixedPointMathLib.sol) library FixedPointMathLib { /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* CUSTOM ERRORS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev The operation failed, as the output exceeds the maximum value of uint256. error ExpOverflow(); /// @dev The operation failed, as the output exceeds the maximum value of uint256. error FactorialOverflow(); /// @dev The operation failed, due to an overflow. error RPowOverflow(); /// @dev The mantissa is too big to fit. error MantissaOverflow(); /// @dev The operation failed, due to an multiplication overflow. error MulWadFailed(); /// @dev The operation failed, due to an multiplication overflow. error SMulWadFailed(); /// @dev The operation failed, either due to a multiplication overflow, or a division by a zero. error DivWadFailed(); /// @dev The operation failed, either due to a multiplication overflow, or a division by a zero. error SDivWadFailed(); /// @dev The operation failed, either due to a multiplication overflow, or a division by a zero. error MulDivFailed(); /// @dev The division failed, as the denominator is zero. error DivFailed(); /// @dev The full precision multiply-divide operation failed, either due /// to the result being larger than 256 bits, or a division by a zero. error FullMulDivFailed(); /// @dev The output is undefined, as the input is less-than-or-equal to zero. error LnWadUndefined(); /// @dev The input outside the acceptable domain. error OutOfDomain(); /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* CONSTANTS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev The scalar of ETH and most ERC20s. uint256 internal constant WAD = 1e18; /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* SIMPLIFIED FIXED POINT OPERATIONS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev Equivalent to `(x * y) / WAD` rounded down. function mulWad(uint256 x, uint256 y) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { // Equivalent to `require(y == 0 || x <= type(uint256).max / y)`. if gt(x, div(not(0), y)) { if y { mstore(0x00, 0xbac65e5b) // `MulWadFailed()`. revert(0x1c, 0x04) } } z := div(mul(x, y), WAD) } } /// @dev Equivalent to `(x * y) / WAD` rounded down. function sMulWad(int256 x, int256 y) internal pure returns (int256 z) { /// @solidity memory-safe-assembly assembly { z := mul(x, y) // Equivalent to `require((x == 0 || z / x == y) && !(x == -1 && y == type(int256).min))`. if iszero(gt(or(iszero(x), eq(sdiv(z, x), y)), lt(not(x), eq(y, shl(255, 1))))) { mstore(0x00, 0xedcd4dd4) // `SMulWadFailed()`. revert(0x1c, 0x04) } z := sdiv(z, WAD) } } /// @dev Equivalent to `(x * y) / WAD` rounded down, but without overflow checks. function rawMulWad(uint256 x, uint256 y) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { z := div(mul(x, y), WAD) } } /// @dev Equivalent to `(x * y) / WAD` rounded down, but without overflow checks. function rawSMulWad(int256 x, int256 y) internal pure returns (int256 z) { /// @solidity memory-safe-assembly assembly { z := sdiv(mul(x, y), WAD) } } /// @dev Equivalent to `(x * y) / WAD` rounded up. function mulWadUp(uint256 x, uint256 y) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { z := mul(x, y) // Equivalent to `require(y == 0 || x <= type(uint256).max / y)`. if iszero(eq(div(z, y), x)) { if y { mstore(0x00, 0xbac65e5b) // `MulWadFailed()`. revert(0x1c, 0x04) } } z := add(iszero(iszero(mod(z, WAD))), div(z, WAD)) } } /// @dev Equivalent to `(x * y) / WAD` rounded up, but without overflow checks. function rawMulWadUp(uint256 x, uint256 y) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { z := add(iszero(iszero(mod(mul(x, y), WAD))), div(mul(x, y), WAD)) } } /// @dev Equivalent to `(x * WAD) / y` rounded down. function divWad(uint256 x, uint256 y) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { // Equivalent to `require(y != 0 && x <= type(uint256).max / WAD)`. if iszero(mul(y, lt(x, add(1, div(not(0), WAD))))) { mstore(0x00, 0x7c5f487d) // `DivWadFailed()`. revert(0x1c, 0x04) } z := div(mul(x, WAD), y) } } /// @dev Equivalent to `(x * WAD) / y` rounded down. function sDivWad(int256 x, int256 y) internal pure returns (int256 z) { /// @solidity memory-safe-assembly assembly { z := mul(x, WAD) // Equivalent to `require(y != 0 && ((x * WAD) / WAD == x))`. if iszero(mul(y, eq(sdiv(z, WAD), x))) { mstore(0x00, 0x5c43740d) // `SDivWadFailed()`. revert(0x1c, 0x04) } z := sdiv(z, y) } } /// @dev Equivalent to `(x * WAD) / y` rounded down, but without overflow and divide by zero checks. function rawDivWad(uint256 x, uint256 y) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { z := div(mul(x, WAD), y) } } /// @dev Equivalent to `(x * WAD) / y` rounded down, but without overflow and divide by zero checks. function rawSDivWad(int256 x, int256 y) internal pure returns (int256 z) { /// @solidity memory-safe-assembly assembly { z := sdiv(mul(x, WAD), y) } } /// @dev Equivalent to `(x * WAD) / y` rounded up. function divWadUp(uint256 x, uint256 y) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { // Equivalent to `require(y != 0 && x <= type(uint256).max / WAD)`. if iszero(mul(y, lt(x, add(1, div(not(0), WAD))))) { mstore(0x00, 0x7c5f487d) // `DivWadFailed()`. revert(0x1c, 0x04) } z := add(iszero(iszero(mod(mul(x, WAD), y))), div(mul(x, WAD), y)) } } /// @dev Equivalent to `(x * WAD) / y` rounded up, but without overflow and divide by zero checks. function rawDivWadUp(uint256 x, uint256 y) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { z := add(iszero(iszero(mod(mul(x, WAD), y))), div(mul(x, WAD), y)) } } /// @dev Equivalent to `x` to the power of `y`. /// because `x ** y = (e ** ln(x)) ** y = e ** (ln(x) * y)`. /// Note: This function is an approximation. function powWad(int256 x, int256 y) internal pure returns (int256) { // Using `ln(x)` means `x` must be greater than 0. return expWad((lnWad(x) * y) / int256(WAD)); } /// @dev Returns `exp(x)`, denominated in `WAD`. /// Credit to Remco Bloemen under MIT license: https://2π.com/22/exp-ln /// Note: This function is an approximation. Monotonically increasing. function expWad(int256 x) internal pure returns (int256 r) { unchecked { // When the result is less than 0.5 we return zero. // This happens when `x <= (log(1e-18) * 1e18) ~ -4.15e19`. if (x <= -41446531673892822313) return r; /// @solidity memory-safe-assembly assembly { // When the result is greater than `(2**255 - 1) / 1e18` we can not represent it as // an int. This happens when `x >= floor(log((2**255 - 1) / 1e18) * 1e18) ≈ 135`. if iszero(slt(x, 135305999368893231589)) { mstore(0x00, 0xa37bfec9) // `ExpOverflow()`. revert(0x1c, 0x04) } } // `x` is now in the range `(-42, 136) * 1e18`. Convert to `(-42, 136) * 2**96` // for more intermediate precision and a binary basis. This base conversion // is a multiplication by 1e18 / 2**96 = 5**18 / 2**78. x = (x << 78) / 5 ** 18; // Reduce range of x to (-½ ln 2, ½ ln 2) * 2**96 by factoring out powers // of two such that exp(x) = exp(x') * 2**k, where k is an integer. // Solving this gives k = round(x / log(2)) and x' = x - k * log(2). int256 k = ((x << 96) / 54916777467707473351141471128 + 2 ** 95) >> 96; x = x - k * 54916777467707473351141471128; // `k` is in the range `[-61, 195]`. // Evaluate using a (6, 7)-term rational approximation. // `p` is made monic, we'll multiply by a scale factor later. int256 y = x + 1346386616545796478920950773328; y = ((y * x) >> 96) + 57155421227552351082224309758442; int256 p = y + x - 94201549194550492254356042504812; p = ((p * y) >> 96) + 28719021644029726153956944680412240; p = p * x + (4385272521454847904659076985693276 << 96); // We leave `p` in `2**192` basis so we don't need to scale it back up for the division. int256 q = x - 2855989394907223263936484059900; q = ((q * x) >> 96) + 50020603652535783019961831881945; q = ((q * x) >> 96) - 533845033583426703283633433725380; q = ((q * x) >> 96) + 3604857256930695427073651918091429; q = ((q * x) >> 96) - 14423608567350463180887372962807573; q = ((q * x) >> 96) + 26449188498355588339934803723976023; /// @solidity memory-safe-assembly assembly { // Div in assembly because solidity adds a zero check despite the unchecked. // The q polynomial won't have zeros in the domain as all its roots are complex. // No scaling is necessary because p is already `2**96` too large. r := sdiv(p, q) } // r should be in the range `(0.09, 0.25) * 2**96`. // We now need to multiply r by: // - The scale factor `s ≈ 6.031367120`. // - The `2**k` factor from the range reduction. // - The `1e18 / 2**96` factor for base conversion. // We do this all at once, with an intermediate result in `2**213` // basis, so the final right shift is always by a positive amount. r = int256( (uint256(r) * 3822833074963236453042738258902158003155416615667) >> uint256(195 - k) ); } } /// @dev Returns `ln(x)`, denominated in `WAD`. /// Credit to Remco Bloemen under MIT license: https://2π.com/22/exp-ln /// Note: This function is an approximation. Monotonically increasing. function lnWad(int256 x) internal pure returns (int256 r) { /// @solidity memory-safe-assembly assembly { // We want to convert `x` from `10**18` fixed point to `2**96` fixed point. // We do this by multiplying by `2**96 / 10**18`. But since // `ln(x * C) = ln(x) + ln(C)`, we can simply do nothing here // and add `ln(2**96 / 10**18)` at the end. // Compute `k = log2(x) - 96`, `r = 159 - k = 255 - log2(x) = 255 ^ log2(x)`. r := shl(7, lt(0xffffffffffffffffffffffffffffffff, x)) r := or(r, shl(6, lt(0xffffffffffffffff, shr(r, x)))) r := or(r, shl(5, lt(0xffffffff, shr(r, x)))) r := or(r, shl(4, lt(0xffff, shr(r, x)))) r := or(r, shl(3, lt(0xff, shr(r, x)))) // We place the check here for more optimal stack operations. if iszero(sgt(x, 0)) { mstore(0x00, 0x1615e638) // `LnWadUndefined()`. revert(0x1c, 0x04) } // forgefmt: disable-next-item r := xor(r, byte(and(0x1f, shr(shr(r, x), 0x8421084210842108cc6318c6db6d54be)), 0xf8f9f9faf9fdfafbf9fdfcfdfafbfcfef9fafdfafcfcfbfefafafcfbffffffff)) // Reduce range of x to (1, 2) * 2**96 // ln(2^k * x) = k * ln(2) + ln(x) x := shr(159, shl(r, x)) // Evaluate using a (8, 8)-term rational approximation. // `p` is made monic, we will multiply by a scale factor later. // forgefmt: disable-next-item let p := sub( // This heavily nested expression is to avoid stack-too-deep for via-ir. sar(96, mul(add(43456485725739037958740375743393, sar(96, mul(add(24828157081833163892658089445524, sar(96, mul(add(3273285459638523848632254066296, x), x))), x))), x)), 11111509109440967052023855526967) p := sub(sar(96, mul(p, x)), 45023709667254063763336534515857) p := sub(sar(96, mul(p, x)), 14706773417378608786704636184526) p := sub(mul(p, x), shl(96, 795164235651350426258249787498)) // We leave `p` in `2**192` basis so we don't need to scale it back up for the division. // `q` is monic by convention. let q := add(5573035233440673466300451813936, x) q := add(71694874799317883764090561454958, sar(96, mul(x, q))) q := add(283447036172924575727196451306956, sar(96, mul(x, q))) q := add(401686690394027663651624208769553, sar(96, mul(x, q))) q := add(204048457590392012362485061816622, sar(96, mul(x, q))) q := add(31853899698501571402653359427138, sar(96, mul(x, q))) q := add(909429971244387300277376558375, sar(96, mul(x, q))) // `p / q` is in the range `(0, 0.125) * 2**96`. // Finalization, we need to: // - Multiply by the scale factor `s = 5.549…`. // - Add `ln(2**96 / 10**18)`. // - Add `k * ln(2)`. // - Multiply by `10**18 / 2**96 = 5**18 >> 78`. // The q polynomial is known not to have zeros in the domain. // No scaling required because p is already `2**96` too large. p := sdiv(p, q) // Multiply by the scaling factor: `s * 5**18 * 2**96`, base is now `5**18 * 2**192`. p := mul(1677202110996718588342820967067443963516166, p) // Add `ln(2) * k * 5**18 * 2**192`. // forgefmt: disable-next-item p := add(mul(16597577552685614221487285958193947469193820559219878177908093499208371, sub(159, r)), p) // Add `ln(2**96 / 10**18) * 5**18 * 2**192`. p := add(600920179829731861736702779321621459595472258049074101567377883020018308, p) // Base conversion: mul `2**18 / 2**192`. r := sar(174, p) } } /// @dev Returns `W_0(x)`, denominated in `WAD`. /// See: https://en.wikipedia.org/wiki/Lambert_W_function /// a.k.a. Product log function. This is an approximation of the principal branch. /// Note: This function is an approximation. Monotonically increasing. function lambertW0Wad(int256 x) internal pure returns (int256 w) { // forgefmt: disable-next-item unchecked { if ((w = x) <= -367879441171442322) revert OutOfDomain(); // `x` less than `-1/e`. (int256 wad, int256 p) = (int256(WAD), x); uint256 c; // Whether we need to avoid catastrophic cancellation. uint256 i = 4; // Number of iterations. if (w <= 0x1ffffffffffff) { if (-0x4000000000000 <= w) { i = 1; // Inputs near zero only take one step to converge. } else if (w <= -0x3ffffffffffffff) { i = 32; // Inputs near `-1/e` take very long to converge. } } else if (uint256(w >> 63) == uint256(0)) { /// @solidity memory-safe-assembly assembly { // Inline log2 for more performance, since the range is small. let v := shr(49, w) let l := shl(3, lt(0xff, v)) l := add(or(l, byte(and(0x1f, shr(shr(l, v), 0x8421084210842108cc6318c6db6d54be)), 0x0706060506020504060203020504030106050205030304010505030400000000)), 49) w := sdiv(shl(l, 7), byte(sub(l, 31), 0x0303030303030303040506080c13)) c := gt(l, 60) i := add(2, add(gt(l, 53), c)) } } else { int256 ll = lnWad(w = lnWad(w)); /// @solidity memory-safe-assembly assembly { // `w = ln(x) - ln(ln(x)) + b * ln(ln(x)) / ln(x)`. w := add(sdiv(mul(ll, 1023715080943847266), w), sub(w, ll)) i := add(3, iszero(shr(68, x))) c := iszero(shr(143, x)) } if (c == uint256(0)) { do { // If `x` is big, use Newton's so that intermediate values won't overflow. int256 e = expWad(w); /// @solidity memory-safe-assembly assembly { let t := mul(w, div(e, wad)) w := sub(w, sdiv(sub(t, x), div(add(e, t), wad))) } if (p <= w) break; p = w; } while (--i != uint256(0)); /// @solidity memory-safe-assembly assembly { w := sub(w, sgt(w, 2)) } return w; } } do { // Otherwise, use Halley's for faster convergence. int256 e = expWad(w); /// @solidity memory-safe-assembly assembly { let t := add(w, wad) let s := sub(mul(w, e), mul(x, wad)) w := sub(w, sdiv(mul(s, wad), sub(mul(e, t), sdiv(mul(add(t, wad), s), add(t, t))))) } if (p <= w) break; p = w; } while (--i != c); /// @solidity memory-safe-assembly assembly { w := sub(w, sgt(w, 2)) } // For certain ranges of `x`, we'll use the quadratic-rate recursive formula of // R. Iacono and J.P. Boyd for the last iteration, to avoid catastrophic cancellation. if (c == uint256(0)) return w; int256 t = w | 1; /// @solidity memory-safe-assembly assembly { x := sdiv(mul(x, wad), t) } x = (t * (wad + lnWad(x))); /// @solidity memory-safe-assembly assembly { w := sdiv(x, add(wad, t)) } } } /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* GENERAL NUMBER UTILITIES */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev Returns `a * b == x * y`, with full precision. function fullMulEq(uint256 a, uint256 b, uint256 x, uint256 y) internal pure returns (bool result) { /// @solidity memory-safe-assembly assembly { result := and(eq(mul(a, b), mul(x, y)), eq(mulmod(x, y, not(0)), mulmod(a, b, not(0)))) } } /// @dev Calculates `floor(x * y / d)` with full precision. /// Throws if result overflows a uint256 or when `d` is zero. /// Credit to Remco Bloemen under MIT license: https://2π.com/21/muldiv function fullMulDiv(uint256 x, uint256 y, uint256 d) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { // 512-bit multiply `[p1 p0] = x * y`. // 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 = p1 * 2**256 + p0`. // Temporarily use `z` as `p0` to save gas. z := mul(x, y) // Lower 256 bits of `x * y`. for {} 1 {} { // If overflows. if iszero(mul(or(iszero(x), eq(div(z, x), y)), d)) { let mm := mulmod(x, y, not(0)) let p1 := sub(mm, add(z, lt(mm, z))) // Upper 256 bits of `x * y`. /*------------------- 512 by 256 division --------------------*/ // Make division exact by subtracting the remainder from `[p1 p0]`. let r := mulmod(x, y, d) // Compute remainder using mulmod. let t := and(d, sub(0, d)) // The least significant bit of `d`. `t >= 1`. // Make sure `z` is less than `2**256`. Also prevents `d == 0`. // Placing the check here seems to give more optimal stack operations. if iszero(gt(d, p1)) { mstore(0x00, 0xae47f702) // `FullMulDivFailed()`. revert(0x1c, 0x04) } d := div(d, t) // Divide `d` by `t`, which is a power of two. // Invert `d mod 2**256` // Now that `d` is an odd number, it has an inverse // modulo `2**256` such that `d * inv = 1 mod 2**256`. // Compute the inverse by starting with a seed that is correct // correct for four bits. That is, `d * inv = 1 mod 2**4`. let inv := xor(2, mul(3, d)) // 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 := mul(inv, sub(2, mul(d, inv))) // inverse mod 2**8 inv := mul(inv, sub(2, mul(d, inv))) // inverse mod 2**16 inv := mul(inv, sub(2, mul(d, inv))) // inverse mod 2**32 inv := mul(inv, sub(2, mul(d, inv))) // inverse mod 2**64 inv := mul(inv, sub(2, mul(d, inv))) // inverse mod 2**128 z := mul( // Divide [p1 p0] by the factors of two. // Shift in bits from `p1` into `p0`. For this we need // to flip `t` such that it is `2**256 / t`. or(mul(sub(p1, gt(r, z)), add(div(sub(0, t), t), 1)), div(sub(z, r), t)), mul(sub(2, mul(d, inv)), inv) // inverse mod 2**256 ) break } z := div(z, d) break } } } /// @dev Calculates `floor(x * y / d)` with full precision. /// Behavior is undefined if `d` is zero or the final result cannot fit in 256 bits. /// Performs the full 512 bit calculation regardless. function fullMulDivUnchecked(uint256 x, uint256 y, uint256 d) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { z := mul(x, y) let mm := mulmod(x, y, not(0)) let p1 := sub(mm, add(z, lt(mm, z))) let t := and(d, sub(0, d)) let r := mulmod(x, y, d) d := div(d, t) let inv := xor(2, mul(3, d)) inv := mul(inv, sub(2, mul(d, inv))) inv := mul(inv, sub(2, mul(d, inv))) inv := mul(inv, sub(2, mul(d, inv))) inv := mul(inv, sub(2, mul(d, inv))) inv := mul(inv, sub(2, mul(d, inv))) z := mul( or(mul(sub(p1, gt(r, z)), add(div(sub(0, t), t), 1)), div(sub(z, r), t)), mul(sub(2, mul(d, inv)), inv) ) } } /// @dev Calculates `floor(x * y / d)` with full precision, rounded up. /// Throws if result overflows a uint256 or when `d` is zero. /// Credit to Uniswap-v3-core under MIT license: /// https://github.com/Uniswap/v3-core/blob/main/contracts/libraries/FullMath.sol function fullMulDivUp(uint256 x, uint256 y, uint256 d) internal pure returns (uint256 z) { z = fullMulDiv(x, y, d); /// @solidity memory-safe-assembly assembly { if mulmod(x, y, d) { z := add(z, 1) if iszero(z) { mstore(0x00, 0xae47f702) // `FullMulDivFailed()`. revert(0x1c, 0x04) } } } } /// @dev Calculates `floor(x * y / 2 ** n)` with full precision. /// Throws if result overflows a uint256. /// Credit to Philogy under MIT license: /// https://github.com/SorellaLabs/angstrom/blob/main/contracts/src/libraries/X128MathLib.sol function fullMulDivN(uint256 x, uint256 y, uint8 n) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { // Temporarily use `z` as `p0` to save gas. z := mul(x, y) // Lower 256 bits of `x * y`. We'll call this `z`. for {} 1 {} { if iszero(or(iszero(x), eq(div(z, x), y))) { let k := and(n, 0xff) // `n`, cleaned. let mm := mulmod(x, y, not(0)) let p1 := sub(mm, add(z, lt(mm, z))) // Upper 256 bits of `x * y`. // | p1 | z | // Before: | p1_0 ¦ p1_1 | z_0 ¦ z_1 | // Final: | 0 ¦ p1_0 | p1_1 ¦ z_0 | // Check that final `z` doesn't overflow by checking that p1_0 = 0. if iszero(shr(k, p1)) { z := add(shl(sub(256, k), p1), shr(k, z)) break } mstore(0x00, 0xae47f702) // `FullMulDivFailed()`. revert(0x1c, 0x04) } z := shr(and(n, 0xff), z) break } } } /// @dev Returns `floor(x * y / d)`. /// Reverts if `x * y` overflows, or `d` is zero. function mulDiv(uint256 x, uint256 y, uint256 d) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { z := mul(x, y) // Equivalent to `require(d != 0 && (y == 0 || x <= type(uint256).max / y))`. if iszero(mul(or(iszero(x), eq(div(z, x), y)), d)) { mstore(0x00, 0xad251c27) // `MulDivFailed()`. revert(0x1c, 0x04) } z := div(z, d) } } /// @dev Returns `ceil(x * y / d)`. /// Reverts if `x * y` overflows, or `d` is zero. function mulDivUp(uint256 x, uint256 y, uint256 d) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { z := mul(x, y) // Equivalent to `require(d != 0 && (y == 0 || x <= type(uint256).max / y))`. if iszero(mul(or(iszero(x), eq(div(z, x), y)), d)) { mstore(0x00, 0xad251c27) // `MulDivFailed()`. revert(0x1c, 0x04) } z := add(iszero(iszero(mod(z, d))), div(z, d)) } } /// @dev Returns `x`, the modular multiplicative inverse of `a`, such that `(a * x) % n == 1`. function invMod(uint256 a, uint256 n) internal pure returns (uint256 x) { /// @solidity memory-safe-assembly assembly { let g := n let r := mod(a, n) for { let y := 1 } 1 {} { let q := div(g, r) let t := g g := r r := sub(t, mul(r, q)) let u := x x := y y := sub(u, mul(y, q)) if iszero(r) { break } } x := mul(eq(g, 1), add(x, mul(slt(x, 0), n))) } } /// @dev Returns `ceil(x / d)`. /// Reverts if `d` is zero. function divUp(uint256 x, uint256 d) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { if iszero(d) { mstore(0x00, 0x65244e4e) // `DivFailed()`. revert(0x1c, 0x04) } z := add(iszero(iszero(mod(x, d))), div(x, d)) } } /// @dev Returns `max(0, x - y)`. function zeroFloorSub(uint256 x, uint256 y) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { z := mul(gt(x, y), sub(x, y)) } } /// @dev Returns `condition ? x : y`, without branching. function ternary(bool condition, uint256 x, uint256 y) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { z := xor(x, mul(xor(x, y), iszero(condition))) } } /// @dev Exponentiate `x` to `y` by squaring, denominated in base `b`. /// Reverts if the computation overflows. function rpow(uint256 x, uint256 y, uint256 b) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { z := mul(b, iszero(y)) // `0 ** 0 = 1`. Otherwise, `0 ** n = 0`. if x { z := xor(b, mul(xor(b, x), and(y, 1))) // `z = isEven(y) ? scale : x` let half := shr(1, b) // Divide `b` by 2. // Divide `y` by 2 every iteration. for { y := shr(1, y) } y { y := shr(1, y) } { let xx := mul(x, x) // Store x squared. let xxRound := add(xx, half) // Round to the nearest number. // Revert if `xx + half` overflowed, or if `x ** 2` overflows. if or(lt(xxRound, xx), shr(128, x)) { mstore(0x00, 0x49f7642b) // `RPowOverflow()`. revert(0x1c, 0x04) } x := div(xxRound, b) // Set `x` to scaled `xxRound`. // If `y` is odd: if and(y, 1) { let zx := mul(z, x) // Compute `z * x`. let zxRound := add(zx, half) // Round to the nearest number. // If `z * x` overflowed or `zx + half` overflowed: if or(xor(div(zx, x), z), lt(zxRound, zx)) { // Revert if `x` is non-zero. if x { mstore(0x00, 0x49f7642b) // `RPowOverflow()`. revert(0x1c, 0x04) } } z := div(zxRound, b) // Return properly scaled `zxRound`. } } } } } /// @dev Returns the square root of `x`, rounded down. function sqrt(uint256 x) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { // `floor(sqrt(2**15)) = 181`. `sqrt(2**15) - 181 = 2.84`. z := 181 // The "correct" value is 1, but this saves a multiplication later. // This segment is to get a reasonable initial estimate for the Babylonian method. With a bad // start, the correct # of bits increases ~linearly each iteration instead of ~quadratically. // Let `y = x / 2**r`. We check `y >= 2**(k + 8)` // but shift right by `k` bits to ensure that if `x >= 256`, then `y >= 256`. let r := shl(7, lt(0xffffffffffffffffffffffffffffffffff, x)) r := or(r, shl(6, lt(0xffffffffffffffffff, shr(r, x)))) r := or(r, shl(5, lt(0xffffffffff, shr(r, x)))) r := or(r, shl(4, lt(0xffffff, shr(r, x)))) z := shl(shr(1, r), z) // Goal was to get `z*z*y` within a small factor of `x`. More iterations could // get y in a tighter range. Currently, we will have y in `[256, 256*(2**16))`. // We ensured `y >= 256` so that the relative difference between `y` and `y+1` is small. // That's not possible if `x < 256` but we can just verify those cases exhaustively. // Now, `z*z*y <= x < z*z*(y+1)`, and `y <= 2**(16+8)`, and either `y >= 256`, or `x < 256`. // Correctness can be checked exhaustively for `x < 256`, so we assume `y >= 256`. // Then `z*sqrt(y)` is within `sqrt(257)/sqrt(256)` of `sqrt(x)`, or about 20bps. // For `s` in the range `[1/256, 256]`, the estimate `f(s) = (181/1024) * (s+1)` // is in the range `(1/2.84 * sqrt(s), 2.84 * sqrt(s))`, // with largest error when `s = 1` and when `s = 256` or `1/256`. // Since `y` is in `[256, 256*(2**16))`, let `a = y/65536`, so that `a` is in `[1/256, 256)`. // Then we can estimate `sqrt(y)` using // `sqrt(65536) * 181/1024 * (a + 1) = 181/4 * (y + 65536)/65536 = 181 * (y + 65536)/2**18`. // There is no overflow risk here since `y < 2**136` after the first branch above. z := shr(18, mul(z, add(shr(r, x), 65536))) // A `mul()` is saved from starting `z` at 181. // Given the worst case multiplicative error of 2.84 above, 7 iterations should be enough. z := shr(1, add(z, div(x, z))) z := shr(1, add(z, div(x, z))) z := shr(1, add(z, div(x, z))) z := shr(1, add(z, div(x, z))) z := shr(1, add(z, div(x, z))) z := shr(1, add(z, div(x, z))) z := shr(1, add(z, div(x, z))) // If `x+1` is a perfect square, the Babylonian method cycles between // `floor(sqrt(x))` and `ceil(sqrt(x))`. This statement ensures we return floor. // See: https://en.wikipedia.org/wiki/Integer_square_root#Using_only_integer_division z := sub(z, lt(div(x, z), z)) } } /// @dev Returns the cube root of `x`, rounded down. /// Credit to bout3fiddy and pcaversaccio under AGPLv3 license: /// https://github.com/pcaversaccio/snekmate/blob/main/src/utils/Math.vy /// Formally verified by xuwinnie: /// https://github.com/vectorized/solady/blob/main/audits/xuwinnie-solady-cbrt-proof.pdf function cbrt(uint256 x) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { let r := shl(7, lt(0xffffffffffffffffffffffffffffffff, x)) r := or(r, shl(6, lt(0xffffffffffffffff, shr(r, x)))) r := or(r, shl(5, lt(0xffffffff, shr(r, x)))) r := or(r, shl(4, lt(0xffff, shr(r, x)))) r := or(r, shl(3, lt(0xff, shr(r, x)))) // Makeshift lookup table to nudge the approximate log2 result. z := div(shl(div(r, 3), shl(lt(0xf, shr(r, x)), 0xf)), xor(7, mod(r, 3))) // Newton-Raphson's. z := div(add(add(div(x, mul(z, z)), z), z), 3) z := div(add(add(div(x, mul(z, z)), z), z), 3) z := div(add(add(div(x, mul(z, z)), z), z), 3) z := div(add(add(div(x, mul(z, z)), z), z), 3) z := div(add(add(div(x, mul(z, z)), z), z), 3) z := div(add(add(div(x, mul(z, z)), z), z), 3) z := div(add(add(div(x, mul(z, z)), z), z), 3) // Round down. z := sub(z, lt(div(x, mul(z, z)), z)) } } /// @dev Returns the square root of `x`, denominated in `WAD`, rounded down. function sqrtWad(uint256 x) internal pure returns (uint256 z) { unchecked { if (x <= type(uint256).max / 10 ** 18) return sqrt(x * 10 ** 18); z = (1 + sqrt(x)) * 10 ** 9; z = (fullMulDivUnchecked(x, 10 ** 18, z) + z) >> 1; } /// @solidity memory-safe-assembly assembly { z := sub(z, gt(999999999999999999, sub(mulmod(z, z, x), 1))) // Round down. } } /// @dev Returns the cube root of `x`, denominated in `WAD`, rounded down. /// Formally verified by xuwinnie: /// https://github.com/vectorized/solady/blob/main/audits/xuwinnie-solady-cbrt-proof.pdf function cbrtWad(uint256 x) internal pure returns (uint256 z) { unchecked { if (x <= type(uint256).max / 10 ** 36) return cbrt(x * 10 ** 36); z = (1 + cbrt(x)) * 10 ** 12; z = (fullMulDivUnchecked(x, 10 ** 36, z * z) + z + z) / 3; } /// @solidity memory-safe-assembly assembly { let p := x for {} 1 {} { if iszero(shr(229, p)) { if iszero(shr(199, p)) { p := mul(p, 100000000000000000) // 10 ** 17. break } p := mul(p, 100000000) // 10 ** 8. break } if iszero(shr(249, p)) { p := mul(p, 100) } break } let t := mulmod(mul(z, z), z, p) z := sub(z, gt(lt(t, shr(1, p)), iszero(t))) // Round down. } } /// @dev Returns the factorial of `x`. function factorial(uint256 x) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { z := 1 if iszero(lt(x, 58)) { mstore(0x00, 0xaba0f2a2) // `FactorialOverflow()`. revert(0x1c, 0x04) } for {} x { x := sub(x, 1) } { z := mul(z, x) } } } /// @dev Returns the log2 of `x`. /// Equivalent to computing the index of the most significant bit (MSB) of `x`. /// Returns 0 if `x` is zero. function log2(uint256 x) internal pure returns (uint256 r) { /// @solidity memory-safe-assembly assembly { r := shl(7, lt(0xffffffffffffffffffffffffffffffff, x)) r := or(r, shl(6, lt(0xffffffffffffffff, shr(r, x)))) r := or(r, shl(5, lt(0xffffffff, shr(r, x)))) r := or(r, shl(4, lt(0xffff, shr(r, x)))) r := or(r, shl(3, lt(0xff, shr(r, x)))) // forgefmt: disable-next-item r := or(r, byte(and(0x1f, shr(shr(r, x), 0x8421084210842108cc6318c6db6d54be)), 0x0706060506020504060203020504030106050205030304010505030400000000)) } } /// @dev Returns the log2 of `x`, rounded up. /// Returns 0 if `x` is zero. function log2Up(uint256 x) internal pure returns (uint256 r) { r = log2(x); /// @solidity memory-safe-assembly assembly { r := add(r, lt(shl(r, 1), x)) } } /// @dev Returns the log10 of `x`. /// Returns 0 if `x` is zero. function log10(uint256 x) internal pure returns (uint256 r) { /// @solidity memory-safe-assembly assembly { if iszero(lt(x, 100000000000000000000000000000000000000)) { x := div(x, 100000000000000000000000000000000000000) r := 38 } if iszero(lt(x, 100000000000000000000)) { x := div(x, 100000000000000000000) r := add(r, 20) } if iszero(lt(x, 10000000000)) { x := div(x, 10000000000) r := add(r, 10) } if iszero(lt(x, 100000)) { x := div(x, 100000) r := add(r, 5) } r := add(r, add(gt(x, 9), add(gt(x, 99), add(gt(x, 999), gt(x, 9999))))) } } /// @dev Returns the log10 of `x`, rounded up. /// Returns 0 if `x` is zero. function log10Up(uint256 x) internal pure returns (uint256 r) { r = log10(x); /// @solidity memory-safe-assembly assembly { r := add(r, lt(exp(10, r), x)) } } /// @dev Returns the log256 of `x`. /// Returns 0 if `x` is zero. function log256(uint256 x) internal pure returns (uint256 r) { /// @solidity memory-safe-assembly assembly { r := shl(7, lt(0xffffffffffffffffffffffffffffffff, x)) r := or(r, shl(6, lt(0xffffffffffffffff, shr(r, x)))) r := or(r, shl(5, lt(0xffffffff, shr(r, x)))) r := or(r, shl(4, lt(0xffff, shr(r, x)))) r := or(shr(3, r), lt(0xff, shr(r, x))) } } /// @dev Returns the log256 of `x`, rounded up. /// Returns 0 if `x` is zero. function log256Up(uint256 x) internal pure returns (uint256 r) { r = log256(x); /// @solidity memory-safe-assembly assembly { r := add(r, lt(shl(shl(3, r), 1), x)) } } /// @dev Returns the scientific notation format `mantissa * 10 ** exponent` of `x`. /// Useful for compressing prices (e.g. using 25 bit mantissa and 7 bit exponent). function sci(uint256 x) internal pure returns (uint256 mantissa, uint256 exponent) { /// @solidity memory-safe-assembly assembly { mantissa := x if mantissa { if iszero(mod(mantissa, 1000000000000000000000000000000000)) { mantissa := div(mantissa, 1000000000000000000000000000000000) exponent := 33 } if iszero(mod(mantissa, 10000000000000000000)) { mantissa := div(mantissa, 10000000000000000000) exponent := add(exponent, 19) } if iszero(mod(mantissa, 1000000000000)) { mantissa := div(mantissa, 1000000000000) exponent := add(exponent, 12) } if iszero(mod(mantissa, 1000000)) { mantissa := div(mantissa, 1000000) exponent := add(exponent, 6) } if iszero(mod(mantissa, 10000)) { mantissa := div(mantissa, 10000) exponent := add(exponent, 4) } if iszero(mod(mantissa, 100)) { mantissa := div(mantissa, 100) exponent := add(exponent, 2) } if iszero(mod(mantissa, 10)) { mantissa := div(mantissa, 10) exponent := add(exponent, 1) } } } } /// @dev Convenience function for packing `x` into a smaller number using `sci`. /// The `mantissa` will be in bits [7..255] (the upper 249 bits). /// The `exponent` will be in bits [0..6] (the lower 7 bits). /// Use `SafeCastLib` to safely ensure that the `packed` number is small /// enough to fit in the desired unsigned integer type: /// ``` /// uint32 packed = SafeCastLib.toUint32(FixedPointMathLib.packSci(777 ether)); /// ``` function packSci(uint256 x) internal pure returns (uint256 packed) { (x, packed) = sci(x); // Reuse for `mantissa` and `exponent`. /// @solidity memory-safe-assembly assembly { if shr(249, x) { mstore(0x00, 0xce30380c) // `MantissaOverflow()`. revert(0x1c, 0x04) } packed := or(shl(7, x), packed) } } /// @dev Convenience function for unpacking a packed number from `packSci`. function unpackSci(uint256 packed) internal pure returns (uint256 unpacked) { unchecked { unpacked = (packed >> 7) * 10 ** (packed & 0x7f); } } /// @dev Returns the average of `x` and `y`. Rounds towards zero. function avg(uint256 x, uint256 y) internal pure returns (uint256 z) { unchecked { z = (x & y) + ((x ^ y) >> 1); } } /// @dev Returns the average of `x` and `y`. Rounds towards negative infinity. function avg(int256 x, int256 y) internal pure returns (int256 z) { unchecked { z = (x >> 1) + (y >> 1) + (x & y & 1); } } /// @dev Returns the absolute value of `x`. function abs(int256 x) internal pure returns (uint256 z) { unchecked { z = (uint256(x) + uint256(x >> 255)) ^ uint256(x >> 255); } } /// @dev Returns the absolute distance between `x` and `y`. function dist(uint256 x, uint256 y) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { z := add(xor(sub(0, gt(x, y)), sub(y, x)), gt(x, y)) } } /// @dev Returns the absolute distance between `x` and `y`. function dist(int256 x, int256 y) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { z := add(xor(sub(0, sgt(x, y)), sub(y, x)), sgt(x, y)) } } /// @dev Returns the minimum of `x` and `y`. function min(uint256 x, uint256 y) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { z := xor(x, mul(xor(x, y), lt(y, x))) } } /// @dev Returns the minimum of `x` and `y`. function min(int256 x, int256 y) internal pure returns (int256 z) { /// @solidity memory-safe-assembly assembly { z := xor(x, mul(xor(x, y), slt(y, x))) } } /// @dev Returns the maximum of `x` and `y`. function max(uint256 x, uint256 y) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { z := xor(x, mul(xor(x, y), gt(y, x))) } } /// @dev Returns the maximum of `x` and `y`. function max(int256 x, int256 y) internal pure returns (int256 z) { /// @solidity memory-safe-assembly assembly { z := xor(x, mul(xor(x, y), sgt(y, x))) } } /// @dev Returns `x`, bounded to `minValue` and `maxValue`. function clamp(uint256 x, uint256 minValue, uint256 maxValue) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { z := xor(x, mul(xor(x, minValue), gt(minValue, x))) z := xor(z, mul(xor(z, maxValue), lt(maxValue, z))) } } /// @dev Returns `x`, bounded to `minValue` and `maxValue`. function clamp(int256 x, int256 minValue, int256 maxValue) internal pure returns (int256 z) { /// @solidity memory-safe-assembly assembly { z := xor(x, mul(xor(x, minValue), sgt(minValue, x))) z := xor(z, mul(xor(z, maxValue), slt(maxValue, z))) } } /// @dev Returns greatest common divisor of `x` and `y`. function gcd(uint256 x, uint256 y) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { for { z := x } y {} { let t := y y := mod(z, y) z := t } } } /// @dev Returns `a + (b - a) * (t - begin) / (end - begin)`, /// with `t` clamped between `begin` and `end` (inclusive). /// Agnostic to the order of (`a`, `b`) and (`end`, `begin`). /// If `begins == end`, returns `t <= begin ? a : b`. function lerp(uint256 a, uint256 b, uint256 t, uint256 begin, uint256 end) internal pure returns (uint256) { if (begin > end) (t, begin, end) = (~t, ~begin, ~end); if (t <= begin) return a; if (t >= end) return b; unchecked { if (b >= a) return a + fullMulDiv(b - a, t - begin, end - begin); return a - fullMulDiv(a - b, t - begin, end - begin); } } /// @dev Returns `a + (b - a) * (t - begin) / (end - begin)`. /// with `t` clamped between `begin` and `end` (inclusive). /// Agnostic to the order of (`a`, `b`) and (`end`, `begin`). /// If `begins == end`, returns `t <= begin ? a : b`. function lerp(int256 a, int256 b, int256 t, int256 begin, int256 end) internal pure returns (int256) { if (begin > end) (t, begin, end) = (~t, ~begin, ~end); if (t <= begin) return a; if (t >= end) return b; // forgefmt: disable-next-item unchecked { if (b >= a) return int256(uint256(a) + fullMulDiv(uint256(b - a), uint256(t - begin), uint256(end - begin))); return int256(uint256(a) - fullMulDiv(uint256(a - b), uint256(t - begin), uint256(end - begin))); } } /// @dev Returns if `x` is an even number. Some people may need this. function isEven(uint256 x) internal pure returns (bool) { return x & uint256(1) == uint256(0); } /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* RAW NUMBER OPERATIONS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev Returns `x + y`, without checking for overflow. function rawAdd(uint256 x, uint256 y) internal pure returns (uint256 z) { unchecked { z = x + y; } } /// @dev Returns `x + y`, without checking for overflow. function rawAdd(int256 x, int256 y) internal pure returns (int256 z) { unchecked { z = x + y; } } /// @dev Returns `x - y`, without checking for underflow. function rawSub(uint256 x, uint256 y) internal pure returns (uint256 z) { unchecked { z = x - y; } } /// @dev Returns `x - y`, without checking for underflow. function rawSub(int256 x, int256 y) internal pure returns (int256 z) { unchecked { z = x - y; } } /// @dev Returns `x * y`, without checking for overflow. function rawMul(uint256 x, uint256 y) internal pure returns (uint256 z) { unchecked { z = x * y; } } /// @dev Returns `x * y`, without checking for overflow. function rawMul(int256 x, int256 y) internal pure returns (int256 z) { unchecked { z = x * y; } } /// @dev Returns `x / y`, returning 0 if `y` is zero. function rawDiv(uint256 x, uint256 y) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { z := div(x, y) } } /// @dev Returns `x / y`, returning 0 if `y` is zero. function rawSDiv(int256 x, int256 y) internal pure returns (int256 z) { /// @solidity memory-safe-assembly assembly { z := sdiv(x, y) } } /// @dev Returns `x % y`, returning 0 if `y` is zero. function rawMod(uint256 x, uint256 y) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { z := mod(x, y) } } /// @dev Returns `x % y`, returning 0 if `y` is zero. function rawSMod(int256 x, int256 y) internal pure returns (int256 z) { /// @solidity memory-safe-assembly assembly { z := smod(x, y) } } /// @dev Returns `(x + y) % d`, return 0 if `d` if zero. function rawAddMod(uint256 x, uint256 y, uint256 d) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { z := addmod(x, y, d) } } /// @dev Returns `(x * y) % d`, return 0 if `d` if zero. function rawMulMod(uint256 x, uint256 y, uint256 d) internal pure returns (uint256 z) { /// @solidity memory-safe-assembly assembly { z := mulmod(x, y, d) } } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.4; /// @notice Library for byte related operations. /// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/LibBytes.sol) library LibBytes { /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* STRUCTS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev Goated bytes storage struct that totally MOGs, no cap, fr. /// Uses less gas and bytecode than Solidity's native bytes storage. It's meta af. /// Packs length with the first 31 bytes if <255 bytes, so it’s mad tight. struct BytesStorage { bytes32 _spacer; } /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* CONSTANTS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev The constant returned when the `search` is not found in the bytes. uint256 internal constant NOT_FOUND = type(uint256).max; /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* BYTE STORAGE OPERATIONS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev Sets the value of the bytes storage `$` to `s`. function set(BytesStorage storage $, bytes memory s) internal { /// @solidity memory-safe-assembly assembly { let n := mload(s) let packed := or(0xff, shl(8, n)) for { let i := 0 } 1 {} { if iszero(gt(n, 0xfe)) { i := 0x1f packed := or(n, shl(8, mload(add(s, i)))) if iszero(gt(n, i)) { break } } let o := add(s, 0x20) mstore(0x00, $.slot) for { let p := keccak256(0x00, 0x20) } 1 {} { sstore(add(p, shr(5, i)), mload(add(o, i))) i := add(i, 0x20) if iszero(lt(i, n)) { break } } break } sstore($.slot, packed) } } /// @dev Sets the value of the bytes storage `$` to `s`. function setCalldata(BytesStorage storage $, bytes calldata s) internal { /// @solidity memory-safe-assembly assembly { let packed := or(0xff, shl(8, s.length)) for { let i := 0 } 1 {} { if iszero(gt(s.length, 0xfe)) { i := 0x1f packed := or(s.length, shl(8, shr(8, calldataload(s.offset)))) if iszero(gt(s.length, i)) { break } } mstore(0x00, $.slot) for { let p := keccak256(0x00, 0x20) } 1 {} { sstore(add(p, shr(5, i)), calldataload(add(s.offset, i))) i := add(i, 0x20) if iszero(lt(i, s.length)) { break } } break } sstore($.slot, packed) } } /// @dev Sets the value of the bytes storage `$` to the empty bytes. function clear(BytesStorage storage $) internal { delete $._spacer; } /// @dev Returns whether the value stored is `$` is the empty bytes "". function isEmpty(BytesStorage storage $) internal view returns (bool) { return uint256($._spacer) & 0xff == uint256(0); } /// @dev Returns the length of the value stored in `$`. function length(BytesStorage storage $) internal view returns (uint256 result) { result = uint256($._spacer); /// @solidity memory-safe-assembly assembly { let n := and(0xff, result) result := or(mul(shr(8, result), eq(0xff, n)), mul(n, iszero(eq(0xff, n)))) } } /// @dev Returns the value stored in `$`. function get(BytesStorage storage $) internal view returns (bytes memory result) { /// @solidity memory-safe-assembly assembly { result := mload(0x40) let o := add(result, 0x20) let packed := sload($.slot) let n := shr(8, packed) for { let i := 0 } 1 {} { if iszero(eq(and(packed, 0xff), 0xff)) { mstore(o, packed) n := and(0xff, packed) i := 0x1f if iszero(gt(n, i)) { break } } mstore(0x00, $.slot) for { let p := keccak256(0x00, 0x20) } 1 {} { mstore(add(o, i), sload(add(p, shr(5, i)))) i := add(i, 0x20) if iszero(lt(i, n)) { break } } break } mstore(result, n) // Store the length of the memory. mstore(add(o, n), 0) // Zeroize the slot after the bytes. mstore(0x40, add(add(o, n), 0x20)) // Allocate memory. } } /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* BYTES OPERATIONS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev Returns `subject` all occurrences of `needle` replaced with `replacement`. function replace(bytes memory subject, bytes memory needle, bytes memory replacement) internal pure returns (bytes memory result) { /// @solidity memory-safe-assembly assembly { result := mload(0x40) let needleLen := mload(needle) let replacementLen := mload(replacement) let d := sub(result, subject) // Memory difference. let i := add(subject, 0x20) // Subject bytes pointer. mstore(0x00, add(i, mload(subject))) // End of subject. if iszero(gt(needleLen, mload(subject))) { let subjectSearchEnd := add(sub(mload(0x00), needleLen), 1) let h := 0 // The hash of `needle`. if iszero(lt(needleLen, 0x20)) { h := keccak256(add(needle, 0x20), needleLen) } let s := mload(add(needle, 0x20)) for { let m := shl(3, sub(0x20, and(needleLen, 0x1f))) } 1 {} { let t := mload(i) // Whether the first `needleLen % 32` bytes of `subject` and `needle` matches. if iszero(shr(m, xor(t, s))) { if h { if iszero(eq(keccak256(i, needleLen), h)) { mstore(add(i, d), t) i := add(i, 1) if iszero(lt(i, subjectSearchEnd)) { break } continue } } // Copy the `replacement` one word at a time. for { let j := 0 } 1 {} { mstore(add(add(i, d), j), mload(add(add(replacement, 0x20), j))) j := add(j, 0x20) if iszero(lt(j, replacementLen)) { break } } d := sub(add(d, replacementLen), needleLen) if needleLen { i := add(i, needleLen) if iszero(lt(i, subjectSearchEnd)) { break } continue } } mstore(add(i, d), t) i := add(i, 1) if iszero(lt(i, subjectSearchEnd)) { break } } } let end := mload(0x00) let n := add(sub(d, add(result, 0x20)), end) // Copy the rest of the bytes one word at a time. for {} lt(i, end) { i := add(i, 0x20) } { mstore(add(i, d), mload(i)) } let o := add(i, d) mstore(o, 0) // Zeroize the slot after the bytes. mstore(0x40, add(o, 0x20)) // Allocate memory. mstore(result, n) // Store the length. } } /// @dev Returns the byte index of the first location of `needle` in `subject`, /// needleing from left to right, starting from `from`. /// Returns `NOT_FOUND` (i.e. `type(uint256).max`) if the `needle` is not found. function indexOf(bytes memory subject, bytes memory needle, uint256 from) internal pure returns (uint256 result) { /// @solidity memory-safe-assembly assembly { result := not(0) // Initialize to `NOT_FOUND`. for { let subjectLen := mload(subject) } 1 {} { if iszero(mload(needle)) { result := from if iszero(gt(from, subjectLen)) { break } result := subjectLen break } let needleLen := mload(needle) let subjectStart := add(subject, 0x20) subject := add(subjectStart, from) let end := add(sub(add(subjectStart, subjectLen), needleLen), 1) let m := shl(3, sub(0x20, and(needleLen, 0x1f))) let s := mload(add(needle, 0x20)) if iszero(and(lt(subject, end), lt(from, subjectLen))) { break } if iszero(lt(needleLen, 0x20)) { for { let h := keccak256(add(needle, 0x20), needleLen) } 1 {} { if iszero(shr(m, xor(mload(subject), s))) { if eq(keccak256(subject, needleLen), h) { result := sub(subject, subjectStart) break } } subject := add(subject, 1) if iszero(lt(subject, end)) { break } } break } for {} 1 {} { if iszero(shr(m, xor(mload(subject), s))) { result := sub(subject, subjectStart) break } subject := add(subject, 1) if iszero(lt(subject, end)) { break } } break } } } /// @dev Returns the byte index of the first location of `needle` in `subject`, /// needleing from left to right. /// Returns `NOT_FOUND` (i.e. `type(uint256).max`) if the `needle` is not found. function indexOf(bytes memory subject, bytes memory needle) internal pure returns (uint256) { return indexOf(subject, needle, 0); } /// @dev Returns the byte index of the first location of `needle` in `subject`, /// needleing from right to left, starting from `from`. /// Returns `NOT_FOUND` (i.e. `type(uint256).max`) if the `needle` is not found. function lastIndexOf(bytes memory subject, bytes memory needle, uint256 from) internal pure returns (uint256 result) { /// @solidity memory-safe-assembly assembly { for {} 1 {} { result := not(0) // Initialize to `NOT_FOUND`. let needleLen := mload(needle) if gt(needleLen, mload(subject)) { break } let w := result let fromMax := sub(mload(subject), needleLen) if iszero(gt(fromMax, from)) { from := fromMax } let end := add(add(subject, 0x20), w) subject := add(add(subject, 0x20), from) if iszero(gt(subject, end)) { break } // As this function is not too often used, // we shall simply use keccak256 for smaller bytecode size. for { let h := keccak256(add(needle, 0x20), needleLen) } 1 {} { if eq(keccak256(subject, needleLen), h) { result := sub(subject, add(end, 1)) break } subject := add(subject, w) // `sub(subject, 1)`. if iszero(gt(subject, end)) { break } } break } } } /// @dev Returns the byte index of the first location of `needle` in `subject`, /// needleing from right to left. /// Returns `NOT_FOUND` (i.e. `type(uint256).max`) if the `needle` is not found. function lastIndexOf(bytes memory subject, bytes memory needle) internal pure returns (uint256) { return lastIndexOf(subject, needle, type(uint256).max); } /// @dev Returns true if `needle` is found in `subject`, false otherwise. function contains(bytes memory subject, bytes memory needle) internal pure returns (bool) { return indexOf(subject, needle) != NOT_FOUND; } /// @dev Returns whether `subject` starts with `needle`. function startsWith(bytes memory subject, bytes memory needle) internal pure returns (bool result) { /// @solidity memory-safe-assembly assembly { let n := mload(needle) // Just using keccak256 directly is actually cheaper. let t := eq(keccak256(add(subject, 0x20), n), keccak256(add(needle, 0x20), n)) result := lt(gt(n, mload(subject)), t) } } /// @dev Returns whether `subject` ends with `needle`. function endsWith(bytes memory subject, bytes memory needle) internal pure returns (bool result) { /// @solidity memory-safe-assembly assembly { let n := mload(needle) let notInRange := gt(n, mload(subject)) // `subject + 0x20 + max(subject.length - needle.length, 0)`. let t := add(add(subject, 0x20), mul(iszero(notInRange), sub(mload(subject), n))) // Just using keccak256 directly is actually cheaper. result := gt(eq(keccak256(t, n), keccak256(add(needle, 0x20), n)), notInRange) } } /// @dev Returns `subject` repeated `times`. function repeat(bytes memory subject, uint256 times) internal pure returns (bytes memory result) { /// @solidity memory-safe-assembly assembly { let l := mload(subject) // Subject length. if iszero(or(iszero(times), iszero(l))) { result := mload(0x40) subject := add(subject, 0x20) let o := add(result, 0x20) for {} 1 {} { // Copy the `subject` one word at a time. for { let j := 0 } 1 {} { mstore(add(o, j), mload(add(subject, j))) j := add(j, 0x20) if iszero(lt(j, l)) { break } } o := add(o, l) times := sub(times, 1) if iszero(times) { break } } mstore(o, 0) // Zeroize the slot after the bytes. mstore(0x40, add(o, 0x20)) // Allocate memory. mstore(result, sub(o, add(result, 0x20))) // Store the length. } } } /// @dev Returns a copy of `subject` sliced from `start` to `end` (exclusive). /// `start` and `end` are byte offsets. function slice(bytes memory subject, uint256 start, uint256 end) internal pure returns (bytes memory result) { /// @solidity memory-safe-assembly assembly { let l := mload(subject) // Subject length. if iszero(gt(l, end)) { end := l } if iszero(gt(l, start)) { start := l } if lt(start, end) { result := mload(0x40) let n := sub(end, start) let i := add(subject, start) let w := not(0x1f) // Copy the `subject` one word at a time, backwards. for { let j := and(add(n, 0x1f), w) } 1 {} { mstore(add(result, j), mload(add(i, j))) j := add(j, w) // `sub(j, 0x20)`. if iszero(j) { break } } let o := add(add(result, 0x20), n) mstore(o, 0) // Zeroize the slot after the bytes. mstore(0x40, add(o, 0x20)) // Allocate memory. mstore(result, n) // Store the length. } } } /// @dev Returns a copy of `subject` sliced from `start` to the end of the bytes. /// `start` is a byte offset. function slice(bytes memory subject, uint256 start) internal pure returns (bytes memory result) { result = slice(subject, start, type(uint256).max); } /// @dev Returns a copy of `subject` sliced from `start` to `end` (exclusive). /// `start` and `end` are byte offsets. Faster than Solidity's native slicing. function sliceCalldata(bytes calldata subject, uint256 start, uint256 end) internal pure returns (bytes calldata result) { /// @solidity memory-safe-assembly assembly { end := xor(end, mul(xor(end, subject.length), lt(subject.length, end))) start := xor(start, mul(xor(start, subject.length), lt(subject.length, start))) result.offset := add(subject.offset, start) result.length := mul(lt(start, end), sub(end, start)) } } /// @dev Returns a copy of `subject` sliced from `start` to the end of the bytes. /// `start` is a byte offset. Faster than Solidity's native slicing. function sliceCalldata(bytes calldata subject, uint256 start) internal pure returns (bytes calldata result) { /// @solidity memory-safe-assembly assembly { start := xor(start, mul(xor(start, subject.length), lt(subject.length, start))) result.offset := add(subject.offset, start) result.length := mul(lt(start, subject.length), sub(subject.length, start)) } } /// @dev Reduces the size of `subject` to `n`. /// If `n` is greater than the size of `subject`, this will be a no-op. function truncate(bytes memory subject, uint256 n) internal pure returns (bytes memory result) { /// @solidity memory-safe-assembly assembly { result := subject mstore(mul(lt(n, mload(result)), result), n) } } /// @dev Returns a copy of `subject`, with the length reduced to `n`. /// If `n` is greater than the size of `subject`, this will be a no-op. function truncatedCalldata(bytes calldata subject, uint256 n) internal pure returns (bytes calldata result) { /// @solidity memory-safe-assembly assembly { result.offset := subject.offset result.length := xor(n, mul(xor(n, subject.length), lt(subject.length, n))) } } /// @dev Returns all the indices of `needle` in `subject`. /// The indices are byte offsets. function indicesOf(bytes memory subject, bytes memory needle) internal pure returns (uint256[] memory result) { /// @solidity memory-safe-assembly assembly { let searchLen := mload(needle) if iszero(gt(searchLen, mload(subject))) { result := mload(0x40) let i := add(subject, 0x20) let o := add(result, 0x20) let subjectSearchEnd := add(sub(add(i, mload(subject)), searchLen), 1) let h := 0 // The hash of `needle`. if iszero(lt(searchLen, 0x20)) { h := keccak256(add(needle, 0x20), searchLen) } let s := mload(add(needle, 0x20)) for { let m := shl(3, sub(0x20, and(searchLen, 0x1f))) } 1 {} { let t := mload(i) // Whether the first `searchLen % 32` bytes of `subject` and `needle` matches. if iszero(shr(m, xor(t, s))) { if h { if iszero(eq(keccak256(i, searchLen), h)) { i := add(i, 1) if iszero(lt(i, subjectSearchEnd)) { break } continue } } mstore(o, sub(i, add(subject, 0x20))) // Append to `result`. o := add(o, 0x20) i := add(i, searchLen) // Advance `i` by `searchLen`. if searchLen { if iszero(lt(i, subjectSearchEnd)) { break } continue } } i := add(i, 1) if iszero(lt(i, subjectSearchEnd)) { break } } mstore(result, shr(5, sub(o, add(result, 0x20)))) // Store the length of `result`. // Allocate memory for result. // We allocate one more word, so this array can be recycled for {split}. mstore(0x40, add(o, 0x20)) } } } /// @dev Returns a arrays of bytess based on the `delimiter` inside of the `subject` bytes. function split(bytes memory subject, bytes memory delimiter) internal pure returns (bytes[] memory result) { uint256[] memory indices = indicesOf(subject, delimiter); /// @solidity memory-safe-assembly assembly { let w := not(0x1f) let indexPtr := add(indices, 0x20) let indicesEnd := add(indexPtr, shl(5, add(mload(indices), 1))) mstore(add(indicesEnd, w), mload(subject)) mstore(indices, add(mload(indices), 1)) for { let prevIndex := 0 } 1 {} { let index := mload(indexPtr) mstore(indexPtr, 0x60) if iszero(eq(index, prevIndex)) { let element := mload(0x40) let l := sub(index, prevIndex) mstore(element, l) // Store the length of the element. // Copy the `subject` one word at a time, backwards. for { let o := and(add(l, 0x1f), w) } 1 {} { mstore(add(element, o), mload(add(add(subject, prevIndex), o))) o := add(o, w) // `sub(o, 0x20)`. if iszero(o) { break } } mstore(add(add(element, 0x20), l), 0) // Zeroize the slot after the bytes. // Allocate memory for the length and the bytes, rounded up to a multiple of 32. mstore(0x40, add(element, and(add(l, 0x3f), w))) mstore(indexPtr, element) // Store the `element` into the array. } prevIndex := add(index, mload(delimiter)) indexPtr := add(indexPtr, 0x20) if iszero(lt(indexPtr, indicesEnd)) { break } } result := indices if iszero(mload(delimiter)) { result := add(indices, 0x20) mstore(result, sub(mload(indices), 2)) } } } /// @dev Returns a concatenated bytes of `a` and `b`. /// Cheaper than `bytes.concat()` and does not de-align the free memory pointer. function concat(bytes memory a, bytes memory b) internal pure returns (bytes memory result) { /// @solidity memory-safe-assembly assembly { result := mload(0x40) let w := not(0x1f) let aLen := mload(a) // Copy `a` one word at a time, backwards. for { let o := and(add(aLen, 0x20), w) } 1 {} { mstore(add(result, o), mload(add(a, o))) o := add(o, w) // `sub(o, 0x20)`. if iszero(o) { break } } let bLen := mload(b) let output := add(result, aLen) // Copy `b` one word at a time, backwards. for { let o := and(add(bLen, 0x20), w) } 1 {} { mstore(add(output, o), mload(add(b, o))) o := add(o, w) // `sub(o, 0x20)`. if iszero(o) { break } } let totalLen := add(aLen, bLen) let last := add(add(result, 0x20), totalLen) mstore(last, 0) // Zeroize the slot after the bytes. mstore(result, totalLen) // Store the length. mstore(0x40, add(last, 0x20)) // Allocate memory. } } /// @dev Returns whether `a` equals `b`. function eq(bytes memory a, bytes memory b) internal pure returns (bool result) { /// @solidity memory-safe-assembly assembly { result := eq(keccak256(add(a, 0x20), mload(a)), keccak256(add(b, 0x20), mload(b))) } } /// @dev Returns whether `a` equals `b`, where `b` is a null-terminated small bytes. function eqs(bytes memory a, bytes32 b) internal pure returns (bool result) { /// @solidity memory-safe-assembly assembly { // These should be evaluated on compile time, as far as possible. let m := not(shl(7, div(not(iszero(b)), 255))) // `0x7f7f ...`. let x := not(or(m, or(b, add(m, and(b, m))))) let r := shl(7, iszero(iszero(shr(128, x)))) r := or(r, shl(6, iszero(iszero(shr(64, shr(r, x)))))) r := or(r, shl(5, lt(0xffffffff, shr(r, x)))) r := or(r, shl(4, lt(0xffff, shr(r, x)))) r := or(r, shl(3, lt(0xff, shr(r, x)))) // forgefmt: disable-next-item result := gt(eq(mload(a), add(iszero(x), xor(31, shr(3, r)))), xor(shr(add(8, r), b), shr(add(8, r), mload(add(a, 0x20))))) } } /// @dev Directly returns `a` without copying. function directReturn(bytes memory a) internal pure { assembly { // Assumes that the bytes does not start from the scratch space. let retStart := sub(a, 0x20) let retUnpaddedSize := add(mload(a), 0x40) // Right pad with zeroes. Just in case the bytes is produced // by a method that doesn't zero right pad. mstore(add(retStart, retUnpaddedSize), 0) mstore(retStart, 0x20) // Store the return offset. // End the transaction, returning the bytes. return(retStart, and(not(0x1f), add(0x1f, retUnpaddedSize))) } } /// @dev Directly returns `a` with minimal copying. function directReturn(bytes[] memory a) internal pure { assembly { let n := mload(a) // `a.length`. let o := add(a, 0x20) // Start of elements in `a`. let u := a // Highest memory slot. let w := not(0x1f) for { let i := 0 } iszero(eq(i, n)) { i := add(i, 1) } { let c := add(o, shl(5, i)) // Location of pointer to `a[i]`. let s := mload(c) // `a[i]`. let l := mload(s) // `a[i].length`. let r := and(l, 0x1f) // `a[i].length % 32`. let z := add(0x20, and(l, w)) // Offset of last word in `a[i]` from `s`. // If `s` comes before `o`, or `s` is not zero right padded. if iszero(lt(lt(s, o), or(iszero(r), iszero(shl(shl(3, r), mload(add(s, z))))))) { let m := mload(0x40) mstore(m, l) // Copy `a[i].length`. for {} 1 {} { mstore(add(m, z), mload(add(s, z))) // Copy `a[i]`, backwards. z := add(z, w) // `sub(z, 0x20)`. if iszero(z) { break } } let e := add(add(m, 0x20), l) mstore(e, 0) // Zeroize the slot after the copied bytes. mstore(0x40, add(e, 0x20)) // Allocate memory. s := m } mstore(c, sub(s, o)) // Convert to calldata offset. let t := add(l, add(s, 0x20)) if iszero(lt(t, u)) { u := t } } let retStart := add(a, w) // Assumes `a` doesn't start from scratch space. mstore(retStart, 0x20) // Store the return offset. return(retStart, add(0x40, sub(u, retStart))) // End the transaction. } } /// @dev Returns the word at `offset`, without any bounds checks. /// To load an address, you can use `address(bytes20(load(a, offset)))`. function load(bytes memory a, uint256 offset) internal pure returns (bytes32 result) { /// @solidity memory-safe-assembly assembly { result := mload(add(add(a, 0x20), offset)) } } /// @dev Returns the word at `offset`, without any bounds checks. /// To load an address, you can use `address(bytes20(loadCalldata(a, offset)))`. function loadCalldata(bytes calldata a, uint256 offset) internal pure returns (bytes32 result) { /// @solidity memory-safe-assembly assembly { result := calldataload(add(a.offset, offset)) } } /// @dev Returns empty calldata bytes. For silencing the compiler. function emptyCalldata() internal pure returns (bytes calldata result) { /// @solidity memory-safe-assembly assembly { result.length := 0 } } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.4; import {LibBytes} from "./LibBytes.sol"; /// @notice Library for converting numbers into strings and other string operations. /// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/LibString.sol) /// @author Modified from Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/LibString.sol) /// /// @dev Note: /// For performance and bytecode compactness, most of the string operations are restricted to /// byte strings (7-bit ASCII), except where otherwise specified. /// Usage of byte string operations on charsets with runes spanning two or more bytes /// can lead to undefined behavior. library LibString { /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* STRUCTS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev Goated string storage struct that totally MOGs, no cap, fr. /// Uses less gas and bytecode than Solidity's native string storage. It's meta af. /// Packs length with the first 31 bytes if <255 bytes, so it’s mad tight. struct StringStorage { bytes32 _spacer; } /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* CUSTOM ERRORS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev The length of the output is too small to contain all the hex digits. error HexLengthInsufficient(); /// @dev The length of the string is more than 32 bytes. error TooBigForSmallString(); /// @dev The input string must be a 7-bit ASCII. error StringNot7BitASCII(); /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* CONSTANTS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev The constant returned when the `search` is not found in the string. uint256 internal constant NOT_FOUND = type(uint256).max; /// @dev Lookup for '0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'. uint128 internal constant ALPHANUMERIC_7_BIT_ASCII = 0x7fffffe07fffffe03ff000000000000; /// @dev Lookup for 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'. uint128 internal constant LETTERS_7_BIT_ASCII = 0x7fffffe07fffffe0000000000000000; /// @dev Lookup for 'abcdefghijklmnopqrstuvwxyz'. uint128 internal constant LOWERCASE_7_BIT_ASCII = 0x7fffffe000000000000000000000000; /// @dev Lookup for 'ABCDEFGHIJKLMNOPQRSTUVWXYZ'. uint128 internal constant UPPERCASE_7_BIT_ASCII = 0x7fffffe0000000000000000; /// @dev Lookup for '0123456789'. uint128 internal constant DIGITS_7_BIT_ASCII = 0x3ff000000000000; /// @dev Lookup for '0123456789abcdefABCDEF'. uint128 internal constant HEXDIGITS_7_BIT_ASCII = 0x7e0000007e03ff000000000000; /// @dev Lookup for '01234567'. uint128 internal constant OCTDIGITS_7_BIT_ASCII = 0xff000000000000; /// @dev Lookup for '0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ!"#$%&\'()*+,-./:;<=>?@[\\]^_`{|}~ \t\n\r\x0b\x0c'. uint128 internal constant PRINTABLE_7_BIT_ASCII = 0x7fffffffffffffffffffffff00003e00; /// @dev Lookup for '!"#$%&\'()*+,-./:;<=>?@[\\]^_`{|}~'. uint128 internal constant PUNCTUATION_7_BIT_ASCII = 0x78000001f8000001fc00fffe00000000; /// @dev Lookup for ' \t\n\r\x0b\x0c'. uint128 internal constant WHITESPACE_7_BIT_ASCII = 0x100003e00; /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* STRING STORAGE OPERATIONS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev Sets the value of the string storage `$` to `s`. function set(StringStorage storage $, string memory s) internal { LibBytes.set(bytesStorage($), bytes(s)); } /// @dev Sets the value of the string storage `$` to `s`. function setCalldata(StringStorage storage $, string calldata s) internal { LibBytes.setCalldata(bytesStorage($), bytes(s)); } /// @dev Sets the value of the string storage `$` to the empty string. function clear(StringStorage storage $) internal { delete $._spacer; } /// @dev Returns whether the value stored is `$` is the empty string "". function isEmpty(StringStorage storage $) internal view returns (bool) { return uint256($._spacer) & 0xff == uint256(0); } /// @dev Returns the length of the value stored in `$`. function length(StringStorage storage $) internal view returns (uint256) { return LibBytes.length(bytesStorage($)); } /// @dev Returns the value stored in `$`. function get(StringStorage storage $) internal view returns (string memory) { return string(LibBytes.get(bytesStorage($))); } /// @dev Helper to cast `$` to a `BytesStorage`. function bytesStorage(StringStorage storage $) internal pure returns (LibBytes.BytesStorage storage casted) { /// @solidity memory-safe-assembly assembly { casted.slot := $.slot } } /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* DECIMAL OPERATIONS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev Returns the base 10 decimal representation of `value`. function toString(uint256 value) internal pure returns (string memory result) { /// @solidity memory-safe-assembly assembly { // The maximum value of a uint256 contains 78 digits (1 byte per digit), but // we allocate 0xa0 bytes to keep the free memory pointer 32-byte word aligned. // We will need 1 word for the trailing zeros padding, 1 word for the length, // and 3 words for a maximum of 78 digits. result := add(mload(0x40), 0x80) mstore(0x40, add(result, 0x20)) // Allocate memory. mstore(result, 0) // Zeroize the slot after the string. let end := result // Cache the end of the memory to calculate the length later. let w := not(0) // Tsk. // We write the string from rightmost digit to leftmost digit. // The following is essentially a do-while loop that also handles the zero case. for { let temp := value } 1 {} { result := add(result, w) // `sub(result, 1)`. // Store the character to the pointer. // The ASCII index of the '0' character is 48. mstore8(result, add(48, mod(temp, 10))) temp := div(temp, 10) // Keep dividing `temp` until zero. if iszero(temp) { break } } let n := sub(end, result) result := sub(result, 0x20) // Move the pointer 32 bytes back to make room for the length. mstore(result, n) // Store the length. } } /// @dev Returns the base 10 decimal representation of `value`. function toString(int256 value) internal pure returns (string memory result) { if (value >= 0) return toString(uint256(value)); unchecked { result = toString(~uint256(value) + 1); } /// @solidity memory-safe-assembly assembly { // We still have some spare memory space on the left, // as we have allocated 3 words (96 bytes) for up to 78 digits. let n := mload(result) // Load the string length. mstore(result, 0x2d) // Store the '-' character. result := sub(result, 1) // Move back the string pointer by a byte. mstore(result, add(n, 1)) // Update the string length. } } /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* HEXADECIMAL OPERATIONS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev Returns the hexadecimal representation of `value`, /// left-padded to an input length of `byteCount` bytes. /// The output is prefixed with "0x" encoded using 2 hexadecimal digits per byte, /// giving a total length of `byteCount * 2 + 2` bytes. /// Reverts if `byteCount` is too small for the output to contain all the digits. function toHexString(uint256 value, uint256 byteCount) internal pure returns (string memory result) { result = toHexStringNoPrefix(value, byteCount); /// @solidity memory-safe-assembly assembly { let n := add(mload(result), 2) // Compute the length. mstore(result, 0x3078) // Store the "0x" prefix. result := sub(result, 2) // Move the pointer. mstore(result, n) // Store the length. } } /// @dev Returns the hexadecimal representation of `value`, /// left-padded to an input length of `byteCount` bytes. /// The output is not prefixed with "0x" and is encoded using 2 hexadecimal digits per byte, /// giving a total length of `byteCount * 2` bytes. /// Reverts if `byteCount` is too small for the output to contain all the digits. function toHexStringNoPrefix(uint256 value, uint256 byteCount) internal pure returns (string memory result) { /// @solidity memory-safe-assembly assembly { // We need 0x20 bytes for the trailing zeros padding, `byteCount * 2` bytes // for the digits, 0x02 bytes for the prefix, and 0x20 bytes for the length. // We add 0x20 to the total and round down to a multiple of 0x20. // (0x20 + 0x20 + 0x02 + 0x20) = 0x62. result := add(mload(0x40), and(add(shl(1, byteCount), 0x42), not(0x1f))) mstore(0x40, add(result, 0x20)) // Allocate memory. mstore(result, 0) // Zeroize the slot after the string. let end := result // Cache the end to calculate the length later. // Store "0123456789abcdef" in scratch space. mstore(0x0f, 0x30313233343536373839616263646566) let start := sub(result, add(byteCount, byteCount)) let w := not(1) // Tsk. let temp := value // We write the string from rightmost digit to leftmost digit. // The following is essentially a do-while loop that also handles the zero case. for {} 1 {} { result := add(result, w) // `sub(result, 2)`. mstore8(add(result, 1), mload(and(temp, 15))) mstore8(result, mload(and(shr(4, temp), 15))) temp := shr(8, temp) if iszero(xor(result, start)) { break } } if temp { mstore(0x00, 0x2194895a) // `HexLengthInsufficient()`. revert(0x1c, 0x04) } let n := sub(end, result) result := sub(result, 0x20) mstore(result, n) // Store the length. } } /// @dev Returns the hexadecimal representation of `value`. /// The output is prefixed with "0x" and encoded using 2 hexadecimal digits per byte. /// As address are 20 bytes long, the output will left-padded to have /// a length of `20 * 2 + 2` bytes. function toHexString(uint256 value) internal pure returns (string memory result) { result = toHexStringNoPrefix(value); /// @solidity memory-safe-assembly assembly { let n := add(mload(result), 2) // Compute the length. mstore(result, 0x3078) // Store the "0x" prefix. result := sub(result, 2) // Move the pointer. mstore(result, n) // Store the length. } } /// @dev Returns the hexadecimal representation of `value`. /// The output is prefixed with "0x". /// The output excludes leading "0" from the `toHexString` output. /// `0x00: "0x0", 0x01: "0x1", 0x12: "0x12", 0x123: "0x123"`. function toMinimalHexString(uint256 value) internal pure returns (string memory result) { result = toHexStringNoPrefix(value); /// @solidity memory-safe-assembly assembly { let o := eq(byte(0, mload(add(result, 0x20))), 0x30) // Whether leading zero is present. let n := add(mload(result), 2) // Compute the length. mstore(add(result, o), 0x3078) // Store the "0x" prefix, accounting for leading zero. result := sub(add(result, o), 2) // Move the pointer, accounting for leading zero. mstore(result, sub(n, o)) // Store the length, accounting for leading zero. } } /// @dev Returns the hexadecimal representation of `value`. /// The output excludes leading "0" from the `toHexStringNoPrefix` output. /// `0x00: "0", 0x01: "1", 0x12: "12", 0x123: "123"`. function toMinimalHexStringNoPrefix(uint256 value) internal pure returns (string memory result) { result = toHexStringNoPrefix(value); /// @solidity memory-safe-assembly assembly { let o := eq(byte(0, mload(add(result, 0x20))), 0x30) // Whether leading zero is present. let n := mload(result) // Get the length. result := add(result, o) // Move the pointer, accounting for leading zero. mstore(result, sub(n, o)) // Store the length, accounting for leading zero. } } /// @dev Returns the hexadecimal representation of `value`. /// The output is encoded using 2 hexadecimal digits per byte. /// As address are 20 bytes long, the output will left-padded to have /// a length of `20 * 2` bytes. function toHexStringNoPrefix(uint256 value) internal pure returns (string memory result) { /// @solidity memory-safe-assembly assembly { // We need 0x20 bytes for the trailing zeros padding, 0x20 bytes for the length, // 0x02 bytes for the prefix, and 0x40 bytes for the digits. // The next multiple of 0x20 above (0x20 + 0x20 + 0x02 + 0x40) is 0xa0. result := add(mload(0x40), 0x80) mstore(0x40, add(result, 0x20)) // Allocate memory. mstore(result, 0) // Zeroize the slot after the string. let end := result // Cache the end to calculate the length later. mstore(0x0f, 0x30313233343536373839616263646566) // Store the "0123456789abcdef" lookup. let w := not(1) // Tsk. // We write the string from rightmost digit to leftmost digit. // The following is essentially a do-while loop that also handles the zero case. for { let temp := value } 1 {} { result := add(result, w) // `sub(result, 2)`. mstore8(add(result, 1), mload(and(temp, 15))) mstore8(result, mload(and(shr(4, temp), 15))) temp := shr(8, temp) if iszero(temp) { break } } let n := sub(end, result) result := sub(result, 0x20) mstore(result, n) // Store the length. } } /// @dev Returns the hexadecimal representation of `value`. /// The output is prefixed with "0x", encoded using 2 hexadecimal digits per byte, /// and the alphabets are capitalized conditionally according to /// https://eips.ethereum.org/EIPS/eip-55 function toHexStringChecksummed(address value) internal pure returns (string memory result) { result = toHexString(value); /// @solidity memory-safe-assembly assembly { let mask := shl(6, div(not(0), 255)) // `0b010000000100000000 ...` let o := add(result, 0x22) let hashed := and(keccak256(o, 40), mul(34, mask)) // `0b10001000 ... ` let t := shl(240, 136) // `0b10001000 << 240` for { let i := 0 } 1 {} { mstore(add(i, i), mul(t, byte(i, hashed))) i := add(i, 1) if eq(i, 20) { break } } mstore(o, xor(mload(o), shr(1, and(mload(0x00), and(mload(o), mask))))) o := add(o, 0x20) mstore(o, xor(mload(o), shr(1, and(mload(0x20), and(mload(o), mask))))) } } /// @dev Returns the hexadecimal representation of `value`. /// The output is prefixed with "0x" and encoded using 2 hexadecimal digits per byte. function toHexString(address value) internal pure returns (string memory result) { result = toHexStringNoPrefix(value); /// @solidity memory-safe-assembly assembly { let n := add(mload(result), 2) // Compute the length. mstore(result, 0x3078) // Store the "0x" prefix. result := sub(result, 2) // Move the pointer. mstore(result, n) // Store the length. } } /// @dev Returns the hexadecimal representation of `value`. /// The output is encoded using 2 hexadecimal digits per byte. function toHexStringNoPrefix(address value) internal pure returns (string memory result) { /// @solidity memory-safe-assembly assembly { result := mload(0x40) // Allocate memory. // We need 0x20 bytes for the trailing zeros padding, 0x20 bytes for the length, // 0x02 bytes for the prefix, and 0x28 bytes for the digits. // The next multiple of 0x20 above (0x20 + 0x20 + 0x02 + 0x28) is 0x80. mstore(0x40, add(result, 0x80)) mstore(0x0f, 0x30313233343536373839616263646566) // Store the "0123456789abcdef" lookup. result := add(result, 2) mstore(result, 40) // Store the length. let o := add(result, 0x20) mstore(add(o, 40), 0) // Zeroize the slot after the string. value := shl(96, value) // We write the string from rightmost digit to leftmost digit. // The following is essentially a do-while loop that also handles the zero case. for { let i := 0 } 1 {} { let p := add(o, add(i, i)) let temp := byte(i, value) mstore8(add(p, 1), mload(and(temp, 15))) mstore8(p, mload(shr(4, temp))) i := add(i, 1) if eq(i, 20) { break } } } } /// @dev Returns the hex encoded string from the raw bytes. /// The output is encoded using 2 hexadecimal digits per byte. function toHexString(bytes memory raw) internal pure returns (string memory result) { result = toHexStringNoPrefix(raw); /// @solidity memory-safe-assembly assembly { let n := add(mload(result), 2) // Compute the length. mstore(result, 0x3078) // Store the "0x" prefix. result := sub(result, 2) // Move the pointer. mstore(result, n) // Store the length. } } /// @dev Returns the hex encoded string from the raw bytes. /// The output is encoded using 2 hexadecimal digits per byte. function toHexStringNoPrefix(bytes memory raw) internal pure returns (string memory result) { /// @solidity memory-safe-assembly assembly { let n := mload(raw) result := add(mload(0x40), 2) // Skip 2 bytes for the optional prefix. mstore(result, add(n, n)) // Store the length of the output. mstore(0x0f, 0x30313233343536373839616263646566) // Store the "0123456789abcdef" lookup. let o := add(result, 0x20) let end := add(raw, n) for {} iszero(eq(raw, end)) {} { raw := add(raw, 1) mstore8(add(o, 1), mload(and(mload(raw), 15))) mstore8(o, mload(and(shr(4, mload(raw)), 15))) o := add(o, 2) } mstore(o, 0) // Zeroize the slot after the string. mstore(0x40, add(o, 0x20)) // Allocate memory. } } /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* RUNE STRING OPERATIONS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev Returns the number of UTF characters in the string. function runeCount(string memory s) internal pure returns (uint256 result) { /// @solidity memory-safe-assembly assembly { if mload(s) { mstore(0x00, div(not(0), 255)) mstore(0x20, 0x0202020202020202020202020202020202020202020202020303030304040506) let o := add(s, 0x20) let end := add(o, mload(s)) for { result := 1 } 1 { result := add(result, 1) } { o := add(o, byte(0, mload(shr(250, mload(o))))) if iszero(lt(o, end)) { break } } } } } /// @dev Returns if this string is a 7-bit ASCII string. /// (i.e. all characters codes are in [0..127]) function is7BitASCII(string memory s) internal pure returns (bool result) { /// @solidity memory-safe-assembly assembly { result := 1 let mask := shl(7, div(not(0), 255)) let n := mload(s) if n { let o := add(s, 0x20) let end := add(o, n) let last := mload(end) mstore(end, 0) for {} 1 {} { if and(mask, mload(o)) { result := 0 break } o := add(o, 0x20) if iszero(lt(o, end)) { break } } mstore(end, last) } } } /// @dev Returns if this string is a 7-bit ASCII string, /// AND all characters are in the `allowed` lookup. /// Note: If `s` is empty, returns true regardless of `allowed`. function is7BitASCII(string memory s, uint128 allowed) internal pure returns (bool result) { /// @solidity memory-safe-assembly assembly { result := 1 if mload(s) { let allowed_ := shr(128, shl(128, allowed)) let o := add(s, 0x20) for { let end := add(o, mload(s)) } 1 {} { result := and(result, shr(byte(0, mload(o)), allowed_)) o := add(o, 1) if iszero(and(result, lt(o, end))) { break } } } } } /// @dev Converts the bytes in the 7-bit ASCII string `s` to /// an allowed lookup for use in `is7BitASCII(s, allowed)`. /// To save runtime gas, you can cache the result in an immutable variable. function to7BitASCIIAllowedLookup(string memory s) internal pure returns (uint128 result) { /// @solidity memory-safe-assembly assembly { if mload(s) { let o := add(s, 0x20) for { let end := add(o, mload(s)) } 1 {} { result := or(result, shl(byte(0, mload(o)), 1)) o := add(o, 1) if iszero(lt(o, end)) { break } } if shr(128, result) { mstore(0x00, 0xc9807e0d) // `StringNot7BitASCII()`. revert(0x1c, 0x04) } } } } /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* BYTE STRING OPERATIONS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ // For performance and bytecode compactness, byte string operations are restricted // to 7-bit ASCII strings. All offsets are byte offsets, not UTF character offsets. // Usage of byte string operations on charsets with runes spanning two or more bytes // can lead to undefined behavior. /// @dev Returns `subject` all occurrences of `needle` replaced with `replacement`. function replace(string memory subject, string memory needle, string memory replacement) internal pure returns (string memory) { return string(LibBytes.replace(bytes(subject), bytes(needle), bytes(replacement))); } /// @dev Returns the byte index of the first location of `needle` in `subject`, /// needleing from left to right, starting from `from`. /// Returns `NOT_FOUND` (i.e. `type(uint256).max`) if the `needle` is not found. function indexOf(string memory subject, string memory needle, uint256 from) internal pure returns (uint256) { return LibBytes.indexOf(bytes(subject), bytes(needle), from); } /// @dev Returns the byte index of the first location of `needle` in `subject`, /// needleing from left to right. /// Returns `NOT_FOUND` (i.e. `type(uint256).max`) if the `needle` is not found. function indexOf(string memory subject, string memory needle) internal pure returns (uint256) { return LibBytes.indexOf(bytes(subject), bytes(needle), 0); } /// @dev Returns the byte index of the first location of `needle` in `subject`, /// needleing from right to left, starting from `from`. /// Returns `NOT_FOUND` (i.e. `type(uint256).max`) if the `needle` is not found. function lastIndexOf(string memory subject, string memory needle, uint256 from) internal pure returns (uint256) { return LibBytes.lastIndexOf(bytes(subject), bytes(needle), from); } /// @dev Returns the byte index of the first location of `needle` in `subject`, /// needleing from right to left. /// Returns `NOT_FOUND` (i.e. `type(uint256).max`) if the `needle` is not found. function lastIndexOf(string memory subject, string memory needle) internal pure returns (uint256) { return LibBytes.lastIndexOf(bytes(subject), bytes(needle), type(uint256).max); } /// @dev Returns true if `needle` is found in `subject`, false otherwise. function contains(string memory subject, string memory needle) internal pure returns (bool) { return LibBytes.contains(bytes(subject), bytes(needle)); } /// @dev Returns whether `subject` starts with `needle`. function startsWith(string memory subject, string memory needle) internal pure returns (bool) { return LibBytes.startsWith(bytes(subject), bytes(needle)); } /// @dev Returns whether `subject` ends with `needle`. function endsWith(string memory subject, string memory needle) internal pure returns (bool) { return LibBytes.endsWith(bytes(subject), bytes(needle)); } /// @dev Returns `subject` repeated `times`. function repeat(string memory subject, uint256 times) internal pure returns (string memory) { return string(LibBytes.repeat(bytes(subject), times)); } /// @dev Returns a copy of `subject` sliced from `start` to `end` (exclusive). /// `start` and `end` are byte offsets. function slice(string memory subject, uint256 start, uint256 end) internal pure returns (string memory) { return string(LibBytes.slice(bytes(subject), start, end)); } /// @dev Returns a copy of `subject` sliced from `start` to the end of the string. /// `start` is a byte offset. function slice(string memory subject, uint256 start) internal pure returns (string memory) { return string(LibBytes.slice(bytes(subject), start, type(uint256).max)); } /// @dev Returns all the indices of `needle` in `subject`. /// The indices are byte offsets. function indicesOf(string memory subject, string memory needle) internal pure returns (uint256[] memory) { return LibBytes.indicesOf(bytes(subject), bytes(needle)); } /// @dev Returns a arrays of strings based on the `delimiter` inside of the `subject` string. function split(string memory subject, string memory delimiter) internal pure returns (string[] memory result) { bytes[] memory a = LibBytes.split(bytes(subject), bytes(delimiter)); /// @solidity memory-safe-assembly assembly { result := a } } /// @dev Returns a concatenated string of `a` and `b`. /// Cheaper than `string.concat()` and does not de-align the free memory pointer. function concat(string memory a, string memory b) internal pure returns (string memory) { return string(LibBytes.concat(bytes(a), bytes(b))); } /// @dev Returns a copy of the string in either lowercase or UPPERCASE. /// WARNING! This function is only compatible with 7-bit ASCII strings. function toCase(string memory subject, bool toUpper) internal pure returns (string memory result) { /// @solidity memory-safe-assembly assembly { let n := mload(subject) if n { result := mload(0x40) let o := add(result, 0x20) let d := sub(subject, result) let flags := shl(add(70, shl(5, toUpper)), 0x3ffffff) for { let end := add(o, n) } 1 {} { let b := byte(0, mload(add(d, o))) mstore8(o, xor(and(shr(b, flags), 0x20), b)) o := add(o, 1) if eq(o, end) { break } } mstore(result, n) // Store the length. mstore(o, 0) // Zeroize the slot after the string. mstore(0x40, add(o, 0x20)) // Allocate memory. } } } /// @dev Returns a string from a small bytes32 string. /// `s` must be null-terminated, or behavior will be undefined. function fromSmallString(bytes32 s) internal pure returns (string memory result) { /// @solidity memory-safe-assembly assembly { result := mload(0x40) let n := 0 for {} byte(n, s) { n := add(n, 1) } {} // Scan for '\0'. mstore(result, n) // Store the length. let o := add(result, 0x20) mstore(o, s) // Store the bytes of the string. mstore(add(o, n), 0) // Zeroize the slot after the string. mstore(0x40, add(result, 0x40)) // Allocate memory. } } /// @dev Returns the small string, with all bytes after the first null byte zeroized. function normalizeSmallString(bytes32 s) internal pure returns (bytes32 result) { /// @solidity memory-safe-assembly assembly { for {} byte(result, s) { result := add(result, 1) } {} // Scan for '\0'. mstore(0x00, s) mstore(result, 0x00) result := mload(0x00) } } /// @dev Returns the string as a normalized null-terminated small string. function toSmallString(string memory s) internal pure returns (bytes32 result) { /// @solidity memory-safe-assembly assembly { result := mload(s) if iszero(lt(result, 33)) { mstore(0x00, 0xec92f9a3) // `TooBigForSmallString()`. revert(0x1c, 0x04) } result := shl(shl(3, sub(32, result)), mload(add(s, result))) } } /// @dev Returns a lowercased copy of the string. /// WARNING! This function is only compatible with 7-bit ASCII strings. function lower(string memory subject) internal pure returns (string memory result) { result = toCase(subject, false); } /// @dev Returns an UPPERCASED copy of the string. /// WARNING! This function is only compatible with 7-bit ASCII strings. function upper(string memory subject) internal pure returns (string memory result) { result = toCase(subject, true); } /// @dev Escapes the string to be used within HTML tags. function escapeHTML(string memory s) internal pure returns (string memory result) { /// @solidity memory-safe-assembly assembly { result := mload(0x40) let end := add(s, mload(s)) let o := add(result, 0x20) // Store the bytes of the packed offsets and strides into the scratch space. // `packed = (stride << 5) | offset`. Max offset is 20. Max stride is 6. mstore(0x1f, 0x900094) mstore(0x08, 0xc0000000a6ab) // Store ""&'<>" into the scratch space. mstore(0x00, shl(64, 0x2671756f743b26616d703b262333393b266c743b2667743b)) for {} iszero(eq(s, end)) {} { s := add(s, 1) let c := and(mload(s), 0xff) // Not in `["\"","'","&","<",">"]`. if iszero(and(shl(c, 1), 0x500000c400000000)) { mstore8(o, c) o := add(o, 1) continue } let t := shr(248, mload(c)) mstore(o, mload(and(t, 0x1f))) o := add(o, shr(5, t)) } mstore(o, 0) // Zeroize the slot after the string. mstore(result, sub(o, add(result, 0x20))) // Store the length. mstore(0x40, add(o, 0x20)) // Allocate memory. } } /// @dev Escapes the string to be used within double-quotes in a JSON. /// If `addDoubleQuotes` is true, the result will be enclosed in double-quotes. function escapeJSON(string memory s, bool addDoubleQuotes) internal pure returns (string memory result) { /// @solidity memory-safe-assembly assembly { result := mload(0x40) let o := add(result, 0x20) if addDoubleQuotes { mstore8(o, 34) o := add(1, o) } // Store "\\u0000" in scratch space. // Store "0123456789abcdef" in scratch space. // Also, store `{0x08:"b", 0x09:"t", 0x0a:"n", 0x0c:"f", 0x0d:"r"}`. // into the scratch space. mstore(0x15, 0x5c75303030303031323334353637383961626364656662746e006672) // Bitmask for detecting `["\"","\\"]`. let e := or(shl(0x22, 1), shl(0x5c, 1)) for { let end := add(s, mload(s)) } iszero(eq(s, end)) {} { s := add(s, 1) let c := and(mload(s), 0xff) if iszero(lt(c, 0x20)) { if iszero(and(shl(c, 1), e)) { // Not in `["\"","\\"]`. mstore8(o, c) o := add(o, 1) continue } mstore8(o, 0x5c) // "\\". mstore8(add(o, 1), c) o := add(o, 2) continue } if iszero(and(shl(c, 1), 0x3700)) { // Not in `["\b","\t","\n","\f","\d"]`. mstore8(0x1d, mload(shr(4, c))) // Hex value. mstore8(0x1e, mload(and(c, 15))) // Hex value. mstore(o, mload(0x19)) // "\\u00XX". o := add(o, 6) continue } mstore8(o, 0x5c) // "\\". mstore8(add(o, 1), mload(add(c, 8))) o := add(o, 2) } if addDoubleQuotes { mstore8(o, 34) o := add(1, o) } mstore(o, 0) // Zeroize the slot after the string. mstore(result, sub(o, add(result, 0x20))) // Store the length. mstore(0x40, add(o, 0x20)) // Allocate memory. } } /// @dev Escapes the string to be used within double-quotes in a JSON. function escapeJSON(string memory s) internal pure returns (string memory result) { result = escapeJSON(s, false); } /// @dev Encodes `s` so that it can be safely used in a URI, /// just like `encodeURIComponent` in JavaScript. /// See: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/encodeURIComponent /// See: https://datatracker.ietf.org/doc/html/rfc2396 /// See: https://datatracker.ietf.org/doc/html/rfc3986 function encodeURIComponent(string memory s) internal pure returns (string memory result) { /// @solidity memory-safe-assembly assembly { result := mload(0x40) // Store "0123456789ABCDEF" in scratch space. // Uppercased to be consistent with JavaScript's implementation. mstore(0x0f, 0x30313233343536373839414243444546) let o := add(result, 0x20) for { let end := add(s, mload(s)) } iszero(eq(s, end)) {} { s := add(s, 1) let c := and(mload(s), 0xff) // If not in `[0-9A-Z-a-z-_.!~*'()]`. if iszero(and(1, shr(c, 0x47fffffe87fffffe03ff678200000000))) { mstore8(o, 0x25) // '%'. mstore8(add(o, 1), mload(and(shr(4, c), 15))) mstore8(add(o, 2), mload(and(c, 15))) o := add(o, 3) continue } mstore8(o, c) o := add(o, 1) } mstore(result, sub(o, add(result, 0x20))) // Store the length. mstore(o, 0) // Zeroize the slot after the string. mstore(0x40, add(o, 0x20)) // Allocate memory. } } /// @dev Returns whether `a` equals `b`. function eq(string memory a, string memory b) internal pure returns (bool result) { /// @solidity memory-safe-assembly assembly { result := eq(keccak256(add(a, 0x20), mload(a)), keccak256(add(b, 0x20), mload(b))) } } /// @dev Returns whether `a` equals `b`, where `b` is a null-terminated small string. function eqs(string memory a, bytes32 b) internal pure returns (bool result) { /// @solidity memory-safe-assembly assembly { // These should be evaluated on compile time, as far as possible. let m := not(shl(7, div(not(iszero(b)), 255))) // `0x7f7f ...`. let x := not(or(m, or(b, add(m, and(b, m))))) let r := shl(7, iszero(iszero(shr(128, x)))) r := or(r, shl(6, iszero(iszero(shr(64, shr(r, x)))))) r := or(r, shl(5, lt(0xffffffff, shr(r, x)))) r := or(r, shl(4, lt(0xffff, shr(r, x)))) r := or(r, shl(3, lt(0xff, shr(r, x)))) // forgefmt: disable-next-item result := gt(eq(mload(a), add(iszero(x), xor(31, shr(3, r)))), xor(shr(add(8, r), b), shr(add(8, r), mload(add(a, 0x20))))) } } /// @dev Packs a single string with its length into a single word. /// Returns `bytes32(0)` if the length is zero or greater than 31. function packOne(string memory a) internal pure returns (bytes32 result) { /// @solidity memory-safe-assembly assembly { // We don't need to zero right pad the string, // since this is our own custom non-standard packing scheme. result := mul( // Load the length and the bytes. mload(add(a, 0x1f)), // `length != 0 && length < 32`. Abuses underflow. // Assumes that the length is valid and within the block gas limit. lt(sub(mload(a), 1), 0x1f) ) } } /// @dev Unpacks a string packed using {packOne}. /// Returns the empty string if `packed` is `bytes32(0)`. /// If `packed` is not an output of {packOne}, the output behavior is undefined. function unpackOne(bytes32 packed) internal pure returns (string memory result) { /// @solidity memory-safe-assembly assembly { result := mload(0x40) // Grab the free memory pointer. mstore(0x40, add(result, 0x40)) // Allocate 2 words (1 for the length, 1 for the bytes). mstore(result, 0) // Zeroize the length slot. mstore(add(result, 0x1f), packed) // Store the length and bytes. mstore(add(add(result, 0x20), mload(result)), 0) // Right pad with zeroes. } } /// @dev Packs two strings with their lengths into a single word. /// Returns `bytes32(0)` if combined length is zero or greater than 30. function packTwo(string memory a, string memory b) internal pure returns (bytes32 result) { /// @solidity memory-safe-assembly assembly { let aLen := mload(a) // We don't need to zero right pad the strings, // since this is our own custom non-standard packing scheme. result := mul( or( // Load the length and the bytes of `a` and `b`. shl(shl(3, sub(0x1f, aLen)), mload(add(a, aLen))), mload(sub(add(b, 0x1e), aLen))), // `totalLen != 0 && totalLen < 31`. Abuses underflow. // Assumes that the lengths are valid and within the block gas limit. lt(sub(add(aLen, mload(b)), 1), 0x1e) ) } } /// @dev Unpacks strings packed using {packTwo}. /// Returns the empty strings if `packed` is `bytes32(0)`. /// If `packed` is not an output of {packTwo}, the output behavior is undefined. function unpackTwo(bytes32 packed) internal pure returns (string memory resultA, string memory resultB) { /// @solidity memory-safe-assembly assembly { resultA := mload(0x40) // Grab the free memory pointer. resultB := add(resultA, 0x40) // Allocate 2 words for each string (1 for the length, 1 for the byte). Total 4 words. mstore(0x40, add(resultB, 0x40)) // Zeroize the length slots. mstore(resultA, 0) mstore(resultB, 0) // Store the lengths and bytes. mstore(add(resultA, 0x1f), packed) mstore(add(resultB, 0x1f), mload(add(add(resultA, 0x20), mload(resultA)))) // Right pad with zeroes. mstore(add(add(resultA, 0x20), mload(resultA)), 0) mstore(add(add(resultB, 0x20), mload(resultB)), 0) } } /// @dev Directly returns `a` without copying. function directReturn(string memory a) internal pure { assembly { // Assumes that the string does not start from the scratch space. let retStart := sub(a, 0x20) let retUnpaddedSize := add(mload(a), 0x40) // Right pad with zeroes. Just in case the string is produced // by a method that doesn't zero right pad. mstore(add(retStart, retUnpaddedSize), 0) mstore(retStart, 0x20) // Store the return offset. // End the transaction, returning the string. return(retStart, and(not(0x1f), add(0x1f, retUnpaddedSize))) } } }
{ "optimizer": { "enabled": true, "runs": 1337 }, "viaIR": true, "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "devdoc", "userdoc", "metadata", "abi" ] } }, "libraries": {} }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"Unauthorized","type":"error"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"buyer","type":"address"},{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256","name":"ethIn","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"tokensOut","type":"uint256"},{"indexed":false,"internalType":"bytes","name":"data","type":"bytes"}],"name":"Bought","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"seller","type":"address"},{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256","name":"ethOut","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"tokensIn","type":"uint256"},{"indexed":false,"internalType":"bytes","name":"data","type":"bytes"}],"name":"Sold","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"address","name":"creator","type":"address"},{"indexed":false,"internalType":"enum LibTokens.LaunchStrategy","name":"strategy","type":"uint8"},{"indexed":false,"internalType":"enum LibDex.Dex","name":"dex","type":"uint8"},{"indexed":false,"internalType":"bytes","name":"data","type":"bytes"},{"indexed":false,"internalType":"bytes","name":"priceData","type":"bytes"}],"name":"TokenCreated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"address","name":"creator","type":"address"},{"indexed":false,"internalType":"enum LibTokens.LaunchStrategy","name":"strategy","type":"uint8"},{"indexed":false,"internalType":"enum LibDex.Dex","name":"dex","type":"uint8"},{"indexed":false,"internalType":"address","name":"pair","type":"address"}],"name":"TokenLaunched","type":"event"},{"inputs":[{"internalType":"address","name":"buyer","type":"address"},{"internalType":"address","name":"token","type":"address"},{"internalType":"uint256","name":"ethIn","type":"uint256"},{"internalType":"uint256","name":"min","type":"uint256"}],"name":"_buy","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"creator","type":"address"},{"internalType":"string","name":"name","type":"string"},{"internalType":"string","name":"symbol","type":"string"},{"internalType":"string","name":"description","type":"string"},{"internalType":"bytes","name":"image","type":"bytes"},{"internalType":"string[]","name":"links","type":"string[]"},{"internalType":"bytes","name":"data","type":"bytes"},{"internalType":"enum LibTokens.LaunchStrategy","name":"strategy","type":"uint8"},{"internalType":"enum LibDex.Dex","name":"dex","type":"uint8"},{"internalType":"uint256","name":"initialBuy","type":"uint256"},{"internalType":"uint256","name":"eth","type":"uint256"}],"name":"_create","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"seller","type":"address"},{"internalType":"address","name":"token","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"uint256","name":"min","type":"uint256"}],"name":"_sell","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"token","type":"address"},{"internalType":"uint256","name":"min","type":"uint256"}],"name":"buy","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"string","name":"name","type":"string"},{"internalType":"string","name":"symbol","type":"string"},{"internalType":"string","name":"description","type":"string"},{"internalType":"bytes","name":"image","type":"bytes"},{"internalType":"string[]","name":"links","type":"string[]"},{"internalType":"bytes","name":"data","type":"bytes"},{"internalType":"enum LibTokens.LaunchStrategy","name":"strategy","type":"uint8"},{"internalType":"enum LibDex.Dex","name":"dex","type":"uint8"},{"internalType":"uint256","name":"initialBuy","type":"uint256"}],"name":"create","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"token","type":"address"}],"name":"launch","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"token","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"bool","name":"ethOut","type":"bool"}],"name":"quote","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"token","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"uint256","name":"min","type":"uint256"}],"name":"sell","outputs":[],"stateMutability":"nonpayable","type":"function"}]
Contract Creation Code
6080806040523461001657614440908161001c8239f35b600080fdfe608060405260043610156200001357600080fd5b6000803560e01c8063214013ca14620015c45780632bafa99314620009c15780636a272462146200093d57806392f87d7b14620008f85780639eed896e1462000691578063bad6b832146200033c578063cce7ec1314620002af5763d2fc17fe146200007e57600080fd5b61012036600319011262000285578067ffffffffffffffff600435818111620002ab57620000b19036906004016200189a565b9190602435828111620002a757620000ce9036906004016200189a565b939092604435818111620002a357620000ec9036906004016200189a565b916064358181116200029f57620001089036906004016200189a565b906084358381116200029b5762000124903690600401620018cb565b92909360a4359081116200029757620001429036906004016200189a565b959096600160c43510156200029357600260e43510156200029357303b1562000293578c9a6040519c8d9b8c9b7f2bafa993000000000000000000000000000000000000000000000000000000008d528c33906004015260248d0161016090526101648d0190620001b39262001990565b906003198c83030160448d0152620001cb9262001990565b906003198a83030160648b0152620001e39262001990565b90600319888303016084890152620001fb9262001990565b906003198683030160a48701526200021392620019b1565b906003198483030160c48501526200022b9262001990565b6200023c60e4830160c43562001a42565b6101046200024f83820160e43562001a66565b3561012483015234610144830152038183305af180156200028857620002725750f35b6200027d906200193b565b620002855780f35b80fd5b6040513d84823e3d90fd5b8c80fd5b8b80fd5b8a80fd5b8880fd5b8680fd5b8480fd5b5050fd5b506040366003190112620002855780620002c86200187e565b303b1562000339576040517f9eed896e0000000000000000000000000000000000000000000000000000000081523360048201526001600160a01b03909116602480830191909152346044830152356064820152818180608481015b038183305af180156200028857620002725750f35b50fd5b503462000285576200034e36620018ff565b6200035b93919362001d0e565b60ff6200039a846001600160a01b03166000527f37c7a69cb64276d9952a1cf15ae5474311c0ddadaf138ab6ce1b4a3e73db5623602052604060002090565b5460a01c169060405191620003af8362001950565b604051620003bd8162001950565b878152835260018110156200067d5762000638576040517ffab021b70000000000000000000000000000000000000000000000000000000081526001600160a01b03851660048201526024810186905290868260448183305af19182156200062d578790889362000600575b506200043790928462001d9e565b5080620005eb575b507fd6636d7e2aa9fa70e65f3eb96671972c70cfa6486eb903d39a283c1e07a1c5986200046e82825462001b1c565b9055303b15620005e75785604051630135129d60e41b81526001600160a01b038086169081600484015260016024840152846044840152838360648183305af1928315620005dc578493620005bc575b506020906064604051809581937f23b872dd0000000000000000000000000000000000000000000000000000000083528660048401523060248401528d60448401528c165af1918215620005b15783928392839287926200057c575b5082821562000572575bf11562000567577f76afc58d2c47c2160750cec6375a6cc87ab5a99635e1c481f956b578e023f99b94620005619251916040519586958662001cd7565b0390a180f35b6040513d87823e3d90fd5b506108fc62000524565b620005a19060203d8111620005a9575b6200059881836200196d565b81019062001cbd565b50386200051a565b503d6200058c565b6040513d85823e3d90fd5b620005ca909391936200193b565b620005d857829138620004be565b8280fd5b6040513d86823e3d90fd5b8580fd5b620005f99082101562001c71565b386200043f565b6200043793506200062691503d808a833e6200061d81836200196d565b81019062001c3f565b9262000429565b6040513d89823e3d90fd5b60405162461bcd60e51b815260206004820152601060248201527f696e76616c6964207374726174656779000000000000000000000000000000006044820152606490fd5b602487634e487b7160e01b81526021600452fd5b50346200028557620006a336620018ff565b9290620006af62001d0e565b60ff620006ee846001600160a01b03166000527f37c7a69cb64276d9952a1cf15ae5474311c0ddadaf138ab6ce1b4a3e73db5623602052604060002090565b5460a01c169060405191620007038362001950565b604051620007118162001950565b878152835260018110156200067d57620006385760405194633960fa3760e01b86526001600160a01b0385166004870152868660248185305af19586156200062d5787908897620008d4575b506200076b90968462001d9e565b5080620008bf575b507fd6636d7e2aa9fa70e65f3eb96671972c70cfa6486eb903d39a283c1e07a1c598805490828201809211620008ab5755303b15620005e757604051630135129d60e41b81526001600160a01b038416600482015260006024820152604481018290528690818160648183305af18015620002885762000893575b505060405163a9059cbb60e01b81526001600160a01b03841660048201526024810186905294602086806044810103818a6001600160a01b038a165af19283156200062d577f4c451962b8f0dd56d788474a5f794da140e33218eee65f7ee9a9a8a363eaf8c096620005619462000870575b5051916040519586958662001cd7565b6200088b9060203d8111620005a9576200059881836200196d565b503862000860565b6200089e906200193b565b620005e7578538620007ee565b602488634e487b7160e01b81526011600452fd5b620008cd9086101562001c71565b3862000773565b6200076b9750620008f191503d808a833e6200061d81836200196d565b966200075d565b5034620002855760603660031901126200028557620009166200187e565b6044359182151583036200028557602062000935846024358562001b40565b604051908152f35b5034620002855760603660031901126200028557806200095c6200187e565b303b1562000339576040517fbad6b8320000000000000000000000000000000000000000000000000000000081523360048201526001600160a01b03909116602480830191909152356044808301919091523560648201528181806084810162000324565b506101603660031901126200028557620009da6200187e565b9060243567ffffffffffffffff8111620015c057620009fe9036906004016200189a565b909260443567ffffffffffffffff8111620015bc5762000a239036906004016200189a565b909260643567ffffffffffffffff8111620005e75762000a489036906004016200189a565b969060843567ffffffffffffffff81116200143d5762000a6d9036906004016200189a565b9060a43567ffffffffffffffff8111620015b85762000a91903690600401620018cb565b92909360c43567ffffffffffffffff8111620002975762000ab79036906004016200189a565b9c9098600160e435101562000293576002610104351015620002935762000add62001d0e565b610144359b60128a111580620015ac575b806200159d575b156200155957600587101562001515578d5b8760ff8216106200145657507fb08236d4879334bece1965ab3a812fad8e40dec4929b65f11b873dfaf386da8954986040519a8b61251b81011067ffffffffffffffff8d61251b01111762001441579362000bbe8c8f9562000bd59562000ba7839f9e9c999662000bec9b9762000c039f9d9a6001600160a01b039061251b62001ef089391661251b870190815261010060208201819052019162001990565b92604061251b820185039161251b01015262001990565b9161251b8c01830360608d61251b01015262001990565b9161251b89018303608061251b8b01015262001990565b9161251b8601830360a061251b88010152620019b1565b9060c061251b840101523060e061251b84010152039085f0948515620005dc5760405162000c318162001950565b60405162000c3f8162001950565b868152815260e43562000638577fb08236d4879334bece1965ab3a812fad8e40dec4929b65f11b873dfaf386da8954604051907f327dca720000000000000000000000000000000000000000000000000000000082526001600160a01b038916600483015260248201526060604482015286818062000cc360648201878962001990565b038183305af180156200062d578790620013f2575b62000ce591508262001d9e565b5060405180606081011067ffffffffffffffff606083011117620013dc57606081016040526001600160a01b03851681526020810160e4358152604082016101043581526001600160a01b0362000d70818c166001600160a01b03166000527f37c7a69cb64276d9952a1cf15ae5474311c0ddadaf138ab6ce1b4a3e73db5623602052604060002090565b93511683549251916001831015620013c857516002811015620013c8579174ff00000000000000000000000000000000000000007fe28e4635d390f60ea0f9513a16dc98c2ee9b4a0762ad677fb000b7f7cb07625198979694927fffffffffffffffffffff0000000000000000000000000000000000000000000075ff00000000000000000000000000000000000000000062000e82989660a81b169416179160a01b16171790555162000e736040519485946001600160a01b038c1686526001600160a01b038916602087015262000e4f6040870160e43562001a42565b62000e61606087016101043562001a66565b60c0608087015260c086019162001990565b9083820360a085015262001af5565b0390a160047fb08236d4879334bece1965ab3a812fad8e40dec4929b65f11b873dfaf386da875460a06001600160a01b037f88acb08f71068c878012515a2672f2b1a3f7940ce7bd383fbab30a631a76b7745416604051938480927ffeaf968c0000000000000000000000000000000000000000000000000000000082525afa80156200056757859286916200136a575b5062000f417f88acb08f71068c878012515a2672f2b1a3f7940ce7bd383fbab30a631a76b775544262001b1c565b1162001326576402540be4008083029280840482148115171562001312577812725dd1d243aba0e75fe645cc4873f9e65afe688c928e1f2283100202156200130557670de0b6b3a7640000020491828110620012c15783808062000fa786829562001b1c565b956001600160a01b037fb08236d4879334bece1965ab3a812fad8e40dec4929b65f11b873dfaf386da865416828215620012b7575bf115620002885761012435806200103a575b5082935081159081156200100157505050f35b83928392839283916200102f575b6001600160a01b031690f115620010235780f35b604051903d90823e3d90fd5b6108fc91506200100f565b6200104792919262001d0e565b60ff6200108f6001600160a01b0387166001600160a01b03166000527f37c7a69cb64276d9952a1cf15ae5474311c0ddadaf138ab6ce1b4a3e73db5623602052604060002090565b5460a01c169160405192620010a48462001950565b604051620010b28162001950565b86815284526001811015620012a357620006385760405192633960fa3760e01b84526001600160a01b0387166004850152858460248186305af193841562001298578690879562001274575b506200110c90948262001d9e565b507fd6636d7e2aa9fa70e65f3eb96671972c70cfa6486eb903d39a283c1e07a1c598968754848101809111620008ab57879855303b15620002a357604051630135129d60e41b81526001600160a01b03871660048201526000602482015260448101859052878160648183305af1908115620012515788916200125c575b505060405163a9059cbb60e01b81526001600160a01b03871660048201526024810186905290602082806044810103818b6001600160a01b0386165af18015620012515762001225967f4c451962b8f0dd56d788474a5f794da140e33218eee65f7ee9a9a8a363eaf8c0946001600160a01b03946200121c936200122d575b50519087604051958695168b8662001cd7565b0390a162001b1c565b903862000fee565b620012499060203d602011620005a9576200059881836200196d565b503862001209565b6040513d8a823e3d90fd5b62001267906200193b565b620002a35786386200118a565b6200110c95506200129191503d8089833e6200061d81836200196d565b94620010fe565b6040513d88823e3d90fd5b602486634e487b7160e01b81526021600452fd5b506108fc62000fdc565b606460405162461bcd60e51b815260206004820152601160248201527f757364207072696365206368616e6765640000000000000000000000000000006044820152fd5b637c5f487d85526004601cfd5b602487634e487b7160e01b81526011600452fd5b606460405162461bcd60e51b815260206004820152600f60248201527f7374616c652070726963656665656400000000000000000000000000000000006044820152fd5b92505060a0823d60a011620013bf575b816200138960a093836200196d565b81010312620002a7576200139d8262001d86565b506020820151620013b660806040850151940162001d86565b50913862000f13565b3d91506200137a565b60248b634e487b7160e01b81526021600452fd5b634e487b7160e01b600052604160045260246000fd5b503d8088833e6200140481836200196d565b8101906020818303126200143d5780519067ffffffffffffffff82116200029f5791620014379162000ce5930162001a99565b62000cd8565b8780fd5b5060248f634e487b7160e01b81526041600452fd5b611fe08160051b16890135601e198a36030181121562001511578901803567ffffffffffffffff81116200150c57602091828236039101136200150c57608010620014c8575060ff908180821614620014b3571660010162000b07565b5060248f634e487b7160e01b81526011600452fd5b6064906040519062461bcd60e51b82526004820152600d60248201527f6c696e6b20746f6f206c6f6e67000000000000000000000000000000000000006044820152fd5b508f80fd5b8f80fd5b606460405162461bcd60e51b815260206004820152600b60248201527f34206c696e6b73206d61780000000000000000000000000000000000000000006044820152fd5b606460405162461bcd60e51b815260206004820152601960248201527f696e76616c6964206e616d652f73796d626f6c2f696d616765000000000000006044820152fd5b50630100000185111562000af5565b50601282111562000aee565b8980fd5b8380fd5b5080fd5b5034620002855760203660031901126200028557620015e26200187e565b620015ec62001d0e565b62001629816001600160a01b03166000527f37c7a69cb64276d9952a1cf15ae5474311c0ddadaf138ab6ce1b4a3e73db5623602052604060002090565b906001600160a01b039081835416156200183a57811690813b15620015bc576040517fa69df4b5000000000000000000000000000000000000000000000000000000008152848160048183875af18015620005675762001824575b50604051927f5562b307000000000000000000000000000000000000000000000000000000008452826004850152620016e681548381166024870152620016d56044870160ff8360a01c1662001a42565b60ff606487019160a81c1662001a66565b8460608560848184305af18015620018175781958291620017cd575b508383541690303b15620005d85760405191630135129d60e41b83526004830152600260248301526044820152818160648183305af180156200028857620017af575b50927ff513c9a3b4ec035ce7af35221bbd0a6985eaba6fd7aa513ba76e0dfabb3daf329285620017a560a094546040519485528381166020860152620017946040860160ff83891c1662001a42565b60ff606086019160a81c1662001a66565b166080820152a180f35b620017be90949392946200193b565b620002a7579091843862001745565b955050506060843d82116200180e575b81620017ec606093836200196d565b81010312620002a757848451948386168603620015c057602001513862001702565b3d9150620017dd565b50604051903d90823e3d90fd5b62001832909491946200193b565b923862001684565b606460405162461bcd60e51b815260206004820152600d60248201527f696e76616c696420746f6b656e000000000000000000000000000000000000006044820152fd5b600435906001600160a01b03821682036200189557565b600080fd5b9181601f84011215620018955782359167ffffffffffffffff83116200189557602083818601950101116200189557565b9181601f84011215620018955782359167ffffffffffffffff831162001895576020808501948460051b0101116200189557565b608090600319011262001895576001600160a01b0390600435828116810362001895579160243590811681036200189557906044359060643590565b67ffffffffffffffff8111620013dc57604052565b6020810190811067ffffffffffffffff821117620013dc57604052565b90601f8019910116810190811067ffffffffffffffff821117620013dc57604052565b908060209392818452848401376000828201840152601f01601f1916010190565b9082818152602080910193818360051b82010194846000925b858410620019dc575050505050505090565b90919293949596601f198282030184528735601e19843603018112156200189557830186810191903567ffffffffffffffff811162001895578036038313620018955762001a308892839260019562001990565b990194019401929594939190620019ca565b90600182101562001a505752565b634e487b7160e01b600052602160045260246000fd5b90600282101562001a505752565b60005b83811062001a885750506000910152565b818101518382015260200162001a77565b81601f820112156200189557805167ffffffffffffffff8111620013dc576040519262001ad1601f8301601f1916602001856200196d565b81845260208284010111620018955762001af2916020808501910162001a74565b90565b9060209162001b108151809281855285808601910162001a74565b601f01601f1916010190565b9190820391821162001b2a57565b634e487b7160e01b600052601160045260246000fd5b909160009260ff62001b84846001600160a01b03166000527f37c7a69cb64276d9952a1cf15ae5474311c0ddadaf138ab6ce1b4a3e73db5623602052604060002090565b5460a01c16600181101562001c2b5762000638576001600160a01b03604051937f4765f669000000000000000000000000000000000000000000000000000000008552166004840152602483015215156044820152602081606481305afa9182156200102357809262001bf657505090565b9091506020823d821162001c22575b8162001c14602093836200196d565b810103126200028557505190565b3d915062001c05565b602485634e487b7160e01b81526021600452fd5b9190916040818403126200189557805192602082015167ffffffffffffffff8111620018955762001af2920162001a99565b1562001c7957565b606460405162461bcd60e51b815260206004820152601960248201527f616d6f756e74206f7574206c6f776572207468616e206d696e000000000000006044820152fd5b908160209103126200189557518015158103620018955790565b919262001af295949160a0946001600160a01b03809216855216602084015260408301526060820152816080820152019062001af5565b6001600160a01b037fc8fcad8db84d3cc18b4c41d551ea0ee66dd599cde068d998e57d5e09332c13215416330362001d4257565b606460405162461bcd60e51b815260206004820152601f60248201527f4c69624469616d6f6e643a204d75737420626520746865206469616d6f6e64006044820152fd5b519069ffffffffffffffffffff821682036200189557565b91909160409283519162001db28362001950565b60609283815285528094825192831562001ee657825193845180910195601f199384870151906605c284b9def77991828082061591040291828a1092602098888a8d84178401011691828a8c8a01169402951562001e6c575b5050505050505060019182811462001e5e575b805185018416835b62001e3c575b5050505050600090838301015252565b81810151878401820152840192831562001e5857928362001e26565b62001e2c565b955060008652809562001e1e565b84878d969798999d1491860101865118171562001ec857918893929184928651968c8801998a938901019052525b808301518a82870101520190811562001eb65790829062001e9a565b50505052933880808080808062001e0b565b50919894505081928689015287010190523880808080808062001e0b565b5090945050505056fe6040608081523462000a38576200251b803803806200001e8162000a3d565b92833961010082828101031262000a38576200003a8262000a63565b60208301519091906001600160401b03811162000a3857620000629082850190850162000ad8565b838501516001600160401b03811162000a3857620000869083860190860162000ad8565b60608501519093906001600160401b03811162000a3857620000ae9084870190870162000ad8565b60808601519091906001600160401b03811162000a38578601848701601f8201121562000a3857620000ea908588019060208151910162000a78565b60a0870151959093906001600160401b03871162000a3857858801601f888a0101121562000a385787870151956001600160401b038711620005ea578660051b60206200013981830162000a3d565b809981520180998b60208483878401930101011162000a38578b91602084838281870101955b0101018310620009f75750505050506200018160e060c08a0151990162000a63565b825190926001600160401b038211620005ea578190620001a360035462000af8565b601f8111620009a4575b50602090601f8311600114620009155760009262000909575b50508160011b916000199060031b1c1916176003555b8051906001600160401b038211620005ea578190620001fd60045462000af8565b601f8111620008b6575b50602090601f831160011462000827576000926200081b575b50508160011b916000199060031b1c1916176004555b600580546001600160a81b0319166001600160a01b0392831617600160a01b179055600680546001600160a01b031916929091169190911790558051906001600160401b038211620005ea5781906200029160075462000af8565b601f8111620007d9575b50602090601f83116001146200075f5760009262000753575b50508160011b916000199060031b1c1916176007555b8051906001600160401b038211620005ea578190620002eb60085462000af8565b601f811162000711575b50602090601f831160011462000697576000926200068b575b50508160011b916000199060031b1c1916176008555b5190680100000000000000008211620005ea576009548260095580831062000600575b5060096000908152600080516020620024fb83398151915292915b828210620004b95785853315620004755760055460ff8160a01c16620003f9575b50600254818101809111620003e35760025533600052600060205281600020818154019055815190815260007fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60203393a35161196c908162000b4f8239f35b634e487b7160e01b600052601160045260246000fd5b6001600160a01b031680159081156200046a575b50156200041b578262000383565b815162461bcd60e51b815260206004820152602260248201527f7472616e73666572206e6f7420616c6c6f776564206265666f7265206c61756e6044820152610c6d60f31b6064820152608490fd5b90503314836200040d565b815162461bcd60e51b815260206004820152601f60248201527f45524332303a206d696e7420746f20746865207a65726f2061646472657373006044820152606490fd5b80518051906001600160401b038211620005ea57620004d9865462000af8565b601f8111620005a8575b50602090601f83116001146200053457928260019493602093869560009262000528575b5050600019600383901b1c191690841b1787555b0194019101909262000362565b01519050388062000507565b908660005260206000209160005b601f19851681106200058f57509260209260019592869583601f1981161062000575575b505050831b830187556200051b565b015160001960f88460031b161c1916905538808062000566565b9192602060018192868501518155019401920162000542565b620005d890876000526020600020601f850160051c81019160208610620005df575b601f0160051c019062000b35565b38620004e3565b9091508190620005ca565b634e487b7160e01b600052604160045260246000fd5b6009600052600080516020620024fb8339815191529081019083015b8181106200062b575062000347565b806200063a6001925462000af8565b8062000649575b50016200061c565b601f81118314620006615750600081555b3862000641565b60009082825262000681601f60208420920160051c820185830162000b35565b818355556200065a565b0151905038806200030e565b60086000908152600080516020620024db8339815191529350601f198516905b818110620006f85750908460019594939210620006de575b505050811b0160085562000324565b015160001960f88460031b161c19169055388080620006cf565b92936020600181928786015181550195019301620006b7565b60086000526200074c90600080516020620024db833981519152601f850160051c81019160208610620005df57601f0160051c019062000b35565b38620002f5565b015190503880620002b4565b60076000908152600080516020620024bb8339815191529350601f198516905b818110620007c05750908460019594939210620007a6575b505050811b01600755620002ca565b015160001960f88460031b161c1916905538808062000797565b929360206001819287860151815501950193016200077f565b60076000526200081490600080516020620024bb833981519152601f850160051c81019160208610620005df57601f0160051c019062000b35565b386200029b565b01519050388062000220565b6004600090815293507f8a35acfbc15ff81a39ae7d344fd709f28e8600b4aa8c65c6b64bfe7fe36bd19b91905b601f19841685106200089a576001945083601f1981161062000880575b505050811b0160045562000236565b015160001960f88460031b161c1916905538808062000871565b8181015183556020948501946001909301929091019062000854565b600460005262000902907f8a35acfbc15ff81a39ae7d344fd709f28e8600b4aa8c65c6b64bfe7fe36bd19b601f850160051c81019160208610620005df57601f0160051c019062000b35565b3862000207565b015190503880620001c6565b6003600090815293507fc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b91905b601f198416851062000988576001945083601f198116106200096e575b505050811b01600355620001dc565b015160001960f88460031b161c191690553880806200095f565b8181015183556020948501946001909301929091019062000942565b6003600052620009f0907fc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b601f850160051c81019160208610620005df57601f0160051c019062000b35565b38620001ad565b8251936001600160401b03851162000a38578260208f96819462000a268392838c8c0191878d01010162000ad8565b8152019501949291959086916200015f565b600080fd5b6040519190601f01601f191682016001600160401b03811183821017620005ea57604052565b51906001600160a01b038216820362000a3857565b919291906001600160401b038111620005ea5760209162000aa2601f8301601f1916840162000a3d565b948286528282011162000a385760005b82811062000ac65750509060009184010152565b81810151868201850152830162000ab2565b9080601f8301121562000a3857815162000af59260200162000a78565b90565b90600182811c9216801562000b2a575b602083101462000b1457565b634e487b7160e01b600052602260045260246000fd5b91607f169162000b08565b81811062000b41575050565b6000815560010162000b3556fe608080604052600436101561001357600080fd5b60003560e01c90816306fdde031461134557508063095ea7b31461131f57806318160ddd1461130157806323b872dd14611243578063313ce5671461122757806339509351146111d457806370a082311461119a57806395d89b411461116f578063a457c2d7146110b0578063a69df4b51461104c578063a9059cbb1461101b578063c06d4e0414610a8d578063c87b56dd146101105763dd62ed3e146100b957600080fd5b3461010b57604036600319011261010b576100d26113b6565b6100da6113cc565b906001600160a01b038091166000526001602052604060002091166000526020526020604060002054604051908152f35b600080fd5b3461010b57602036600319011261010b576040516020810181811067ffffffffffffffff821117610a775760405260008152600090600954915b8281106108d557506006546040519060808201604052600f6f3031323334353637383961626364656681527fffffffffffffffffffffffffffffffffffffffff0000000000000000000000006002840192602884526000604a86015260601b1690600160005b8080018601602285831a85811651602384015360041c5191015301601481146101db576001906101b0565b50505061307860028251019152815260405190610202826101fb8161144a565b0383611575565b6040519261021a84610213816114ee565b0385611575565b670de0b6b3a76400006002540490604051608081019260a0820160405260008452925b6000190192600a90603082820601855304928361023d5790819003608001601f1982015260405160085490935083600061027683611410565b80835292600181169081156108b65750600114610855575b61029a92500384611575565b82516060938082610760575b5050506040519586947f7b0000000000000000000000000000000000000000000000000000000000000060208701527f2263726561746f72223a22000000000000000000000000000000000000000000602187015261030f815180926020602c8a019101611367565b8501957f222c0000000000000000000000000000000000000000000000000000000000009687602c8201527f226e616d65223a22000000000000000000000000000000000000000000000000602e820152610374825180936020603685019101611367565b018660368201527f2273796d626f6c223a220000000000000000000000000000000000000000000060388201526103b5825180936020604285019101611367565b018560428201527f22737570706c79223a220000000000000000000000000000000000000000000060448201526103f7601f198301518093604e840190611367565b0184604e8201527f226465736372697074696f6e223a22000000000000000000000000000000000060508201526000906007549061043482611410565b916001811690811561073b57506001146106df575b50509184918361057a9660509695527f226c696e6b73223a5b00000000000000000000000000000000000000000000006002820152610492825180936020600b85019101611367565b017f5d2c000000000000000000000000000000000000000000000000000000000000600b8201527f22696d6167655f64617461223a22646174613a696d6167652f776562703b6261600d8201527f736536342c000000000000000000000000000000000000000000000000000000602d820152610519825180936020603285019101611367565b019060328201527f226261636b67726f756e645f636f6c6f72223a2230303030303022000000000060348201527f7d00000000000000000000000000000000000000000000000000000000000000604f820152036030810184520182611575565b806060908251806105ed575b6105e9836105dd603d60405180937f646174613a6170706c69636174696f6e2f6a736f6e3b6261736536342c00000060208301526105cd8151809260208686019101611367565b810103601d810184520182611575565b6040519182918261138a565b0390f35b9091506003600282010460021b90604051927f4142434445464748494a4b4c4d4e4f505152535455565758595a616263646566601f526106707f6768696a6b6c6d6e6f707172737475767778797a303132333435363738392d5f18603f526020840160046003602085890101519360006020878b0101525b0191603f8351818160121c16516000538181600c1c1651600153818160061c16516002531651600353600051815201906020858701018210156106ad57600490600390610665565b5090600383613d3d60f01b9260206000966105e99a010152604086880101604052066002048203525281528280610586565b90915060076000527fa66cc928b5edb82af9bd49922954155ab7b0942694bea4ce44661d9a8736c6886000905b83821061072157505001605f01816050610449565b8054605f838501015288965060209091019060010161070c565b60ff1916605f8381019190915283151590930290910190910191508290506050610449565b90919450604051947f4142434445464748494a4b4c4d4e4f505152535455565758595a616263646566601f526106707f6768696a6b6c6d6e6f707172737475767778797a303132333435363738392d5f18603f5260208601602082816003600283010460021b8a01019501019060046003835195600085525b0191603f8351818160121c16516000538181600c1c1651600153818160061c16516002531651600353600051815201908582101561081c576004906003906107d9565b50600394506002936000925260408585850104851b890101604052613d3d60f01b858406850482035252010460021b83528680806102a6565b509060086000527ff3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee3906000915b81831061089a57505090602061029a9282010161028e565b6020919350806001915483858a01015201910190918592610882565b6020925061029a94915060ff191682840152151560051b82010161028e565b6009600052604051826108f2829451809260208086019101611367565b601160f91b60208284010152600090837f6e1540171b6c0c960b71a7020d9f60077f6af931a8bbf590da0223dacf75c7af019081549161093183611410565b9260018116908115610a4c5750600114610a03575b5050506109669181601160f91b6001935203601e19810185520183611575565b818360001981011161099057600019840182106109a6575b5060001981146109905760010161014a565b634e487b7160e01b600052601160045260246000fd5b9091506109fc6021604051836109c6829551809260208086019101611367565b81017f2c000000000000000000000000000000000000000000000000000000000000006020820152036001810184520182611575565b908361097e565b919350915060005260206000206000905b838210610a335750508360216109669360019383010101918193610946565b8054602184880184010152602090910190600101610a14565b9291505060019350610966949260219260ff1916838386010152801515029083010101918193610946565b634e487b7160e01b600052604160045260246000fd5b3461010b57606036600319011261010b5767ffffffffffffffff60043581811161010b57610abf9036906004016113e2565b91602490813581811161010b57610ada9036906004016113e2565b909360449182359584871161010b573660238801121561010b5786600401359785891161010b5760059336888b871b8b01011161010b576001600160a01b0360065416330361010b5781610ec9575b505081610d83575b5050600954808703610b3f57005b680100000000000000008711610d6e5786600955808710610cdd575b5083859695019360096000527f6e1540171b6c0c960b71a7020d9f60077f6af931a8bbf590da0223dacf75c7af9460009760421981360301915b888a10610b9e57005b80358381121561010b578201848101359088821161010b578136038882011361010b578790610bcd8b54611410565b90601f91828111610ca3575b506000918411600114610c27579280602093600196938796600093610c1a575b505050600019600383901b1c191690841b178b555b01980199019896610b95565b01013590508a3880610bf9565b60008c8152602080822094929391601f198716915b828210610c84575050928592602095926001989597899810610c68575b50505050831b83018b55610c0e565b60001960f88660031b161c19920101351690558e808b81610c59565b600184978294959683949789010135815501960193018c939291610c3c565b610cce908d6000526020908482600020918d828a01901c8301938910610cd4575b018c1c019061191f565b8e610bd9565b92508192610cc4565b6009600052867f6e1540171b6c0c960b71a7020d9f60077f6af931a8bbf590da0223dacf75c7af91820191015b818110610d175750610b5b565b80610d2460019254611410565b80610d31575b5001610d0a565b601f908181118414610d4a575050600081555b89610d2a565b610d65600092848452602084209201871c820185830161191f565b81835555610d44565b84634e487b7160e01b60005260416004526000fd5b848211610eb4578190610d97600854611410565b601f8111610e61575b50600090601f8311600114610ddf57600092610dd4575b50508160011b916000199060031b1c1916176008555b8680610b31565b013590508880610db7565b7ff3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee3925090601f19841660005b818110610e495750908460019594939210610e2f575b505050811b01600855610dcd565b0135600019600384901b60f8161c19169055888080610e21565b91936020600181928787013581550195019201610e0b565b610ea4907ff3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee3601f8501871c81019160208610610eaa575b601f01871c019061191f565b89610da0565b9091508190610e98565b85634e487b7160e01b60005260416004526000fd5b868211611006578190610edd600754611410565b601f8111610fae575b50600090601f8311600114610f2557600092610f1a575b50508160011b916000199060031b1c1916176007555b8880610b29565b013590508a80610efd565b909150601f1983169160076000527fa66cc928b5edb82af9bd49922954155ab7b0942694bea4ce44661d9a8736c6889260005b818110610f965750908460019594939210610f7c575b505050811b01600755610f13565b0135600019600384901b60f8161c191690558a8080610f6e565b91936020600181928787013581550195019201610f58565b610ff69060076000527fa66cc928b5edb82af9bd49922954155ab7b0942694bea4ce44661d9a8736c688601f8501891c81019160208610610ffc575b601f01891c019061191f565b8b610ee6565b9091508190610fea565b87634e487b7160e01b60005260416004526000fd5b3461010b57604036600319011261010b576110416110376113b6565b6024359033611597565b602060405160018152f35b3461010b57600036600319011261010b576005546001600160a01b03811633148061109e575b1561010b577fffffffffffffffffffffff00ffffffffffffffffffffffffffffffffffffffff16600555005b50600160ff8260a01c16151514611072565b3461010b57604036600319011261010b576110c96113b6565b6024359033600052600160205260406000206001600160a01b0382166000526020526040600020549180831061110557611041920390336117eb565b608460405162461bcd60e51b815260206004820152602560248201527f45524332303a2064656372656173656420616c6c6f77616e63652062656c6f7760448201527f207a65726f0000000000000000000000000000000000000000000000000000006064820152fd5b3461010b57600036600319011261010b576105e96040516105dd81611193816114ee565b0382611575565b3461010b57602036600319011261010b576001600160a01b036111bb6113b6565b1660005260006020526020604060002054604051908152f35b3461010b57604036600319011261010b576111ed6113b6565b33600052600160205260406000206001600160a01b03821660005260205260406000205460243581018091116109905761104191336117eb565b3461010b57600036600319011261010b57602060405160128152f35b3461010b57606036600319011261010b5761125c6113b6565b6112646113cc565b604435906001600160a01b03831660005260016020526040600020336000526020526040600020549260001984036112a1575b6110419350611597565b8284106112bd576112b883611041950333836117eb565b611297565b606460405162461bcd60e51b815260206004820152601d60248201527f45524332303a20696e73756666696369656e7420616c6c6f77616e63650000006044820152fd5b3461010b57600036600319011261010b576020600254604051908152f35b3461010b57604036600319011261010b5761104161133b6113b6565b60243590336117eb565b3461010b57600036600319011261010b57806105dd816111936105e99461144a565b60005b83811061137a5750506000910152565b818101518382015260200161136a565b604091602082526113aa8151809281602086015260208686019101611367565b601f01601f1916010190565b600435906001600160a01b038216820361010b57565b602435906001600160a01b038216820361010b57565b9181601f8401121561010b5782359167ffffffffffffffff831161010b576020838186019501011161010b57565b90600182811c92168015611440575b602083101461142a57565b634e487b7160e01b600052602260045260246000fd5b91607f169161141f565b6003546000929161145a82611410565b808252916001908181169081156114d1575060011461147857505050565b9192935060036000527fc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b916000925b8484106114b957505060209250010190565b805460208585018101919091529093019281016114a7565b915050602093945060ff929192191683830152151560051b010190565b600454600092916114fe82611410565b808252916001908181169081156114d1575060011461151c57505050565b9192935060046000527f8a35acfbc15ff81a39ae7d344fd709f28e8600b4aa8c65c6b64bfe7fe36bd19b916000925b84841061155d57505060209250010190565b8054602085850181019190915290930192810161154b565b90601f8019910116810190811067ffffffffffffffff821117610a7757604052565b916001600160a01b038093169182156117815783169283156117175760055460ff8160a01c1661168a575b505060008281528060205260408120549180831061162057604082827fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef958760209652828652038282205586815220818154019055604051908152a3565b608460405162461bcd60e51b815260206004820152602660248201527f45524332303a207472616e7366657220616d6f756e742065786365656473206260448201527f616c616e636500000000000000000000000000000000000000000000000000006064820152fd5b1680831490811561170d575b50156116a35738806115c2565b608460405162461bcd60e51b815260206004820152602260248201527f7472616e73666572206e6f7420616c6c6f776564206265666f7265206c61756e60448201527f63680000000000000000000000000000000000000000000000000000000000006064820152fd5b9050831438611696565b608460405162461bcd60e51b815260206004820152602360248201527f45524332303a207472616e7366657220746f20746865207a65726f206164647260448201527f65737300000000000000000000000000000000000000000000000000000000006064820152fd5b608460405162461bcd60e51b815260206004820152602560248201527f45524332303a207472616e736665722066726f6d20746865207a65726f20616460448201527f64726573730000000000000000000000000000000000000000000000000000006064820152fd5b6001600160a01b038091169182156118b6571691821561184c5760207f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b925918360005260018252604060002085600052825280604060002055604051908152a3565b608460405162461bcd60e51b815260206004820152602260248201527f45524332303a20617070726f766520746f20746865207a65726f20616464726560448201527f73730000000000000000000000000000000000000000000000000000000000006064820152fd5b608460405162461bcd60e51b8152602060048201526024808201527f45524332303a20617070726f76652066726f6d20746865207a65726f2061646460448201527f72657373000000000000000000000000000000000000000000000000000000006064820152fd5b81811061192a575050565b6000815560010161191f56fea26469706673582212206f0073e94722fc733abd696a5db6d427088ca4421acd011a0c52951face0c40064736f6c63430008120033a66cc928b5edb82af9bd49922954155ab7b0942694bea4ce44661d9a8736c688f3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee36e1540171b6c0c960b71a7020d9f60077f6af931a8bbf590da0223dacf75c7afa2646970667358221220bd5ff7a481c323ccdb753746c981005cd2ba8bbca792558150e591f71d633da664736f6c63430008120033
Deployed Bytecode
0x608060405260043610156200001357600080fd5b6000803560e01c8063214013ca14620015c45780632bafa99314620009c15780636a272462146200093d57806392f87d7b14620008f85780639eed896e1462000691578063bad6b832146200033c578063cce7ec1314620002af5763d2fc17fe146200007e57600080fd5b61012036600319011262000285578067ffffffffffffffff600435818111620002ab57620000b19036906004016200189a565b9190602435828111620002a757620000ce9036906004016200189a565b939092604435818111620002a357620000ec9036906004016200189a565b916064358181116200029f57620001089036906004016200189a565b906084358381116200029b5762000124903690600401620018cb565b92909360a4359081116200029757620001429036906004016200189a565b959096600160c43510156200029357600260e43510156200029357303b1562000293578c9a6040519c8d9b8c9b7f2bafa993000000000000000000000000000000000000000000000000000000008d528c33906004015260248d0161016090526101648d0190620001b39262001990565b906003198c83030160448d0152620001cb9262001990565b906003198a83030160648b0152620001e39262001990565b90600319888303016084890152620001fb9262001990565b906003198683030160a48701526200021392620019b1565b906003198483030160c48501526200022b9262001990565b6200023c60e4830160c43562001a42565b6101046200024f83820160e43562001a66565b3561012483015234610144830152038183305af180156200028857620002725750f35b6200027d906200193b565b620002855780f35b80fd5b6040513d84823e3d90fd5b8c80fd5b8b80fd5b8a80fd5b8880fd5b8680fd5b8480fd5b5050fd5b506040366003190112620002855780620002c86200187e565b303b1562000339576040517f9eed896e0000000000000000000000000000000000000000000000000000000081523360048201526001600160a01b03909116602480830191909152346044830152356064820152818180608481015b038183305af180156200028857620002725750f35b50fd5b503462000285576200034e36620018ff565b6200035b93919362001d0e565b60ff6200039a846001600160a01b03166000527f37c7a69cb64276d9952a1cf15ae5474311c0ddadaf138ab6ce1b4a3e73db5623602052604060002090565b5460a01c169060405191620003af8362001950565b604051620003bd8162001950565b878152835260018110156200067d5762000638576040517ffab021b70000000000000000000000000000000000000000000000000000000081526001600160a01b03851660048201526024810186905290868260448183305af19182156200062d578790889362000600575b506200043790928462001d9e565b5080620005eb575b507fd6636d7e2aa9fa70e65f3eb96671972c70cfa6486eb903d39a283c1e07a1c5986200046e82825462001b1c565b9055303b15620005e75785604051630135129d60e41b81526001600160a01b038086169081600484015260016024840152846044840152838360648183305af1928315620005dc578493620005bc575b506020906064604051809581937f23b872dd0000000000000000000000000000000000000000000000000000000083528660048401523060248401528d60448401528c165af1918215620005b15783928392839287926200057c575b5082821562000572575bf11562000567577f76afc58d2c47c2160750cec6375a6cc87ab5a99635e1c481f956b578e023f99b94620005619251916040519586958662001cd7565b0390a180f35b6040513d87823e3d90fd5b506108fc62000524565b620005a19060203d8111620005a9575b6200059881836200196d565b81019062001cbd565b50386200051a565b503d6200058c565b6040513d85823e3d90fd5b620005ca909391936200193b565b620005d857829138620004be565b8280fd5b6040513d86823e3d90fd5b8580fd5b620005f99082101562001c71565b386200043f565b6200043793506200062691503d808a833e6200061d81836200196d565b81019062001c3f565b9262000429565b6040513d89823e3d90fd5b60405162461bcd60e51b815260206004820152601060248201527f696e76616c6964207374726174656779000000000000000000000000000000006044820152606490fd5b602487634e487b7160e01b81526021600452fd5b50346200028557620006a336620018ff565b9290620006af62001d0e565b60ff620006ee846001600160a01b03166000527f37c7a69cb64276d9952a1cf15ae5474311c0ddadaf138ab6ce1b4a3e73db5623602052604060002090565b5460a01c169060405191620007038362001950565b604051620007118162001950565b878152835260018110156200067d57620006385760405194633960fa3760e01b86526001600160a01b0385166004870152868660248185305af19586156200062d5787908897620008d4575b506200076b90968462001d9e565b5080620008bf575b507fd6636d7e2aa9fa70e65f3eb96671972c70cfa6486eb903d39a283c1e07a1c598805490828201809211620008ab5755303b15620005e757604051630135129d60e41b81526001600160a01b038416600482015260006024820152604481018290528690818160648183305af18015620002885762000893575b505060405163a9059cbb60e01b81526001600160a01b03841660048201526024810186905294602086806044810103818a6001600160a01b038a165af19283156200062d577f4c451962b8f0dd56d788474a5f794da140e33218eee65f7ee9a9a8a363eaf8c096620005619462000870575b5051916040519586958662001cd7565b6200088b9060203d8111620005a9576200059881836200196d565b503862000860565b6200089e906200193b565b620005e7578538620007ee565b602488634e487b7160e01b81526011600452fd5b620008cd9086101562001c71565b3862000773565b6200076b9750620008f191503d808a833e6200061d81836200196d565b966200075d565b5034620002855760603660031901126200028557620009166200187e565b6044359182151583036200028557602062000935846024358562001b40565b604051908152f35b5034620002855760603660031901126200028557806200095c6200187e565b303b1562000339576040517fbad6b8320000000000000000000000000000000000000000000000000000000081523360048201526001600160a01b03909116602480830191909152356044808301919091523560648201528181806084810162000324565b506101603660031901126200028557620009da6200187e565b9060243567ffffffffffffffff8111620015c057620009fe9036906004016200189a565b909260443567ffffffffffffffff8111620015bc5762000a239036906004016200189a565b909260643567ffffffffffffffff8111620005e75762000a489036906004016200189a565b969060843567ffffffffffffffff81116200143d5762000a6d9036906004016200189a565b9060a43567ffffffffffffffff8111620015b85762000a91903690600401620018cb565b92909360c43567ffffffffffffffff8111620002975762000ab79036906004016200189a565b9c9098600160e435101562000293576002610104351015620002935762000add62001d0e565b610144359b60128a111580620015ac575b806200159d575b156200155957600587101562001515578d5b8760ff8216106200145657507fb08236d4879334bece1965ab3a812fad8e40dec4929b65f11b873dfaf386da8954986040519a8b61251b81011067ffffffffffffffff8d61251b01111762001441579362000bbe8c8f9562000bd59562000ba7839f9e9c999662000bec9b9762000c039f9d9a6001600160a01b039061251b62001ef089391661251b870190815261010060208201819052019162001990565b92604061251b820185039161251b01015262001990565b9161251b8c01830360608d61251b01015262001990565b9161251b89018303608061251b8b01015262001990565b9161251b8601830360a061251b88010152620019b1565b9060c061251b840101523060e061251b84010152039085f0948515620005dc5760405162000c318162001950565b60405162000c3f8162001950565b868152815260e43562000638577fb08236d4879334bece1965ab3a812fad8e40dec4929b65f11b873dfaf386da8954604051907f327dca720000000000000000000000000000000000000000000000000000000082526001600160a01b038916600483015260248201526060604482015286818062000cc360648201878962001990565b038183305af180156200062d578790620013f2575b62000ce591508262001d9e565b5060405180606081011067ffffffffffffffff606083011117620013dc57606081016040526001600160a01b03851681526020810160e4358152604082016101043581526001600160a01b0362000d70818c166001600160a01b03166000527f37c7a69cb64276d9952a1cf15ae5474311c0ddadaf138ab6ce1b4a3e73db5623602052604060002090565b93511683549251916001831015620013c857516002811015620013c8579174ff00000000000000000000000000000000000000007fe28e4635d390f60ea0f9513a16dc98c2ee9b4a0762ad677fb000b7f7cb07625198979694927fffffffffffffffffffff0000000000000000000000000000000000000000000075ff00000000000000000000000000000000000000000062000e82989660a81b169416179160a01b16171790555162000e736040519485946001600160a01b038c1686526001600160a01b038916602087015262000e4f6040870160e43562001a42565b62000e61606087016101043562001a66565b60c0608087015260c086019162001990565b9083820360a085015262001af5565b0390a160047fb08236d4879334bece1965ab3a812fad8e40dec4929b65f11b873dfaf386da875460a06001600160a01b037f88acb08f71068c878012515a2672f2b1a3f7940ce7bd383fbab30a631a76b7745416604051938480927ffeaf968c0000000000000000000000000000000000000000000000000000000082525afa80156200056757859286916200136a575b5062000f417f88acb08f71068c878012515a2672f2b1a3f7940ce7bd383fbab30a631a76b775544262001b1c565b1162001326576402540be4008083029280840482148115171562001312577812725dd1d243aba0e75fe645cc4873f9e65afe688c928e1f2283100202156200130557670de0b6b3a7640000020491828110620012c15783808062000fa786829562001b1c565b956001600160a01b037fb08236d4879334bece1965ab3a812fad8e40dec4929b65f11b873dfaf386da865416828215620012b7575bf115620002885761012435806200103a575b5082935081159081156200100157505050f35b83928392839283916200102f575b6001600160a01b031690f115620010235780f35b604051903d90823e3d90fd5b6108fc91506200100f565b6200104792919262001d0e565b60ff6200108f6001600160a01b0387166001600160a01b03166000527f37c7a69cb64276d9952a1cf15ae5474311c0ddadaf138ab6ce1b4a3e73db5623602052604060002090565b5460a01c169160405192620010a48462001950565b604051620010b28162001950565b86815284526001811015620012a357620006385760405192633960fa3760e01b84526001600160a01b0387166004850152858460248186305af193841562001298578690879562001274575b506200110c90948262001d9e565b507fd6636d7e2aa9fa70e65f3eb96671972c70cfa6486eb903d39a283c1e07a1c598968754848101809111620008ab57879855303b15620002a357604051630135129d60e41b81526001600160a01b03871660048201526000602482015260448101859052878160648183305af1908115620012515788916200125c575b505060405163a9059cbb60e01b81526001600160a01b03871660048201526024810186905290602082806044810103818b6001600160a01b0386165af18015620012515762001225967f4c451962b8f0dd56d788474a5f794da140e33218eee65f7ee9a9a8a363eaf8c0946001600160a01b03946200121c936200122d575b50519087604051958695168b8662001cd7565b0390a162001b1c565b903862000fee565b620012499060203d602011620005a9576200059881836200196d565b503862001209565b6040513d8a823e3d90fd5b62001267906200193b565b620002a35786386200118a565b6200110c95506200129191503d8089833e6200061d81836200196d565b94620010fe565b6040513d88823e3d90fd5b602486634e487b7160e01b81526021600452fd5b506108fc62000fdc565b606460405162461bcd60e51b815260206004820152601160248201527f757364207072696365206368616e6765640000000000000000000000000000006044820152fd5b637c5f487d85526004601cfd5b602487634e487b7160e01b81526011600452fd5b606460405162461bcd60e51b815260206004820152600f60248201527f7374616c652070726963656665656400000000000000000000000000000000006044820152fd5b92505060a0823d60a011620013bf575b816200138960a093836200196d565b81010312620002a7576200139d8262001d86565b506020820151620013b660806040850151940162001d86565b50913862000f13565b3d91506200137a565b60248b634e487b7160e01b81526021600452fd5b634e487b7160e01b600052604160045260246000fd5b503d8088833e6200140481836200196d565b8101906020818303126200143d5780519067ffffffffffffffff82116200029f5791620014379162000ce5930162001a99565b62000cd8565b8780fd5b5060248f634e487b7160e01b81526041600452fd5b611fe08160051b16890135601e198a36030181121562001511578901803567ffffffffffffffff81116200150c57602091828236039101136200150c57608010620014c8575060ff908180821614620014b3571660010162000b07565b5060248f634e487b7160e01b81526011600452fd5b6064906040519062461bcd60e51b82526004820152600d60248201527f6c696e6b20746f6f206c6f6e67000000000000000000000000000000000000006044820152fd5b508f80fd5b8f80fd5b606460405162461bcd60e51b815260206004820152600b60248201527f34206c696e6b73206d61780000000000000000000000000000000000000000006044820152fd5b606460405162461bcd60e51b815260206004820152601960248201527f696e76616c6964206e616d652f73796d626f6c2f696d616765000000000000006044820152fd5b50630100000185111562000af5565b50601282111562000aee565b8980fd5b8380fd5b5080fd5b5034620002855760203660031901126200028557620015e26200187e565b620015ec62001d0e565b62001629816001600160a01b03166000527f37c7a69cb64276d9952a1cf15ae5474311c0ddadaf138ab6ce1b4a3e73db5623602052604060002090565b906001600160a01b039081835416156200183a57811690813b15620015bc576040517fa69df4b5000000000000000000000000000000000000000000000000000000008152848160048183875af18015620005675762001824575b50604051927f5562b307000000000000000000000000000000000000000000000000000000008452826004850152620016e681548381166024870152620016d56044870160ff8360a01c1662001a42565b60ff606487019160a81c1662001a66565b8460608560848184305af18015620018175781958291620017cd575b508383541690303b15620005d85760405191630135129d60e41b83526004830152600260248301526044820152818160648183305af180156200028857620017af575b50927ff513c9a3b4ec035ce7af35221bbd0a6985eaba6fd7aa513ba76e0dfabb3daf329285620017a560a094546040519485528381166020860152620017946040860160ff83891c1662001a42565b60ff606086019160a81c1662001a66565b166080820152a180f35b620017be90949392946200193b565b620002a7579091843862001745565b955050506060843d82116200180e575b81620017ec606093836200196d565b81010312620002a757848451948386168603620015c057602001513862001702565b3d9150620017dd565b50604051903d90823e3d90fd5b62001832909491946200193b565b923862001684565b606460405162461bcd60e51b815260206004820152600d60248201527f696e76616c696420746f6b656e000000000000000000000000000000000000006044820152fd5b600435906001600160a01b03821682036200189557565b600080fd5b9181601f84011215620018955782359167ffffffffffffffff83116200189557602083818601950101116200189557565b9181601f84011215620018955782359167ffffffffffffffff831162001895576020808501948460051b0101116200189557565b608090600319011262001895576001600160a01b0390600435828116810362001895579160243590811681036200189557906044359060643590565b67ffffffffffffffff8111620013dc57604052565b6020810190811067ffffffffffffffff821117620013dc57604052565b90601f8019910116810190811067ffffffffffffffff821117620013dc57604052565b908060209392818452848401376000828201840152601f01601f1916010190565b9082818152602080910193818360051b82010194846000925b858410620019dc575050505050505090565b90919293949596601f198282030184528735601e19843603018112156200189557830186810191903567ffffffffffffffff811162001895578036038313620018955762001a308892839260019562001990565b990194019401929594939190620019ca565b90600182101562001a505752565b634e487b7160e01b600052602160045260246000fd5b90600282101562001a505752565b60005b83811062001a885750506000910152565b818101518382015260200162001a77565b81601f820112156200189557805167ffffffffffffffff8111620013dc576040519262001ad1601f8301601f1916602001856200196d565b81845260208284010111620018955762001af2916020808501910162001a74565b90565b9060209162001b108151809281855285808601910162001a74565b601f01601f1916010190565b9190820391821162001b2a57565b634e487b7160e01b600052601160045260246000fd5b909160009260ff62001b84846001600160a01b03166000527f37c7a69cb64276d9952a1cf15ae5474311c0ddadaf138ab6ce1b4a3e73db5623602052604060002090565b5460a01c16600181101562001c2b5762000638576001600160a01b03604051937f4765f669000000000000000000000000000000000000000000000000000000008552166004840152602483015215156044820152602081606481305afa9182156200102357809262001bf657505090565b9091506020823d821162001c22575b8162001c14602093836200196d565b810103126200028557505190565b3d915062001c05565b602485634e487b7160e01b81526021600452fd5b9190916040818403126200189557805192602082015167ffffffffffffffff8111620018955762001af2920162001a99565b1562001c7957565b606460405162461bcd60e51b815260206004820152601960248201527f616d6f756e74206f7574206c6f776572207468616e206d696e000000000000006044820152fd5b908160209103126200189557518015158103620018955790565b919262001af295949160a0946001600160a01b03809216855216602084015260408301526060820152816080820152019062001af5565b6001600160a01b037fc8fcad8db84d3cc18b4c41d551ea0ee66dd599cde068d998e57d5e09332c13215416330362001d4257565b606460405162461bcd60e51b815260206004820152601f60248201527f4c69624469616d6f6e643a204d75737420626520746865206469616d6f6e64006044820152fd5b519069ffffffffffffffffffff821682036200189557565b91909160409283519162001db28362001950565b60609283815285528094825192831562001ee657825193845180910195601f199384870151906605c284b9def77991828082061591040291828a1092602098888a8d84178401011691828a8c8a01169402951562001e6c575b5050505050505060019182811462001e5e575b805185018416835b62001e3c575b5050505050600090838301015252565b81810151878401820152840192831562001e5857928362001e26565b62001e2c565b955060008652809562001e1e565b84878d969798999d1491860101865118171562001ec857918893929184928651968c8801998a938901019052525b808301518a82870101520190811562001eb65790829062001e9a565b50505052933880808080808062001e0b565b50919894505081928689015287010190523880808080808062001e0b565b5090945050505056fe6040608081523462000a38576200251b803803806200001e8162000a3d565b92833961010082828101031262000a38576200003a8262000a63565b60208301519091906001600160401b03811162000a3857620000629082850190850162000ad8565b838501516001600160401b03811162000a3857620000869083860190860162000ad8565b60608501519093906001600160401b03811162000a3857620000ae9084870190870162000ad8565b60808601519091906001600160401b03811162000a38578601848701601f8201121562000a3857620000ea908588019060208151910162000a78565b60a0870151959093906001600160401b03871162000a3857858801601f888a0101121562000a385787870151956001600160401b038711620005ea578660051b60206200013981830162000a3d565b809981520180998b60208483878401930101011162000a38578b91602084838281870101955b0101018310620009f75750505050506200018160e060c08a0151990162000a63565b825190926001600160401b038211620005ea578190620001a360035462000af8565b601f8111620009a4575b50602090601f8311600114620009155760009262000909575b50508160011b916000199060031b1c1916176003555b8051906001600160401b038211620005ea578190620001fd60045462000af8565b601f8111620008b6575b50602090601f831160011462000827576000926200081b575b50508160011b916000199060031b1c1916176004555b600580546001600160a81b0319166001600160a01b0392831617600160a01b179055600680546001600160a01b031916929091169190911790558051906001600160401b038211620005ea5781906200029160075462000af8565b601f8111620007d9575b50602090601f83116001146200075f5760009262000753575b50508160011b916000199060031b1c1916176007555b8051906001600160401b038211620005ea578190620002eb60085462000af8565b601f811162000711575b50602090601f831160011462000697576000926200068b575b50508160011b916000199060031b1c1916176008555b5190680100000000000000008211620005ea576009548260095580831062000600575b5060096000908152600080516020620024fb83398151915292915b828210620004b95785853315620004755760055460ff8160a01c16620003f9575b50600254818101809111620003e35760025533600052600060205281600020818154019055815190815260007fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60203393a35161196c908162000b4f8239f35b634e487b7160e01b600052601160045260246000fd5b6001600160a01b031680159081156200046a575b50156200041b578262000383565b815162461bcd60e51b815260206004820152602260248201527f7472616e73666572206e6f7420616c6c6f776564206265666f7265206c61756e6044820152610c6d60f31b6064820152608490fd5b90503314836200040d565b815162461bcd60e51b815260206004820152601f60248201527f45524332303a206d696e7420746f20746865207a65726f2061646472657373006044820152606490fd5b80518051906001600160401b038211620005ea57620004d9865462000af8565b601f8111620005a8575b50602090601f83116001146200053457928260019493602093869560009262000528575b5050600019600383901b1c191690841b1787555b0194019101909262000362565b01519050388062000507565b908660005260206000209160005b601f19851681106200058f57509260209260019592869583601f1981161062000575575b505050831b830187556200051b565b015160001960f88460031b161c1916905538808062000566565b9192602060018192868501518155019401920162000542565b620005d890876000526020600020601f850160051c81019160208610620005df575b601f0160051c019062000b35565b38620004e3565b9091508190620005ca565b634e487b7160e01b600052604160045260246000fd5b6009600052600080516020620024fb8339815191529081019083015b8181106200062b575062000347565b806200063a6001925462000af8565b8062000649575b50016200061c565b601f81118314620006615750600081555b3862000641565b60009082825262000681601f60208420920160051c820185830162000b35565b818355556200065a565b0151905038806200030e565b60086000908152600080516020620024db8339815191529350601f198516905b818110620006f85750908460019594939210620006de575b505050811b0160085562000324565b015160001960f88460031b161c19169055388080620006cf565b92936020600181928786015181550195019301620006b7565b60086000526200074c90600080516020620024db833981519152601f850160051c81019160208610620005df57601f0160051c019062000b35565b38620002f5565b015190503880620002b4565b60076000908152600080516020620024bb8339815191529350601f198516905b818110620007c05750908460019594939210620007a6575b505050811b01600755620002ca565b015160001960f88460031b161c1916905538808062000797565b929360206001819287860151815501950193016200077f565b60076000526200081490600080516020620024bb833981519152601f850160051c81019160208610620005df57601f0160051c019062000b35565b386200029b565b01519050388062000220565b6004600090815293507f8a35acfbc15ff81a39ae7d344fd709f28e8600b4aa8c65c6b64bfe7fe36bd19b91905b601f19841685106200089a576001945083601f1981161062000880575b505050811b0160045562000236565b015160001960f88460031b161c1916905538808062000871565b8181015183556020948501946001909301929091019062000854565b600460005262000902907f8a35acfbc15ff81a39ae7d344fd709f28e8600b4aa8c65c6b64bfe7fe36bd19b601f850160051c81019160208610620005df57601f0160051c019062000b35565b3862000207565b015190503880620001c6565b6003600090815293507fc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b91905b601f198416851062000988576001945083601f198116106200096e575b505050811b01600355620001dc565b015160001960f88460031b161c191690553880806200095f565b8181015183556020948501946001909301929091019062000942565b6003600052620009f0907fc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b601f850160051c81019160208610620005df57601f0160051c019062000b35565b38620001ad565b8251936001600160401b03851162000a38578260208f96819462000a268392838c8c0191878d01010162000ad8565b8152019501949291959086916200015f565b600080fd5b6040519190601f01601f191682016001600160401b03811183821017620005ea57604052565b51906001600160a01b038216820362000a3857565b919291906001600160401b038111620005ea5760209162000aa2601f8301601f1916840162000a3d565b948286528282011162000a385760005b82811062000ac65750509060009184010152565b81810151868201850152830162000ab2565b9080601f8301121562000a3857815162000af59260200162000a78565b90565b90600182811c9216801562000b2a575b602083101462000b1457565b634e487b7160e01b600052602260045260246000fd5b91607f169162000b08565b81811062000b41575050565b6000815560010162000b3556fe608080604052600436101561001357600080fd5b60003560e01c90816306fdde031461134557508063095ea7b31461131f57806318160ddd1461130157806323b872dd14611243578063313ce5671461122757806339509351146111d457806370a082311461119a57806395d89b411461116f578063a457c2d7146110b0578063a69df4b51461104c578063a9059cbb1461101b578063c06d4e0414610a8d578063c87b56dd146101105763dd62ed3e146100b957600080fd5b3461010b57604036600319011261010b576100d26113b6565b6100da6113cc565b906001600160a01b038091166000526001602052604060002091166000526020526020604060002054604051908152f35b600080fd5b3461010b57602036600319011261010b576040516020810181811067ffffffffffffffff821117610a775760405260008152600090600954915b8281106108d557506006546040519060808201604052600f6f3031323334353637383961626364656681527fffffffffffffffffffffffffffffffffffffffff0000000000000000000000006002840192602884526000604a86015260601b1690600160005b8080018601602285831a85811651602384015360041c5191015301601481146101db576001906101b0565b50505061307860028251019152815260405190610202826101fb8161144a565b0383611575565b6040519261021a84610213816114ee565b0385611575565b670de0b6b3a76400006002540490604051608081019260a0820160405260008452925b6000190192600a90603082820601855304928361023d5790819003608001601f1982015260405160085490935083600061027683611410565b80835292600181169081156108b65750600114610855575b61029a92500384611575565b82516060938082610760575b5050506040519586947f7b0000000000000000000000000000000000000000000000000000000000000060208701527f2263726561746f72223a22000000000000000000000000000000000000000000602187015261030f815180926020602c8a019101611367565b8501957f222c0000000000000000000000000000000000000000000000000000000000009687602c8201527f226e616d65223a22000000000000000000000000000000000000000000000000602e820152610374825180936020603685019101611367565b018660368201527f2273796d626f6c223a220000000000000000000000000000000000000000000060388201526103b5825180936020604285019101611367565b018560428201527f22737570706c79223a220000000000000000000000000000000000000000000060448201526103f7601f198301518093604e840190611367565b0184604e8201527f226465736372697074696f6e223a22000000000000000000000000000000000060508201526000906007549061043482611410565b916001811690811561073b57506001146106df575b50509184918361057a9660509695527f226c696e6b73223a5b00000000000000000000000000000000000000000000006002820152610492825180936020600b85019101611367565b017f5d2c000000000000000000000000000000000000000000000000000000000000600b8201527f22696d6167655f64617461223a22646174613a696d6167652f776562703b6261600d8201527f736536342c000000000000000000000000000000000000000000000000000000602d820152610519825180936020603285019101611367565b019060328201527f226261636b67726f756e645f636f6c6f72223a2230303030303022000000000060348201527f7d00000000000000000000000000000000000000000000000000000000000000604f820152036030810184520182611575565b806060908251806105ed575b6105e9836105dd603d60405180937f646174613a6170706c69636174696f6e2f6a736f6e3b6261736536342c00000060208301526105cd8151809260208686019101611367565b810103601d810184520182611575565b6040519182918261138a565b0390f35b9091506003600282010460021b90604051927f4142434445464748494a4b4c4d4e4f505152535455565758595a616263646566601f526106707f6768696a6b6c6d6e6f707172737475767778797a303132333435363738392d5f18603f526020840160046003602085890101519360006020878b0101525b0191603f8351818160121c16516000538181600c1c1651600153818160061c16516002531651600353600051815201906020858701018210156106ad57600490600390610665565b5090600383613d3d60f01b9260206000966105e99a010152604086880101604052066002048203525281528280610586565b90915060076000527fa66cc928b5edb82af9bd49922954155ab7b0942694bea4ce44661d9a8736c6886000905b83821061072157505001605f01816050610449565b8054605f838501015288965060209091019060010161070c565b60ff1916605f8381019190915283151590930290910190910191508290506050610449565b90919450604051947f4142434445464748494a4b4c4d4e4f505152535455565758595a616263646566601f526106707f6768696a6b6c6d6e6f707172737475767778797a303132333435363738392d5f18603f5260208601602082816003600283010460021b8a01019501019060046003835195600085525b0191603f8351818160121c16516000538181600c1c1651600153818160061c16516002531651600353600051815201908582101561081c576004906003906107d9565b50600394506002936000925260408585850104851b890101604052613d3d60f01b858406850482035252010460021b83528680806102a6565b509060086000527ff3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee3906000915b81831061089a57505090602061029a9282010161028e565b6020919350806001915483858a01015201910190918592610882565b6020925061029a94915060ff191682840152151560051b82010161028e565b6009600052604051826108f2829451809260208086019101611367565b601160f91b60208284010152600090837f6e1540171b6c0c960b71a7020d9f60077f6af931a8bbf590da0223dacf75c7af019081549161093183611410565b9260018116908115610a4c5750600114610a03575b5050506109669181601160f91b6001935203601e19810185520183611575565b818360001981011161099057600019840182106109a6575b5060001981146109905760010161014a565b634e487b7160e01b600052601160045260246000fd5b9091506109fc6021604051836109c6829551809260208086019101611367565b81017f2c000000000000000000000000000000000000000000000000000000000000006020820152036001810184520182611575565b908361097e565b919350915060005260206000206000905b838210610a335750508360216109669360019383010101918193610946565b8054602184880184010152602090910190600101610a14565b9291505060019350610966949260219260ff1916838386010152801515029083010101918193610946565b634e487b7160e01b600052604160045260246000fd5b3461010b57606036600319011261010b5767ffffffffffffffff60043581811161010b57610abf9036906004016113e2565b91602490813581811161010b57610ada9036906004016113e2565b909360449182359584871161010b573660238801121561010b5786600401359785891161010b5760059336888b871b8b01011161010b576001600160a01b0360065416330361010b5781610ec9575b505081610d83575b5050600954808703610b3f57005b680100000000000000008711610d6e5786600955808710610cdd575b5083859695019360096000527f6e1540171b6c0c960b71a7020d9f60077f6af931a8bbf590da0223dacf75c7af9460009760421981360301915b888a10610b9e57005b80358381121561010b578201848101359088821161010b578136038882011361010b578790610bcd8b54611410565b90601f91828111610ca3575b506000918411600114610c27579280602093600196938796600093610c1a575b505050600019600383901b1c191690841b178b555b01980199019896610b95565b01013590508a3880610bf9565b60008c8152602080822094929391601f198716915b828210610c84575050928592602095926001989597899810610c68575b50505050831b83018b55610c0e565b60001960f88660031b161c19920101351690558e808b81610c59565b600184978294959683949789010135815501960193018c939291610c3c565b610cce908d6000526020908482600020918d828a01901c8301938910610cd4575b018c1c019061191f565b8e610bd9565b92508192610cc4565b6009600052867f6e1540171b6c0c960b71a7020d9f60077f6af931a8bbf590da0223dacf75c7af91820191015b818110610d175750610b5b565b80610d2460019254611410565b80610d31575b5001610d0a565b601f908181118414610d4a575050600081555b89610d2a565b610d65600092848452602084209201871c820185830161191f565b81835555610d44565b84634e487b7160e01b60005260416004526000fd5b848211610eb4578190610d97600854611410565b601f8111610e61575b50600090601f8311600114610ddf57600092610dd4575b50508160011b916000199060031b1c1916176008555b8680610b31565b013590508880610db7565b7ff3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee3925090601f19841660005b818110610e495750908460019594939210610e2f575b505050811b01600855610dcd565b0135600019600384901b60f8161c19169055888080610e21565b91936020600181928787013581550195019201610e0b565b610ea4907ff3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee3601f8501871c81019160208610610eaa575b601f01871c019061191f565b89610da0565b9091508190610e98565b85634e487b7160e01b60005260416004526000fd5b868211611006578190610edd600754611410565b601f8111610fae575b50600090601f8311600114610f2557600092610f1a575b50508160011b916000199060031b1c1916176007555b8880610b29565b013590508a80610efd565b909150601f1983169160076000527fa66cc928b5edb82af9bd49922954155ab7b0942694bea4ce44661d9a8736c6889260005b818110610f965750908460019594939210610f7c575b505050811b01600755610f13565b0135600019600384901b60f8161c191690558a8080610f6e565b91936020600181928787013581550195019201610f58565b610ff69060076000527fa66cc928b5edb82af9bd49922954155ab7b0942694bea4ce44661d9a8736c688601f8501891c81019160208610610ffc575b601f01891c019061191f565b8b610ee6565b9091508190610fea565b87634e487b7160e01b60005260416004526000fd5b3461010b57604036600319011261010b576110416110376113b6565b6024359033611597565b602060405160018152f35b3461010b57600036600319011261010b576005546001600160a01b03811633148061109e575b1561010b577fffffffffffffffffffffff00ffffffffffffffffffffffffffffffffffffffff16600555005b50600160ff8260a01c16151514611072565b3461010b57604036600319011261010b576110c96113b6565b6024359033600052600160205260406000206001600160a01b0382166000526020526040600020549180831061110557611041920390336117eb565b608460405162461bcd60e51b815260206004820152602560248201527f45524332303a2064656372656173656420616c6c6f77616e63652062656c6f7760448201527f207a65726f0000000000000000000000000000000000000000000000000000006064820152fd5b3461010b57600036600319011261010b576105e96040516105dd81611193816114ee565b0382611575565b3461010b57602036600319011261010b576001600160a01b036111bb6113b6565b1660005260006020526020604060002054604051908152f35b3461010b57604036600319011261010b576111ed6113b6565b33600052600160205260406000206001600160a01b03821660005260205260406000205460243581018091116109905761104191336117eb565b3461010b57600036600319011261010b57602060405160128152f35b3461010b57606036600319011261010b5761125c6113b6565b6112646113cc565b604435906001600160a01b03831660005260016020526040600020336000526020526040600020549260001984036112a1575b6110419350611597565b8284106112bd576112b883611041950333836117eb565b611297565b606460405162461bcd60e51b815260206004820152601d60248201527f45524332303a20696e73756666696369656e7420616c6c6f77616e63650000006044820152fd5b3461010b57600036600319011261010b576020600254604051908152f35b3461010b57604036600319011261010b5761104161133b6113b6565b60243590336117eb565b3461010b57600036600319011261010b57806105dd816111936105e99461144a565b60005b83811061137a5750506000910152565b818101518382015260200161136a565b604091602082526113aa8151809281602086015260208686019101611367565b601f01601f1916010190565b600435906001600160a01b038216820361010b57565b602435906001600160a01b038216820361010b57565b9181601f8401121561010b5782359167ffffffffffffffff831161010b576020838186019501011161010b57565b90600182811c92168015611440575b602083101461142a57565b634e487b7160e01b600052602260045260246000fd5b91607f169161141f565b6003546000929161145a82611410565b808252916001908181169081156114d1575060011461147857505050565b9192935060036000527fc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b916000925b8484106114b957505060209250010190565b805460208585018101919091529093019281016114a7565b915050602093945060ff929192191683830152151560051b010190565b600454600092916114fe82611410565b808252916001908181169081156114d1575060011461151c57505050565b9192935060046000527f8a35acfbc15ff81a39ae7d344fd709f28e8600b4aa8c65c6b64bfe7fe36bd19b916000925b84841061155d57505060209250010190565b8054602085850181019190915290930192810161154b565b90601f8019910116810190811067ffffffffffffffff821117610a7757604052565b916001600160a01b038093169182156117815783169283156117175760055460ff8160a01c1661168a575b505060008281528060205260408120549180831061162057604082827fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef958760209652828652038282205586815220818154019055604051908152a3565b608460405162461bcd60e51b815260206004820152602660248201527f45524332303a207472616e7366657220616d6f756e742065786365656473206260448201527f616c616e636500000000000000000000000000000000000000000000000000006064820152fd5b1680831490811561170d575b50156116a35738806115c2565b608460405162461bcd60e51b815260206004820152602260248201527f7472616e73666572206e6f7420616c6c6f776564206265666f7265206c61756e60448201527f63680000000000000000000000000000000000000000000000000000000000006064820152fd5b9050831438611696565b608460405162461bcd60e51b815260206004820152602360248201527f45524332303a207472616e7366657220746f20746865207a65726f206164647260448201527f65737300000000000000000000000000000000000000000000000000000000006064820152fd5b608460405162461bcd60e51b815260206004820152602560248201527f45524332303a207472616e736665722066726f6d20746865207a65726f20616460448201527f64726573730000000000000000000000000000000000000000000000000000006064820152fd5b6001600160a01b038091169182156118b6571691821561184c5760207f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b925918360005260018252604060002085600052825280604060002055604051908152a3565b608460405162461bcd60e51b815260206004820152602260248201527f45524332303a20617070726f766520746f20746865207a65726f20616464726560448201527f73730000000000000000000000000000000000000000000000000000000000006064820152fd5b608460405162461bcd60e51b8152602060048201526024808201527f45524332303a20617070726f76652066726f6d20746865207a65726f2061646460448201527f72657373000000000000000000000000000000000000000000000000000000006064820152fd5b81811061192a575050565b6000815560010161191f56fea26469706673582212206f0073e94722fc733abd696a5db6d427088ca4421acd011a0c52951face0c40064736f6c63430008120033a66cc928b5edb82af9bd49922954155ab7b0942694bea4ce44661d9a8736c688f3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee36e1540171b6c0c960b71a7020d9f60077f6af931a8bbf590da0223dacf75c7afa2646970667358221220bd5ff7a481c323ccdb753746c981005cd2ba8bbca792558150e591f71d633da664736f6c63430008120033
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 30 Chains
Chain | Token | Portfolio % | Price | Amount | Value |
---|
A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.