Overview
ETH Balance
0 ETH
ETH Value
$0.00More Info
Private Name Tags
ContractCreator
Transaction Hash |
Method
|
Block
|
From
|
To
|
|||||
---|---|---|---|---|---|---|---|---|---|
Latest 1 internal transaction
Parent Transaction Hash | Block | From | To | |||
---|---|---|---|---|---|---|
415609 | 30 days ago | Contract Creation | 0 ETH |
Loading...
Loading
This contract may be a proxy contract. Click on More Options and select Is this a proxy? to confirm and enable the "Read as Proxy" & "Write as Proxy" tabs.
Contract Name:
ERC721GeneralSequence
Compiler Version
v0.8.10+commit.fc410830
ZkSolc Version
v1.5.6
Optimization Enabled:
Yes with Mode z
Other Settings:
default evmVersion
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; import "./ERC721Base.sol"; import "../metadata/MetadataEncryption.sol"; import "../tokenManager/interfaces/IPostTransfer.sol"; import "../tokenManager/interfaces/IPostBurn.sol"; import "./interfaces/IERC721GeneralMint.sol"; import "./ERC721GeneralSequenceBase.sol"; /** * @title Generalized ERC721 that expects tokenIds to increment in a monotonically increasing sequence * @author highlight.xyz * @notice Generalized NFT smart contract */ contract ERC721GeneralSequence is ERC721GeneralSequenceBase { using EnumerableSet for EnumerableSet.AddressSet; /** * @notice Initialize the contract * @param creator Creator/owner of contract * @param _contractURI Contract metadata * @param defaultRoyalty Default royalty object for contract (optional) * @param _defaultTokenManager Default token manager for contract (optional) * @param _name Name of token edition * @param _symbol Symbol of the token edition * @param trustedForwarder Trusted minimal forwarder * @param initialMinter Initial minter to register * @param newBaseURI Base URI for contract * @param _limitSupply Initial limit supply * @param useMarketplaceFiltererRegistry Denotes whether to use marketplace filterer registry * @param _observability Observability contract address */ function initialize( address creator, string memory _contractURI, IRoyaltyManager.Royalty memory defaultRoyalty, address _defaultTokenManager, string memory _name, string memory _symbol, address trustedForwarder, address initialMinter, string memory newBaseURI, uint256 _limitSupply, bool useMarketplaceFiltererRegistry, address _observability ) external initializer { _initialize( creator, _contractURI, defaultRoyalty, _defaultTokenManager, _name, _symbol, trustedForwarder, initialMinter, newBaseURI, _limitSupply, useMarketplaceFiltererRegistry, _observability ); } /** * @notice Initialize the contract * @param creator Creator/owner of contract * @param data Data to initialize the contract * @ param _contractURI Contract metadata * @ param defaultRoyalty Default royalty object for contract (optional) * @ param _defaultTokenManager Default token manager for contract (optional) * @ param _name Name of token edition * @ param _symbol Symbol of the token edition * @ param trustedForwarder Trusted minimal forwarder * @ param initialMinter Initial minter to register * @ param newBaseURI Base URI for contract * @ param _limitSupply Initial limit supply * @ param useMarketplaceFiltererRegistry Denotes whether to use marketplace filterer registry * @ param _observability Observability contract address */ function initialize(address creator, bytes memory data) external initializer { ( string memory _contractURI, IRoyaltyManager.Royalty memory defaultRoyalty, address _defaultTokenManager, string memory _name, string memory _symbol, address trustedForwarder, address initialMinter, string memory newBaseURI, uint256 _limitSupply, bool useMarketplaceFiltererRegistry, address _observability ) = abi.decode( data, ( string, IRoyaltyManager.Royalty, address, string, string, address, address, string, uint256, bool, address ) ); _initialize( creator, _contractURI, defaultRoyalty, _defaultTokenManager, _name, _symbol, trustedForwarder, initialMinter, newBaseURI, _limitSupply, useMarketplaceFiltererRegistry, _observability ); } /** * @notice Used for meta-transactions */ function _msgSender() internal view virtual override(ERC721GeneralSequenceBase) returns (address sender) { return ERC721GeneralSequenceBase._msgSender(); } /** * @notice Used for meta-transactions */ function _msgData() internal view virtual override(ERC721GeneralSequenceBase) returns (bytes calldata) { return ERC721GeneralSequenceBase._msgData(); } /** * @dev For more efficient reverts. */ function _revert(bytes4 errorSelector) internal pure override(ERC721GeneralSequenceBase) { assembly { mstore(0x00, errorSelector) revert(0x00, 0x04) } } /** * @notice Initialize the contract * @param creator Creator/owner of contract * @param _contractURI Contract metadata * @param defaultRoyalty Default royalty object for contract (optional) * @param _defaultTokenManager Default token manager for contract (optional) * @param _name Name of token edition * @param _symbol Symbol of the token edition * @param trustedForwarder Trusted minimal forwarder * @param initialMinter Initial minter to register * @param newBaseURI Base URI for contract * @param _limitSupply Initial limit supply * @param useMarketplaceFiltererRegistry Denotes whether to use marketplace filterer registry * @param _observability Observability contract address */ function _initialize( address creator, string memory _contractURI, IRoyaltyManager.Royalty memory defaultRoyalty, address _defaultTokenManager, string memory _name, string memory _symbol, address trustedForwarder, address initialMinter, string memory newBaseURI, uint256 _limitSupply, bool useMarketplaceFiltererRegistry, address _observability ) private { __ERC721URIStorage_init(); __ERC721Base_initialize(creator, defaultRoyalty, _defaultTokenManager); __ERC2771ContextUpgradeable__init__(trustedForwarder); __ERC721A_init(_name, _symbol); // deprecate but keep input for backwards-compatibility: // __MarketplaceFilterer__init__(useMarketplaceFiltererRegistry); _minters.add(initialMinter); contractURI = _contractURI; IObservability(_observability).emitSeriesDeployed(address(this)); observability = IObservability(_observability); if (bytes(newBaseURI).length > 0) { _setBaseURI(newBaseURI); // don't emit on observability contract here } if (_limitSupply > 0) { limitSupply = _limitSupply; // don't emit on observability contract here } } }
// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; import "../royaltyManager/interfaces/IRoyaltyManager.sol"; import "../tokenManager/interfaces/ITokenManager.sol"; import "../utils/Ownable.sol"; import "../utils/ERC2981/IERC2981Upgradeable.sol"; import "../utils/ERC165/ERC165CheckerUpgradeable.sol"; import "../metatx/ERC2771ContextUpgradeable.sol"; import "../observability/IObservability.sol"; import "@openzeppelin/contracts/utils/structs/EnumerableSet.sol"; import "@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol"; import "@openzeppelin/contracts-upgradeable/security/ReentrancyGuardUpgradeable.sol"; import { UUPSUpgradeable } from "@openzeppelin/contracts/proxy/utils/UUPSUpgradeable.sol"; /** * @title Base ERC721 * @author highlight.xyz * @notice Core piece of Highlight NFT contracts (v2) */ abstract contract ERC721Base is OwnableUpgradeable, IERC2981Upgradeable, ERC2771ContextUpgradeable, ReentrancyGuardUpgradeable, UUPSUpgradeable { using EnumerableSet for EnumerableSet.AddressSet; using ERC165CheckerUpgradeable for address; /** * @notice Throw when token or royalty manager is invalid */ error InvalidManager(); /** * @notice Throw when token or royalty manager does not exist */ error ManagerDoesNotExist(); /** * @notice Throw when sender is unauthorized to perform action */ error Unauthorized(); /** * @notice Throw when sender is not a minter */ error NotMinter(); /** * @notice Throw when token manager or royalty manager swap is blocked */ error ManagerSwapBlocked(); /** * @notice Throw when token manager or royalty manager remove is blocked */ error ManagerRemoveBlocked(); /** * @notice Throw when setting default or granular royalty is blocked */ error RoyaltySetBlocked(); /** * @notice Throw when royalty BPS is invalid */ error RoyaltyBPSInvalid(); /** * @notice Throw when minter registration is invalid */ error MinterRegistrationInvalid(); /** * @notice Set of minters allowed to mint on contract */ EnumerableSet.AddressSet internal _minters; /** * @notice Global token/edition manager default */ address public defaultManager; /** * @notice Token/edition managers per token grouping. * Edition ID if implemented by Editions contract, and token ID if implemented by General contract. */ mapping(uint256 => address) internal _managers; /** * @notice Default royalty for entire contract */ IRoyaltyManager.Royalty internal _defaultRoyalty; /** * @notice Royalty per token grouping. * Edition ID if implemented by Editions contract, and token ID if implemented by General contract. */ mapping(uint256 => IRoyaltyManager.Royalty) internal _royalties; /** * @notice Royalty manager - optional contract that defines the conditions around setting royalties */ address public royaltyManager; /** * @notice Freezes minting on smart contract forever */ uint8 internal _mintFrozen; /** * @notice Observability contract */ IObservability public observability; /** * @notice Emitted when minter is registered or unregistered * @param minter Minter that was changed * @param registered True if the minter was registered, false if unregistered */ event MinterRegistrationChanged(address indexed minter, bool indexed registered); /** * @notice Emitted when token managers are set for token/edition ids * @param _ids Edition / token ids * @param _tokenManagers Token managers to set for tokens / editions */ event GranularTokenManagersSet(uint256[] _ids, address[] _tokenManagers); /** * @notice Emitted when token managers are removed for token/edition ids * @param _ids Edition / token ids to remove token managers for */ event GranularTokenManagersRemoved(uint256[] _ids); /** * @notice Emitted when default token manager changed * @param newDefaultTokenManager New default token manager. Zero address if old one was removed */ event DefaultTokenManagerChanged(address indexed newDefaultTokenManager); /** * @notice Emitted when default royalty is set * @param recipientAddress Royalty recipient * @param royaltyPercentageBPS Percentage of sale (in basis points) owed to royalty recipient */ event DefaultRoyaltySet(address indexed recipientAddress, uint16 indexed royaltyPercentageBPS); /** * @notice Emitted when royalties are set for edition / token ids * @param ids Token / edition ids * @param _newRoyalties New royalties for each token / edition */ event GranularRoyaltiesSet(uint256[] ids, IRoyaltyManager.Royalty[] _newRoyalties); /** * @notice Emitted when royalty manager is updated * @param newRoyaltyManager New royalty manager. Zero address if old one was removed */ event RoyaltyManagerChanged(address indexed newRoyaltyManager); /** * @notice Emitted when mints are frozen permanently */ event MintsFrozen(); /** * @notice Restricts calls to minters */ modifier onlyMinter() { if (!_minters.contains(_msgSender())) { _revert(NotMinter.selector); } _; } /** * @notice Restricts calls if input royalty bps is over 10000 */ modifier royaltyValid(uint16 _royaltyBPS) { if (!_royaltyBPSValid(_royaltyBPS)) { _revert(RoyaltyBPSInvalid.selector); } _; } /** * @notice Registers a minter * @param minter New minter */ function registerMinter(address minter) external onlyOwner { if (!_minters.add(minter)) { _revert(MinterRegistrationInvalid.selector); } emit MinterRegistrationChanged(minter, true); observability.emitMinterRegistrationChanged(minter, true); } /** * @notice Unregisters a minter * @param minter Minter to unregister */ function unregisterMinter(address minter) external onlyOwner { if (!_minters.remove(minter)) { _revert(MinterRegistrationInvalid.selector); } emit MinterRegistrationChanged(minter, false); observability.emitMinterRegistrationChanged(minter, false); } /** * @notice Sets granular token managers if current token manager(s) allow it * @param _ids Edition / token ids * @param _tokenManagers Token managers to set for tokens / editions */ function setGranularTokenManagers( uint256[] calldata _ids, address[] calldata _tokenManagers ) external nonReentrant { address msgSender = _msgSender(); address tempOwner = owner(); uint256 idsLength = _ids.length; for (uint256 i = 0; i < idsLength; i++) { if (!_isValidTokenManager(_tokenManagers[i])) { _revert(InvalidManager.selector); } address currentTokenManager = tokenManager(_ids[i]); if (currentTokenManager == address(0)) { if (msgSender != tempOwner) { _revert(Unauthorized.selector); } } else { if (!ITokenManager(currentTokenManager).canSwap(msgSender, _ids[i], _managers[i])) { _revert(ManagerSwapBlocked.selector); } } _managers[_ids[i]] = _tokenManagers[i]; } emit GranularTokenManagersSet(_ids, _tokenManagers); observability.emitGranularTokenManagersSet(_ids, _tokenManagers); } /** * @notice Remove granular token managers * @param _ids Edition / token ids to remove token managers for */ function removeGranularTokenManagers(uint256[] calldata _ids) external nonReentrant { address msgSender = _msgSender(); uint256 idsLength = _ids.length; for (uint256 i = 0; i < idsLength; i++) { address currentTokenManager = _managers[_ids[i]]; if (currentTokenManager == address(0)) { _revert(ManagerDoesNotExist.selector); } if (!ITokenManager(currentTokenManager).canRemoveItself(msgSender, _ids[i])) { _revert(ManagerRemoveBlocked.selector); } _managers[_ids[i]] = address(0); } emit GranularTokenManagersRemoved(_ids); observability.emitGranularTokenManagersRemoved(_ids); } /** * @notice Set default token manager if current token manager allows it * @param _defaultTokenManager New default token manager */ function setDefaultTokenManager(address _defaultTokenManager) external nonReentrant { if (!_isValidTokenManager(_defaultTokenManager)) { _revert(InvalidManager.selector); } address msgSender = _msgSender(); address currentTokenManager = defaultManager; if (currentTokenManager == address(0)) { if (msgSender != owner()) { _revert(Unauthorized.selector); } } else { if (!ITokenManager(currentTokenManager).canSwap(msgSender, 0, _defaultTokenManager)) { _revert(ManagerSwapBlocked.selector); } } defaultManager = _defaultTokenManager; emit DefaultTokenManagerChanged(_defaultTokenManager); observability.emitDefaultTokenManagerChanged(_defaultTokenManager); } /** * @notice Removes default token manager if current token manager allows it */ function removeDefaultTokenManager() external nonReentrant { address msgSender = _msgSender(); address currentTokenManager = defaultManager; if (currentTokenManager == address(0)) { _revert(ManagerDoesNotExist.selector); } if (!ITokenManager(currentTokenManager).canRemoveItself(msgSender, 0)) { _revert(ManagerRemoveBlocked.selector); } defaultManager = address(0); emit DefaultTokenManagerChanged(address(0)); observability.emitDefaultTokenManagerChanged(address(0)); } /** * @notice Sets default royalty if royalty manager allows it * @param _royalty New default royalty */ function setDefaultRoyalty( IRoyaltyManager.Royalty calldata _royalty ) external nonReentrant royaltyValid(_royalty.royaltyPercentageBPS) { address msgSender = _msgSender(); address _royaltyManager = royaltyManager; if (_royaltyManager == address(0)) { if (msgSender != owner()) { _revert(Unauthorized.selector); } } else { if (!IRoyaltyManager(_royaltyManager).canSetDefaultRoyalty(_royalty, msgSender)) { _revert(RoyaltySetBlocked.selector); } } _defaultRoyalty = _royalty; emit DefaultRoyaltySet(_royalty.recipientAddress, _royalty.royaltyPercentageBPS); observability.emitDefaultRoyaltySet(_royalty.recipientAddress, _royalty.royaltyPercentageBPS); } /** * @notice Sets granular royalties (per token-grouping) if royalty manager allows it * @param ids Token / edition ids * @param _newRoyalties New royalties for each token / edition */ function setGranularRoyalties( uint256[] calldata ids, IRoyaltyManager.Royalty[] calldata _newRoyalties ) external nonReentrant { address msgSender = _msgSender(); address tempOwner = owner(); address _royaltyManager = royaltyManager; uint256 idsLength = ids.length; if (_royaltyManager == address(0)) { if (msgSender != tempOwner) { _revert(Unauthorized.selector); } for (uint256 i = 0; i < idsLength; i++) { if (!_royaltyBPSValid(_newRoyalties[i].royaltyPercentageBPS)) { _revert(RoyaltyBPSInvalid.selector); } _royalties[ids[i]] = _newRoyalties[i]; } } else { for (uint256 i = 0; i < idsLength; i++) { if (!_royaltyBPSValid(_newRoyalties[i].royaltyPercentageBPS)) { _revert(RoyaltyBPSInvalid.selector); } if (!IRoyaltyManager(_royaltyManager).canSetGranularRoyalty(ids[i], _newRoyalties[i], msgSender)) { _revert(RoyaltySetBlocked.selector); } _royalties[ids[i]] = _newRoyalties[i]; } } emit GranularRoyaltiesSet(ids, _newRoyalties); observability.emitGranularRoyaltiesSet(ids, _newRoyalties); } /** * @notice Sets royalty manager if current one allows it * @param _royaltyManager New royalty manager */ function setRoyaltyManager(address _royaltyManager) external nonReentrant { if (!_isValidRoyaltyManager(_royaltyManager)) { _revert(InvalidManager.selector); } address msgSender = _msgSender(); address currentRoyaltyManager = royaltyManager; if (currentRoyaltyManager == address(0)) { if (msgSender != owner()) { _revert(Unauthorized.selector); } } else { if (!IRoyaltyManager(currentRoyaltyManager).canSwap(_royaltyManager, msgSender)) { _revert(ManagerSwapBlocked.selector); } } royaltyManager = _royaltyManager; emit RoyaltyManagerChanged(_royaltyManager); observability.emitRoyaltyManagerChanged(_royaltyManager); } /** * @notice Removes royalty manager if current one allows it */ function removeRoyaltyManager() external nonReentrant { address msgSender = _msgSender(); address currentRoyaltyManager = royaltyManager; if (currentRoyaltyManager == address(0)) { _revert(ManagerDoesNotExist.selector); } if (!IRoyaltyManager(currentRoyaltyManager).canRemoveItself(msgSender)) { _revert(ManagerRemoveBlocked.selector); } royaltyManager = address(0); emit RoyaltyManagerChanged(address(0)); observability.emitRoyaltyManagerChanged(address(0)); } /** * @notice Freeze mints on contract forever */ function freezeMints() external onlyOwner nonReentrant { _mintFrozen = 1; emit MintsFrozen(); observability.emitMintsFrozen(); } /** * @notice Return allowed minters on contract */ function minters() external view returns (address[] memory) { return _minters.values(); } /** * @notice Conforms to ERC-2981. Editions should overwrite to return royalty for entire edition * @param _tokenGroupingId Token id if on general, and edition id if on editions * @param _salePrice Sale price of token */ function royaltyInfo( uint256 _tokenGroupingId, uint256 _salePrice ) public view virtual override returns (address receiver, uint256 royaltyAmount) { IRoyaltyManager.Royalty memory royalty = _royalties[_tokenGroupingId]; if (royalty.recipientAddress == address(0)) { royalty = _defaultRoyalty; } receiver = royalty.recipientAddress; royaltyAmount = (_salePrice * uint256(royalty.royaltyPercentageBPS)) / 10000; } /** * @notice Returns the token manager for the id passed in. * @param id Token ID or Edition ID for Editions implementing contracts */ function tokenManager(uint256 id) public view returns (address manager) { manager = defaultManager; address granularManager = _managers[id]; if (granularManager != address(0)) { manager = granularManager; } } /** * @notice Initializes the contract, setting the creator as the initial owner. * @param _creator Contract creator * @param defaultRoyalty Default royalty for the contract * @param _defaultTokenManager Default token manager for the contract */ function __ERC721Base_initialize( address _creator, IRoyaltyManager.Royalty memory defaultRoyalty, address _defaultTokenManager ) internal onlyInitializing royaltyValid(defaultRoyalty.royaltyPercentageBPS) { __Ownable_init(); __ReentrancyGuard_init(); _transferOwnership(_creator); if (defaultRoyalty.recipientAddress != address(0)) { _defaultRoyalty = defaultRoyalty; } if (_defaultTokenManager != address(0)) { defaultManager = _defaultTokenManager; } } /** * @notice See {UUPSUpgradeable-_authorizeUpgrade} * @param // New implementation to upgrade to */ function _authorizeUpgrade(address) internal override { revert("Not upgradeable"); } /** * @notice Returns true if address is a valid tokenManager * @param _tokenManager Token manager being checked */ function _isValidTokenManager(address _tokenManager) internal view returns (bool) { return _tokenManager.supportsInterface(type(ITokenManager).interfaceId); } /** * @notice Returns true if address is a valid royaltyManager * @param _royaltyManager Royalty manager being checked */ function _isValidRoyaltyManager(address _royaltyManager) internal view returns (bool) { return _royaltyManager.supportsInterface(type(IRoyaltyManager).interfaceId); } /** * @notice Used for meta-transactions */ function _msgSender() internal view virtual override(ContextUpgradeable, ERC2771ContextUpgradeable) returns (address sender) { return ERC2771ContextUpgradeable._msgSender(); } /** * @dev For more efficient reverts. */ function _revert(bytes4 errorSelector) internal pure virtual { assembly { mstore(0x00, errorSelector) revert(0x00, 0x04) } } /** * @notice Used for meta-transactions */ function _msgData() internal view virtual override(ContextUpgradeable, ERC2771ContextUpgradeable) returns (bytes calldata) { return ERC2771ContextUpgradeable._msgData(); } /** * @notice Returns true if royalty bps passed in is valid (<= 10000) * @param _royaltyBPS Royalty basis points */ function _royaltyBPSValid(uint16 _royaltyBPS) private pure returns (bool) { return _royaltyBPS <= 10000; } }
// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; /** * @dev Utilities for metadata encryption and decryption * @author highlight.xyz */ abstract contract MetadataEncryption { /// @dev See: https://ethereum.stackexchange.com/questions/69825/decrypt-message-on-chain function encryptDecrypt(bytes memory data, bytes calldata key) public pure returns (bytes memory result) { // Store data length on stack for later use uint256 length = data.length; // solhint-disable-next-line no-inline-assembly assembly { // Set result to free memory pointer result := mload(0x40) // Increase free memory pointer by lenght + 32 mstore(0x40, add(add(result, length), 32)) // Set result length mstore(result, length) } // Iterate over the data stepping by 32 bytes for (uint256 i = 0; i < length; i += 32) { // Generate hash of the key and offset bytes32 hash = keccak256(abi.encodePacked(key, i)); bytes32 chunk; // solhint-disable-next-line no-inline-assembly assembly { // Read 32-bytes data chunk chunk := mload(add(data, add(i, 32))) } // XOR the chunk with hash chunk ^= hash; // solhint-disable-next-line no-inline-assembly assembly { // Write 32-byte encrypted chunk mstore(add(result, add(i, 32)), chunk) } } } function _sliceUint(bytes memory bs, uint256 start) internal pure returns (uint256) { require(bs.length >= start + 32, "slicing out of range"); uint256 x; assembly { x := mload(add(bs, add(0x20, start))) } return x; } }
// SPDX-License-Identifier: MIT pragma solidity 0.8.10; /** * @author highlight.xyz * @notice If token managers implement this, transfer actions will call * postBurn on the token manager. */ interface IPostBurn { /** * @notice Hook called by contract after burn, if token manager of burned token implements this * interface. * @param operator Operator burning tokens * @param sender Msg sender * @param id Burned token's id or id of edition of token that is burned */ function postBurn(address operator, address sender, uint256 id) external; }
// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; import "./ERC721Base.sol"; import "../metadata/MetadataEncryption.sol"; import "../tokenManager/interfaces/IPostTransfer.sol"; import "../tokenManager/interfaces/IPostBurn.sol"; import "./interfaces/IERC721GeneralSequenceMint.sol"; import "./erc721a/ERC721AURIStorageUpgradeable.sol"; import "./inchain-rendering/interfaces/IHLRenderer.sol"; /** * @title Generalized Base ERC721 * @author highlight.xyz * @notice Generalized Base NFT smart contract */ abstract contract ERC721GeneralSequenceBase is ERC721Base, ERC721AURIStorageUpgradeable, IERC721GeneralSequenceMint { using EnumerableSet for EnumerableSet.AddressSet; /** * @notice Throw when attempting to mint, while mint is frozen */ error MintFrozen(); /** * @notice Throw when requested token is not in range within bounds of limit supply */ error TokenNotInRange(); /** * @notice Throw when new supply is over limit supply */ error OverLimitSupply(); /** * @notice Throw when array lengths are mismatched */ error MismatchedArrayLengths(); /** * @notice Throw when string is empty */ error EmptyString(); /** * @notice Custom renderer config, used for collections where metadata is rendered "in-chain" * @param renderer Renderer address * @param processMintDataOnRenderer If true, process mint data on renderer */ struct CustomRendererConfig { address renderer; bool processMintDataOnRenderer; } /** * @notice Contract metadata */ string public contractURI; /** * @notice Limit the supply to take advantage of over-promising in summation with multiple mint vectors */ uint256 public limitSupply; /** * @notice Custom renderer config */ CustomRendererConfig public customRendererConfig; /** * @notice Emitted when uris are set for tokens * @param ids IDs of tokens to set uris for * @param uris Uris to set on tokens */ event TokenURIsSet(uint256[] ids, string[] uris); /** * @notice Emitted when limit supply is set * @param newLimitSupply Limit supply to set */ event LimitSupplySet(uint256 indexed newLimitSupply); /** * @notice See {IERC721GeneralMint-mintOneToOneRecipient} */ function mintOneToOneRecipient(address recipient) external virtual onlyMinter nonReentrant returns (uint256) { if (_mintFrozen == 1) { _revert(MintFrozen.selector); } uint256 tempSupply = _nextTokenId(); _requireLimitSupply(tempSupply); _mint(recipient, 1); // process mint on custom renderer if present CustomRendererConfig memory _customRendererConfig = customRendererConfig; if (_customRendererConfig.processMintDataOnRenderer) { IHLRenderer(_customRendererConfig.renderer).processOneRecipientMint(tempSupply, 1, recipient); } return tempSupply; } /** * @notice See {IERC721GeneralMint-mintAmountToOneRecipient} */ function mintAmountToOneRecipient(address recipient, uint256 amount) external virtual onlyMinter nonReentrant { if (_mintFrozen == 1) { _revert(MintFrozen.selector); } uint256 tempSupply = _nextTokenId() - 1; // cache _mint(recipient, amount); _requireLimitSupply(tempSupply + amount); // process mint on custom renderer if present CustomRendererConfig memory _customRendererConfig = customRendererConfig; if (_customRendererConfig.processMintDataOnRenderer) { IHLRenderer(_customRendererConfig.renderer).processOneRecipientMint(tempSupply + 1, amount, recipient); } } /** * @notice See {IERC721GeneralMint-mintOneToMultipleRecipients} */ function mintOneToMultipleRecipients(address[] calldata recipients) external onlyMinter nonReentrant { if (_mintFrozen == 1) { _revert(MintFrozen.selector); } uint256 recipientsLength = recipients.length; uint256 tempSupply = _nextTokenId() - 1; // cache for (uint256 i = 0; i < recipientsLength; i++) { _mint(recipients[i], 1); } _requireLimitSupply(tempSupply + recipientsLength); // process mint on custom renderer if present CustomRendererConfig memory _customRendererConfig = customRendererConfig; if (_customRendererConfig.processMintDataOnRenderer) { IHLRenderer(_customRendererConfig.renderer).processMultipleRecipientMint(tempSupply + 1, 1, recipients); } } /** * @notice See {IERC721GeneralMint-mintSameAmountToMultipleRecipients} */ function mintSameAmountToMultipleRecipients( address[] calldata recipients, uint256 amount ) external onlyMinter nonReentrant { if (_mintFrozen == 1) { _revert(MintFrozen.selector); } uint256 recipientsLength = recipients.length; uint256 tempSupply = _nextTokenId() - 1; // cache for (uint256 i = 0; i < recipientsLength; i++) { _mint(recipients[i], amount); } _requireLimitSupply(tempSupply + recipientsLength * amount); // process mint on custom renderer if present CustomRendererConfig memory _customRendererConfig = customRendererConfig; if (_customRendererConfig.processMintDataOnRenderer) { IHLRenderer(_customRendererConfig.renderer).processMultipleRecipientMint( tempSupply + 1, amount, recipients ); } } /** * @notice Set custom renderer and processing config * @param _customRendererConfig New custom renderer config */ function setCustomRenderer(CustomRendererConfig calldata _customRendererConfig) external onlyOwner { require(_customRendererConfig.renderer != address(0), "Invalid input"); customRendererConfig = _customRendererConfig; } /** * @notice Override base URI system for select tokens, with custom per-token metadata * @param ids IDs of tokens to override base uri system for with custom uris * @param uris Custom uris */ function setTokenURIs(uint256[] calldata ids, string[] calldata uris) external nonReentrant { uint256 idsLength = ids.length; if (idsLength != uris.length) { _revert(MismatchedArrayLengths.selector); } for (uint256 i = 0; i < idsLength; i++) { _setTokenURI(ids[i], uris[i]); } emit TokenURIsSet(ids, uris); observability.emitTokenURIsSet(ids, uris); } /** * @notice Set base uri * @param newBaseURI New base uri to set */ function setBaseURI(string calldata newBaseURI) external nonReentrant { if (bytes(newBaseURI).length == 0) { _revert(EmptyString.selector); } address _manager = defaultManager; if (_manager == address(0)) { if (_msgSender() != owner()) { _revert(Unauthorized.selector); } } else { if (!ITokenManager(_manager).canUpdateMetadata(_msgSender(), 0, bytes(newBaseURI))) { _revert(Unauthorized.selector); } } _setBaseURI(newBaseURI); observability.emitBaseUriSet(newBaseURI); } /** * @notice Set limit supply * @param _limitSupply Limit supply to set */ function setLimitSupply(uint256 _limitSupply) external onlyOwner nonReentrant { // allow it to be 0, for post-mint limitSupply = _limitSupply; emit LimitSupplySet(_limitSupply); observability.emitLimitSupplySet(_limitSupply); } /** * @notice Set contract name * @param newName New name * @param newSymbol New symbol * @param newContractUri New contractURI */ function setContractMetadata( string calldata newName, string calldata newSymbol, string calldata newContractUri ) external onlyOwner { _setContractMetadata(newName, newSymbol); contractURI = newContractUri; observability.emitContractMetadataSet(newName, newSymbol, newContractUri); } /** * @notice Return the total number of minted tokens on the collection */ function supply() external view returns (uint256) { return ERC721AUpgradeable._totalMinted(); } /** * @notice See {IERC721-burn}. Overrides default behaviour to check associated tokenManager. */ function burn(uint256 tokenId) public nonReentrant { address _manager = tokenManager(tokenId); address msgSender = _msgSender(); if (_manager != address(0) && IERC165Upgradeable(_manager).supportsInterface(type(IPostBurn).interfaceId)) { address owner = ownerOf(tokenId); IPostBurn(_manager).postBurn(msgSender, owner, tokenId); } else { // default to restricting burn to owner or operator if a valid TM isn't present if (!_isApprovedOrOwner(msgSender, tokenId)) { _revert(Unauthorized.selector); } } _burn(tokenId); observability.emitTransfer(msgSender, address(0), tokenId); } /** * @notice Overrides tokenURI to first rotate the token id * @param tokenId ID of token to get uri for */ function tokenURI(uint256 tokenId) public view virtual override returns (string memory) { if (customRendererConfig.renderer != address(0)) { return IHLRenderer(customRendererConfig.renderer).tokenURI(tokenId); } return ERC721AURIStorageUpgradeable.tokenURI(tokenId); } /** * @notice See {IERC165-supportsInterface}. */ function supportsInterface( bytes4 interfaceId ) public view virtual override(IERC165Upgradeable, ERC721AUpgradeable) returns (bool) { return ERC721AUpgradeable.supportsInterface(interfaceId); } /** * @notice Hook called after transfers * @param from Account token is being transferred from * @param to Account token is being transferred to * @param tokenId ID of token being transferred */ function _afterTokenTransfers(address from, address to, uint256 tokenId) internal override { address _manager = tokenManager(tokenId); if (_manager != address(0) && IERC165Upgradeable(_manager).supportsInterface(type(IPostTransfer).interfaceId)) { IPostTransfer(_manager).postSafeTransferFrom(_msgSender(), from, to, tokenId, ""); } observability.emitTransfer(from, to, tokenId); } /** * @notice Used for meta-transactions */ function _msgSender() internal view virtual override(ERC721Base, ContextUpgradeable) returns (address sender) { return ERC2771ContextUpgradeable._msgSender(); } /** * @notice Used for meta-transactions */ function _msgData() internal view virtual override(ERC721Base, ContextUpgradeable) returns (bytes calldata) { return ERC2771ContextUpgradeable._msgData(); } /** * @dev For more efficient reverts. */ function _revert(bytes4 errorSelector) internal pure virtual override(ERC721AUpgradeable, ERC721Base) { ERC721AUpgradeable._revert(errorSelector); } /** * @notice Override base URI system for select tokens, with custom per-token metadata * @param tokenId Token to set uri for * @param _uri Uri to set on token */ function _setTokenURI(uint256 tokenId, string calldata _uri) private { address _manager = tokenManager(tokenId); address msgSender = _msgSender(); address tempOwner = owner(); if (_manager == address(0)) { if (msgSender != tempOwner) { _revert(Unauthorized.selector); } } else { if (!ITokenManager(_manager).canUpdateMetadata(msgSender, tokenId, bytes(_uri))) { _revert(Unauthorized.selector); } } _tokenURIs[tokenId] = _uri; } /** * @notice Require the new supply of tokens after mint to be less than limit supply * @param newSupply New supply */ function _requireLimitSupply(uint256 newSupply) internal view { uint256 _limitSupply = limitSupply; if (_limitSupply != 0 && newSupply > _limitSupply) { _revert(OverLimitSupply.selector); } } }
// SPDX-License-Identifier: MIT pragma solidity 0.8.10; /** * @author highlight.xyz * @notice If token managers implement this, transfer actions will call * postSafeTransferFrom or postTransferFrom on the token manager. */ interface IPostTransfer { /** * @notice Hook called by community after safe transfers, if token manager of transferred token implements this * interface. * @param operator Operator transferring tokens * @param from Token(s) sender * @param to Token(s) recipient * @param id Transferred token's id * @param data Arbitrary data */ function postSafeTransferFrom(address operator, address from, address to, uint256 id, bytes memory data) external; /** * @notice Hook called by community after transfers, if token manager of transferred token implements * this interface. * @param operator Operator transferring tokens * @param from Token(s) sender * @param to Token(s) recipient * @param id Transferred token's id */ function postTransferFrom(address operator, address from, address to, uint256 id) external; }
// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; /** * @notice General721 mint interface * @author highlight.xyz */ interface IERC721GeneralMint { /** * @notice Mint one token to one recipient * @param recipient Recipient of minted NFT */ function mintOneToOneRecipient(address recipient) external returns (uint256); /** * @notice Mint an amount of tokens to one recipient * @param recipient Recipient of minted NFTs * @param amount Amount of NFTs minted */ function mintAmountToOneRecipient(address recipient, uint256 amount) external; /** * @notice Mint one token to multiple recipients. Useful for use-cases like airdrops * @param recipients Recipients of minted NFTs */ function mintOneToMultipleRecipients(address[] calldata recipients) external; /** * @notice Mint the same amount of tokens to multiple recipients * @param recipients Recipients of minted NFTs * @param amount Amount of NFTs minted to each recipient */ function mintSameAmountToMultipleRecipients(address[] calldata recipients, uint256 amount) external; /** * @notice Mint a chosen token id to a single recipient * @param recipient Recipient of chosen NFT * @param tokenId ID of NFT to mint */ function mintSpecificTokenToOneRecipient(address recipient, uint256 tokenId) external; /** * @notice Mint chosen token ids to a single recipient * @param recipient Recipient of chosen NFT * @param tokenIds IDs of NFTs to mint */ function mintSpecificTokensToOneRecipient(address recipient, uint256[] calldata tokenIds) external; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (access/Ownable.sol) pragma solidity 0.8.10; import "@openzeppelin/contracts/utils/Context.sol"; /** * @dev Contract module which provides a basic access control mechanism, where * there is an account (an owner) that can be granted exclusive access to * specific functions. * * By default, the owner account will be the one that deploys the contract. This * can later be changed with {transferOwnership}. * * This module is used through inheritance. It will make available the modifier * `onlyOwner`, which can be applied to your functions to restrict their use to * the owner. */ /* solhint-disable */ abstract contract Ownable is Context { address private _owner; event OwnershipTransferred(address indexed previousOwner, address indexed newOwner); /** * @dev Initializes the contract setting the deployer as the initial owner. */ constructor() { _transferOwnership(_msgSender()); } /** * @dev Throws if called by any account other than the owner. */ modifier onlyOwner() { _checkOwner(); _; } /** * @dev Returns the address of the current owner. */ function owner() public view virtual returns (address) { return _owner; } /** * @dev Throws if the sender is not the owner. */ function _checkOwner() internal view virtual { require(owner() == _msgSender(), "Ownable: caller is not the owner"); } /** * @dev Leaves the contract without owner. It will not be possible to call * `onlyOwner` functions anymore. Can only be called by the current owner. * * NOTE: Renouncing ownership will leave the contract without an owner, * thereby removing any functionality that is only available to the owner. */ function renounceOwnership() public virtual onlyOwner { _transferOwnership(address(0)); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Can only be called by the current owner. */ function transferOwnership(address newOwner) public virtual onlyOwner { require(newOwner != address(0), "Ownable: new owner is the zero address"); _transferOwnership(newOwner); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Internal function without access restriction. */ function _transferOwnership(address newOwner) internal virtual { address oldOwner = _owner; _owner = newOwner; emit OwnershipTransferred(oldOwner, newOwner); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.5.0) (metatx/ERC2771Context.sol) pragma solidity 0.8.10; import "@openzeppelin/contracts-upgradeable/proxy/utils/Initializable.sol"; /** * @dev Context variant with ERC2771 support. * Openzeppelin contract slightly modified by ishan@ highlight.xyz to be upgradeable. */ abstract contract ERC2771ContextUpgradeable is Initializable { address private _trustedForwarder; /** * @dev This empty reserved space is put in place to allow future versions to add new * variables without shifting down storage in the inheritance chain. * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps */ uint256[49] private __gap; function isTrustedForwarder(address forwarder) public view virtual returns (bool) { return forwarder == _trustedForwarder; } function __ERC2771ContextUpgradeable__init__(address trustedForwarder) internal onlyInitializing { _trustedForwarder = trustedForwarder; } function _msgSender() internal view virtual returns (address sender) { if (isTrustedForwarder(msg.sender)) { // The assembly code is more direct than the Solidity version using `abi.decode`. /* solhint-disable no-inline-assembly */ assembly { sender := shr(96, calldataload(sub(calldatasize(), 20))) } /* solhint-enable no-inline-assembly */ } else { return msg.sender; } } function _msgData() internal view virtual returns (bytes calldata) { if (isTrustedForwarder(msg.sender)) { return msg.data[:msg.data.length - 20]; } else { return msg.data; } } }
// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; /** * @title ITokenManager * @author highlight.xyz * @notice Enables interfacing with custom token managers */ interface ITokenManager { /** * @notice Returns whether metadata updater is allowed to update * @param sender Updater * @param id Token/edition who's uri is being updated * If id is 0, implementation should decide behaviour for base uri update * @param newData Token's new uri if called by general contract, and any metadata field if called by editions * @return If invocation can update metadata */ function canUpdateMetadata(address sender, uint256 id, bytes calldata newData) external returns (bool); /** * @notice Returns whether token manager can be swapped for another one by invocator * @notice Default token manager implementations should ignore id * @param sender Swapper * @param id Token grouping id (token id or edition id) * @param newTokenManager New token manager being swapped to * @return If invocation can swap token managers */ function canSwap(address sender, uint256 id, address newTokenManager) external returns (bool); /** * @notice Returns whether token manager can be removed * @notice Default token manager implementations should ignore id * @param sender Swapper * @param id Token grouping id (token id or edition id) * @return If invocation can remove token manager */ function canRemoveItself(address sender, uint256 id) external returns (bool); }
// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; import "../royaltyManager/interfaces/IRoyaltyManager.sol"; /** * @title IObservability * @author highlight.xyz * @notice Interface to interact with the Highlight observability singleton * @dev Singleton to coalesce select Highlight protocol events */ interface IObservability { /************************** ERC721Base / ERC721MinimizedBase events **************************/ /** * @notice Emitted when minter is registered or unregistered * @param contractAddress Initial contract that emitted event * @param minter Minter that was changed * @param registered True if the minter was registered, false if unregistered */ event MinterRegistrationChanged(address indexed contractAddress, address indexed minter, bool indexed registered); /** * @notice Emitted when token managers are set for token/edition ids * @param contractAddress Initial contract that emitted event * @param _ids Edition / token ids * @param _tokenManagers Token managers to set for tokens / editions */ event GranularTokenManagersSet(address indexed contractAddress, uint256[] _ids, address[] _tokenManagers); /** * @notice Emitted when token managers are removed for token/edition ids * @param contractAddress Initial contract that emitted event * @param _ids Edition / token ids to remove token managers for */ event GranularTokenManagersRemoved(address indexed contractAddress, uint256[] _ids); /** * @notice Emitted when default token manager changed * @param contractAddress Initial contract that emitted event * @param newDefaultTokenManager New default token manager. Zero address if old one was removed */ event DefaultTokenManagerChanged(address indexed contractAddress, address indexed newDefaultTokenManager); /** * @notice Emitted when default royalty is set * @param contractAddress Initial contract that emitted event * @param recipientAddress Royalty recipient * @param royaltyPercentageBPS Percentage of sale (in basis points) owed to royalty recipient */ event DefaultRoyaltySet( address indexed contractAddress, address indexed recipientAddress, uint16 indexed royaltyPercentageBPS ); /** * @notice Emitted when royalties are set for edition / token ids * @param contractAddress Initial contract that emitted event * @param ids Token / edition ids * @param _newRoyalties New royalties for each token / edition */ event GranularRoyaltiesSet(address indexed contractAddress, uint256[] ids, IRoyaltyManager.Royalty[] _newRoyalties); /** * @notice Emitted when royalty manager is updated * @param contractAddress Initial contract that emitted event * @param newRoyaltyManager New royalty manager. Zero address if old one was removed */ event RoyaltyManagerChanged(address indexed contractAddress, address indexed newRoyaltyManager); /** * @notice Emitted when mints are frozen permanently * @param contractAddress Initial contract that emitted event */ event MintsFrozen(address indexed contractAddress); /** * @notice Emitted when contract metadata is set * @param contractAddress Initial contract that emitted event * @param name New name * @param symbol New symbol * @param contractURI New contract uri */ event ContractMetadataSet(address indexed contractAddress, string name, string symbol, string contractURI); /************************** ERC721General events **************************/ /** * @notice Emitted when hashed metadata config is set * @param contractAddress Initial contract that emitted event * @param hashedURIData Hashed uri data * @param hashedRotationData Hashed rotation key * @param _supply Supply of tokens to mint w/ reveal */ event HashedMetadataConfigSet( address indexed contractAddress, bytes hashedURIData, bytes hashedRotationData, uint256 indexed _supply ); /** * @notice Emitted when metadata is revealed * @param contractAddress Initial contract that emitted event * @param key Key used to decode hashed data * @param newRotationKey Actual rotation key to be used */ event Revealed(address indexed contractAddress, bytes key, uint256 newRotationKey); /************************** ERC721GeneralBase events **************************/ /** * @notice Emitted when uris are set for tokens * @param contractAddress Initial contract that emitted event * @param ids IDs of tokens to set uris for * @param uris Uris to set on tokens */ event TokenURIsSet(address indexed contractAddress, uint256[] ids, string[] uris); /** * @notice Emitted when limit supply is set * @param contractAddress Initial contract that emitted event * @param newLimitSupply Limit supply to set */ event LimitSupplySet(address indexed contractAddress, uint256 indexed newLimitSupply); /************************** ERC721StorageUri events **************************/ /** * @notice Emits when a series collection has its base uri set * @param contractAddress Contract with updated base uri * @param newBaseUri New base uri */ event BaseUriSet(address indexed contractAddress, string newBaseUri); /************************** ERC721Editions / ERC721SingleEdition events **************************/ // Not adding EditionCreated, EditionMintedToOneRecipient, EditionMintedToRecipients // EditionCreated - handled by MetadataInitialized // EditionMintedToOneRecipient / EditionMintedToRecipients - handled via mint module events /************************** Deployment events **************************/ /** * @notice Emitted when Generative Series contract is deployed * @param deployer Contract deployer * @param contractAddress Address of contract that was deployed */ event GenerativeSeriesDeployed(address indexed deployer, address indexed contractAddress); /** * @notice Emitted when Series contract is deployed * @param deployer Contract deployer * @param contractAddress Address of contract that was deployed */ event SeriesDeployed(address indexed deployer, address indexed contractAddress); /** * @notice Emitted when MultipleEditions contract is deployed * @param deployer Contract deployer * @param contractAddress Address of contract that was deployed */ event MultipleEditionsDeployed(address indexed deployer, address indexed contractAddress); /** * @notice Emitted when SingleEdition contract is deployed * @param deployer Contract deployer * @param contractAddress Address of contract that was deployed */ event SingleEditionDeployed(address indexed deployer, address indexed contractAddress); /************************** ERC721 events **************************/ /** * @notice Emitted when `tokenId` token is transferred from `from` to `to` on contractAddress * @param contractAddress NFT contract token resides on * @param from Token sender * @param to Token receiver * @param tokenId Token being sent */ event Transfer(address indexed contractAddress, address indexed from, address to, uint256 indexed tokenId); /** * @notice Emit MinterRegistrationChanged */ function emitMinterRegistrationChanged(address minter, bool registered) external; /** * @notice Emit GranularTokenManagersSet */ function emitGranularTokenManagersSet(uint256[] calldata _ids, address[] calldata _tokenManagers) external; /** * @notice Emit GranularTokenManagersRemoved */ function emitGranularTokenManagersRemoved(uint256[] calldata _ids) external; /** * @notice Emit DefaultTokenManagerChanged */ function emitDefaultTokenManagerChanged(address newDefaultTokenManager) external; /** * @notice Emit DefaultRoyaltySet */ function emitDefaultRoyaltySet(address recipientAddress, uint16 royaltyPercentageBPS) external; /** * @notice Emit GranularRoyaltiesSet */ function emitGranularRoyaltiesSet( uint256[] calldata ids, IRoyaltyManager.Royalty[] calldata _newRoyalties ) external; /** * @notice Emit RoyaltyManagerChanged */ function emitRoyaltyManagerChanged(address newRoyaltyManager) external; /** * @notice Emit MintsFrozen */ function emitMintsFrozen() external; /** * @notice Emit ContractMetadataSet */ function emitContractMetadataSet( string calldata name, string calldata symbol, string calldata contractURI ) external; /** * @notice Emit HashedMetadataConfigSet */ function emitHashedMetadataConfigSet( bytes calldata hashedURIData, bytes calldata hashedRotationData, uint256 _supply ) external; /** * @notice Emit Revealed */ function emitRevealed(bytes calldata key, uint256 newRotationKey) external; /** * @notice Emit TokenURIsSet */ function emitTokenURIsSet(uint256[] calldata ids, string[] calldata uris) external; /** * @notice Emit LimitSupplySet */ function emitLimitSupplySet(uint256 newLimitSupply) external; /** * @notice Emit BaseUriSet */ function emitBaseUriSet(string calldata newBaseUri) external; /** * @notice Emit GenerativeSeriesDeployed */ function emitGenerativeSeriesDeployed(address contractAddress) external; /** * @notice Emit SeriesDeployed */ function emitSeriesDeployed(address contractAddress) external; /** * @notice Emit MultipleEditionsDeployed */ function emitMultipleEditionsDeployed(address contractAddress) external; /** * @notice Emit SingleEditionDeployed */ function emitSingleEditionDeployed(address contractAddress) external; /** * @notice Emit Transfer */ function emitTransfer(address from, address to, uint256 tokenId) external; }
// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; /** * @title IRoyaltyManager * @author highlight.xyz * @notice Enables interfacing with custom royalty managers that define conditions on setting royalties for * NFT contracts */ interface IRoyaltyManager { /** * @notice Struct containing values required to adhere to ERC-2981 * @param recipientAddress Royalty recipient - can be EOA, royalty splitter contract, etc. * @param royaltyPercentageBPS Royalty cut, in basis points */ struct Royalty { address recipientAddress; uint16 royaltyPercentageBPS; } /** * @notice Defines conditions around being able to swap royalty manager for another one * @param newRoyaltyManager New royalty manager being swapped in * @param sender msg sender */ function canSwap(address newRoyaltyManager, address sender) external view returns (bool); /** * @notice Defines conditions around being able to remove current royalty manager * @param sender msg sender */ function canRemoveItself(address sender) external view returns (bool); /** * @notice Defines conditions around being able to set granular royalty (per token or per edition) * @param id Edition / token ID whose royalty is being set * @param royalty Royalty being set * @param sender msg sender */ function canSetGranularRoyalty(uint256 id, Royalty calldata royalty, address sender) external view returns (bool); /** * @notice Defines conditions around being able to set default royalty * @param royalty Royalty being set * @param sender msg sender */ function canSetDefaultRoyalty(Royalty calldata royalty, address sender) external view returns (bool); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/introspection/ERC165Checker.sol) pragma solidity 0.8.10; import "./IERC165Upgradeable.sol"; /** * @dev Library used to query support of an interface declared via {IERC165}. * * Note that these functions return the actual result of the query: they do not * `revert` if an interface is not supported. It is up to the caller to decide * what to do in these cases. */ /* solhint-disable */ library ERC165CheckerUpgradeable { // As per the EIP-165 spec, no interface should ever match 0xffffffff bytes4 private constant _INTERFACE_ID_INVALID = 0xffffffff; /** * @dev Returns true if `account` supports the {IERC165} interface, */ function supportsERC165(address account) internal view returns (bool) { // Any contract that implements ERC165 must explicitly indicate support of // InterfaceId_ERC165 and explicitly indicate non-support of InterfaceId_Invalid return supportsERC165InterfaceUnchecked(account, type(IERC165Upgradeable).interfaceId) && !supportsERC165InterfaceUnchecked(account, _INTERFACE_ID_INVALID); } /** * @dev Returns true if `account` supports the interface defined by * `interfaceId`. Support for {IERC165} itself is queried automatically. * * See {IERC165-supportsInterface}. */ function supportsInterface(address account, bytes4 interfaceId) internal view returns (bool) { // query support of both ERC165 as per the spec and support of _interfaceId return supportsERC165(account) && supportsERC165InterfaceUnchecked(account, interfaceId); } /** * @dev Returns a boolean array where each value corresponds to the * interfaces passed in and whether they're supported or not. This allows * you to batch check interfaces for a contract where your expectation * is that some interfaces may not be supported. * * See {IERC165-supportsInterface}. * * _Available since v3.4._ */ function getSupportedInterfaces( address account, bytes4[] memory interfaceIds ) internal view returns (bool[] memory) { // an array of booleans corresponding to interfaceIds and whether they're supported or not bool[] memory interfaceIdsSupported = new bool[](interfaceIds.length); // query support of ERC165 itself if (supportsERC165(account)) { // query support of each interface in interfaceIds for (uint256 i = 0; i < interfaceIds.length; i++) { interfaceIdsSupported[i] = supportsERC165InterfaceUnchecked(account, interfaceIds[i]); } } return interfaceIdsSupported; } /** * @dev Returns true if `account` supports all the interfaces defined in * `interfaceIds`. Support for {IERC165} itself is queried automatically. * * Batch-querying can lead to gas savings by skipping repeated checks for * {IERC165} support. * * See {IERC165-supportsInterface}. */ function supportsAllInterfaces(address account, bytes4[] memory interfaceIds) internal view returns (bool) { // query support of ERC165 itself if (!supportsERC165(account)) { return false; } // query support of each interface in _interfaceIds for (uint256 i = 0; i < interfaceIds.length; i++) { if (!supportsERC165InterfaceUnchecked(account, interfaceIds[i])) { return false; } } // all interfaces supported return true; } /** * @notice Query if a contract implements an interface, does not check ERC165 support * @param account The address of the contract to query for support of an interface * @param interfaceId The interface identifier, as specified in ERC-165 * @return true if the contract at account indicates support of the interface with * identifier interfaceId, false otherwise * @dev Assumes that account contains a contract that supports ERC165, otherwise * the behavior of this method is undefined. This precondition can be checked * with {supportsERC165}. * Interface identification is specified in ERC-165. */ function supportsERC165InterfaceUnchecked(address account, bytes4 interfaceId) internal view returns (bool) { // prepare call bytes memory encodedParams = abi.encodeWithSelector(IERC165Upgradeable.supportsInterface.selector, interfaceId); // perform static call bool success; uint256 returnSize; uint256 returnValue; assembly { success := staticcall(30000, account, add(encodedParams, 0x20), mload(encodedParams), 0x00, 0x20) returnSize := returndatasize() returnValue := mload(0x00) } return success && returnSize >= 0x20 && returnValue > 0; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.6.0) (interfaces/IERC2981Upgradeable.sol) pragma solidity 0.8.10; import "../ERC165/IERC165Upgradeable.sol"; /** * @dev Interface for the NFT Royalty Standard. * * A standardized way to retrieve royalty payment information for non-fungible tokens (NFTs) to enable universal * support for royalty payments across all NFT marketplaces and ecosystem participants. * * _Available since v4.5._ */ interface IERC2981Upgradeable is IERC165Upgradeable { /** * @dev Returns how much royalty is owed and to whom, based on a sale price that may be denominated in any unit of * exchange. The royalty amount is denominated and should be paid in that same unit of exchange. */ function royaltyInfo( uint256 tokenId, uint256 salePrice ) external view returns (address receiver, uint256 royaltyAmount); }
// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; /** * @notice General721 mint interface for sequentially minted collections * @author highlight.xyz */ interface IERC721GeneralSequenceMint { /** * @notice Mint one token to one recipient * @param recipient Recipient of minted NFT */ function mintOneToOneRecipient(address recipient) external returns (uint256); /** * @notice Mint an amount of tokens to one recipient * @param recipient Recipient of minted NFTs * @param amount Amount of NFTs minted */ function mintAmountToOneRecipient(address recipient, uint256 amount) external; /** * @notice Mint one token to multiple recipients. Useful for use-cases like airdrops * @param recipients Recipients of minted NFTs */ function mintOneToMultipleRecipients(address[] calldata recipients) external; /** * @notice Mint the same amount of tokens to multiple recipients * @param recipients Recipients of minted NFTs * @param amount Amount of NFTs minted to each recipient */ function mintSameAmountToMultipleRecipients(address[] calldata recipients, uint256 amount) external; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (token/ERC721/extensions/ERC721URIStorage.sol) pragma solidity 0.8.10; import "./ERC721AUpgradeable.sol"; import "@openzeppelin/contracts-upgradeable/proxy/utils/Initializable.sol"; import "@openzeppelin/contracts-upgradeable/utils/StringsUpgradeable.sol"; /** * @title Appending URI storage utilities onto template ERC721A contract * @author highlight.xyz, OpenZeppelin * @dev ERC721 token with storage based token URI management. OpenZeppelin template edited by Highlight */ /* solhint-disable */ abstract contract ERC721AURIStorageUpgradeable is Initializable, ERC721AUpgradeable { /** * @notice Throw when token doesn't exist */ error TokenDoesNotExist(); function __ERC721URIStorage_init() internal onlyInitializing {} function __ERC721URIStorage_init_unchained() internal onlyInitializing {} using StringsUpgradeable for uint256; // Optional mapping for token URIs mapping(uint256 => string) internal _tokenURIs; /** * @dev Hashed rotation key data */ bytes internal _hashedRotationKeyData; /** * @dev Hashed base uri data */ bytes internal _hashedBaseURIData; /** * @dev Rotation key */ uint256 internal _rotationKey; /** * @dev Contract baseURI */ string public baseURI; /** * @notice Emitted when base uri is set * @param oldBaseUri Old base uri * @param newBaseURI New base uri */ event BaseURISet(string oldBaseUri, string newBaseURI); /** * @dev Set contract baseURI */ function _setBaseURI(string memory newBaseURI) internal { emit BaseURISet(baseURI, newBaseURI); baseURI = newBaseURI; } /** * @dev See {IERC721Metadata-tokenURI}. */ function tokenURI(uint256 tokenId) public view virtual override returns (string memory) { if (!_exists(tokenId)) { _revert(TokenDoesNotExist.selector); } string memory _tokenURI = _tokenURIs[tokenId]; // If there is no token URI, return the base URI. if (bytes(_tokenURI).length == 0) { return super.tokenURI(tokenId); } return _tokenURI; } /** * @dev See {ERC721-_burn}. This override additionally checks to see if a * token-specific URI was set for the token, and if so, it deletes the token URI from * the storage mapping. */ function _burn(uint256 tokenId) internal virtual override { super._burn(tokenId); if (bytes(_tokenURIs[tokenId]).length != 0) { delete _tokenURIs[tokenId]; } } function _baseURI() internal view override returns (string memory) { return baseURI; } /** * @dev This empty reserved space is put in place to allow future versions to add new * variables without shifting down storage in the inheritance chain. * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps */ uint256[49] private __gap; }
//SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; /** * @notice Highlight's custom renderer interface for collections */ interface IHLRenderer { /** * @notice Process a mint to multiple recipients (likely store mint details) * @dev Implementations should assume msg.sender to be the NFT contract * @param firstTokenId ID of first token to be minted (next ones are minted sequentially) * @param numTokensPerRecipient Number of tokens minted to each recipient * @param orderedRecipients Recipients to mint tokens to, sequentially */ function processMultipleRecipientMint( uint256 firstTokenId, uint256 numTokensPerRecipient, address[] calldata orderedRecipients ) external; /** * @notice Process a mint to one recipient (likely store mint details) * @dev Implementations should assume msg.sender to be the NFT contract * @param firstTokenId ID of first token to be minted (next ones are minted sequentially) * @param numTokens Number of tokens minted * @param recipient Recipient to mint to */ function processOneRecipientMint(uint256 firstTokenId, uint256 numTokens, address recipient) external; /** * @notice Return token metadata for a token * @dev Implementations should assume msg.sender to be the NFT contract * @param tokenId ID of token to return metadata for */ function tokenURI(uint256 tokenId) external view returns (string memory); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (security/ReentrancyGuard.sol) pragma solidity ^0.8.0; import {Initializable} from "../proxy/utils/Initializable.sol"; /** * @dev Contract module that helps prevent reentrant calls to a function. * * Inheriting from `ReentrancyGuard` will make the {nonReentrant} modifier * available, which can be applied to functions to make sure there are no nested * (reentrant) calls to them. * * Note that because there is a single `nonReentrant` guard, functions marked as * `nonReentrant` may not call one another. This can be worked around by making * those functions `private`, and then adding `external` `nonReentrant` entry * points to them. * * TIP: If you would like to learn more about reentrancy and alternative ways * to protect against it, check out our blog post * https://blog.openzeppelin.com/reentrancy-after-istanbul/[Reentrancy After Istanbul]. */ abstract contract ReentrancyGuardUpgradeable is Initializable { // Booleans are more expensive than uint256 or any type that takes up a full // word because each write operation emits an extra SLOAD to first read the // slot's contents, replace the bits taken up by the boolean, and then write // back. This is the compiler's defense against contract upgrades and // pointer aliasing, and it cannot be disabled. // The values being non-zero value makes deployment a bit more expensive, // but in exchange the refund on every call to nonReentrant will be lower in // amount. Since refunds are capped to a percentage of the total // transaction's gas, it is best to keep them low in cases like this one, to // increase the likelihood of the full refund coming into effect. uint256 private constant _NOT_ENTERED = 1; uint256 private constant _ENTERED = 2; uint256 private _status; function __ReentrancyGuard_init() internal onlyInitializing { __ReentrancyGuard_init_unchained(); } function __ReentrancyGuard_init_unchained() internal onlyInitializing { _status = _NOT_ENTERED; } /** * @dev Prevents a contract from calling itself, directly or indirectly. * Calling a `nonReentrant` function from another `nonReentrant` * function is not supported. It is possible to prevent this from happening * by making the `nonReentrant` function external, and making it call a * `private` function that does the actual work. */ modifier nonReentrant() { _nonReentrantBefore(); _; _nonReentrantAfter(); } function _nonReentrantBefore() private { // On the first call to nonReentrant, _status will be _NOT_ENTERED require(_status != _ENTERED, "ReentrancyGuard: reentrant call"); // Any calls to nonReentrant after this point will fail _status = _ENTERED; } function _nonReentrantAfter() private { // By storing the original value once again, a refund is triggered (see // https://eips.ethereum.org/EIPS/eip-2200) _status = _NOT_ENTERED; } /** * @dev Returns true if the reentrancy guard is currently set to "entered", which indicates there is a * `nonReentrant` function in the call stack. */ function _reentrancyGuardEntered() internal view returns (bool) { return _status == _ENTERED; } /** * @dev This empty reserved space is put in place to allow future versions to add new * variables without shifting down storage in the inheritance chain. * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps */ uint256[49] private __gap; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (access/Ownable.sol) pragma solidity ^0.8.0; import "../utils/ContextUpgradeable.sol"; import {Initializable} from "../proxy/utils/Initializable.sol"; /** * @dev Contract module which provides a basic access control mechanism, where * there is an account (an owner) that can be granted exclusive access to * specific functions. * * By default, the owner account will be the one that deploys the contract. This * can later be changed with {transferOwnership}. * * This module is used through inheritance. It will make available the modifier * `onlyOwner`, which can be applied to your functions to restrict their use to * the owner. */ abstract contract OwnableUpgradeable is Initializable, ContextUpgradeable { address private _owner; event OwnershipTransferred(address indexed previousOwner, address indexed newOwner); /** * @dev Initializes the contract setting the deployer as the initial owner. */ function __Ownable_init() internal onlyInitializing { __Ownable_init_unchained(); } function __Ownable_init_unchained() internal onlyInitializing { _transferOwnership(_msgSender()); } /** * @dev Throws if called by any account other than the owner. */ modifier onlyOwner() { _checkOwner(); _; } /** * @dev Returns the address of the current owner. */ function owner() public view virtual returns (address) { return _owner; } /** * @dev Throws if the sender is not the owner. */ function _checkOwner() internal view virtual { require(owner() == _msgSender(), "Ownable: caller is not the owner"); } /** * @dev Leaves the contract without owner. It will not be possible to call * `onlyOwner` functions. Can only be called by the current owner. * * NOTE: Renouncing ownership will leave the contract without an owner, * thereby disabling any functionality that is only available to the owner. */ function renounceOwnership() public virtual onlyOwner { _transferOwnership(address(0)); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Can only be called by the current owner. */ function transferOwnership(address newOwner) public virtual onlyOwner { require(newOwner != address(0), "Ownable: new owner is the zero address"); _transferOwnership(newOwner); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Internal function without access restriction. */ function _transferOwnership(address newOwner) internal virtual { address oldOwner = _owner; _owner = newOwner; emit OwnershipTransferred(oldOwner, newOwner); } /** * @dev This empty reserved space is put in place to allow future versions to add new * variables without shifting down storage in the inheritance chain. * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps */ uint256[49] private __gap; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (proxy/utils/UUPSUpgradeable.sol) pragma solidity ^0.8.0; import "../../interfaces/draft-IERC1822.sol"; import "../ERC1967/ERC1967Upgrade.sol"; /** * @dev An upgradeability mechanism designed for UUPS proxies. The functions included here can perform an upgrade of an * {ERC1967Proxy}, when this contract is set as the implementation behind such a proxy. * * A security mechanism ensures that an upgrade does not turn off upgradeability accidentally, although this risk is * reinstated if the upgrade retains upgradeability but removes the security mechanism, e.g. by replacing * `UUPSUpgradeable` with a custom implementation of upgrades. * * The {_authorizeUpgrade} function must be overridden to include access restriction to the upgrade mechanism. * * _Available since v4.1._ */ abstract contract UUPSUpgradeable is IERC1822Proxiable, ERC1967Upgrade { /// @custom:oz-upgrades-unsafe-allow state-variable-immutable state-variable-assignment address private immutable __self = address(this); /** * @dev Check that the execution is being performed through a delegatecall call and that the execution context is * a proxy contract with an implementation (as defined in ERC1967) pointing to self. This should only be the case * for UUPS and transparent proxies that are using the current contract as their implementation. Execution of a * function through ERC1167 minimal proxies (clones) would not normally pass this test, but is not guaranteed to * fail. */ modifier onlyProxy() { require(address(this) != __self, "Function must be called through delegatecall"); require(_getImplementation() == __self, "Function must be called through active proxy"); _; } /** * @dev Check that the execution is not being performed through a delegate call. This allows a function to be * callable on the implementing contract but not through proxies. */ modifier notDelegated() { require(address(this) == __self, "UUPSUpgradeable: must not be called through delegatecall"); _; } /** * @dev Implementation of the ERC1822 {proxiableUUID} function. This returns the storage slot used by the * implementation. It is used to validate the implementation's compatibility when performing an upgrade. * * IMPORTANT: A proxy pointing at a proxiable contract should not be considered proxiable itself, because this risks * bricking a proxy that upgrades to it, by delegating to itself until out of gas. Thus it is critical that this * function revert if invoked through a proxy. This is guaranteed by the `notDelegated` modifier. */ function proxiableUUID() external view virtual override notDelegated returns (bytes32) { return _IMPLEMENTATION_SLOT; } /** * @dev Upgrade the implementation of the proxy to `newImplementation`. * * Calls {_authorizeUpgrade}. * * Emits an {Upgraded} event. * * @custom:oz-upgrades-unsafe-allow-reachable delegatecall */ function upgradeTo(address newImplementation) public virtual onlyProxy { _authorizeUpgrade(newImplementation); _upgradeToAndCallUUPS(newImplementation, new bytes(0), false); } /** * @dev Upgrade the implementation of the proxy to `newImplementation`, and subsequently execute the function call * encoded in `data`. * * Calls {_authorizeUpgrade}. * * Emits an {Upgraded} event. * * @custom:oz-upgrades-unsafe-allow-reachable delegatecall */ function upgradeToAndCall(address newImplementation, bytes memory data) public payable virtual onlyProxy { _authorizeUpgrade(newImplementation); _upgradeToAndCallUUPS(newImplementation, data, true); } /** * @dev Function that should revert when `msg.sender` is not authorized to upgrade the contract. Called by * {upgradeTo} and {upgradeToAndCall}. * * Normally, this function will use an xref:access.adoc[access control] modifier such as {Ownable-onlyOwner}. * * ```solidity * function _authorizeUpgrade(address) internal override onlyOwner {} * ``` */ function _authorizeUpgrade(address newImplementation) internal virtual; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (utils/structs/EnumerableSet.sol) // This file was procedurally generated from scripts/generate/templates/EnumerableSet.js. pragma solidity ^0.8.0; /** * @dev Library for managing * https://en.wikipedia.org/wiki/Set_(abstract_data_type)[sets] of primitive * types. * * Sets have the following properties: * * - Elements are added, removed, and checked for existence in constant time * (O(1)). * - Elements are enumerated in O(n). No guarantees are made on the ordering. * * ```solidity * contract Example { * // Add the library methods * using EnumerableSet for EnumerableSet.AddressSet; * * // Declare a set state variable * EnumerableSet.AddressSet private mySet; * } * ``` * * As of v3.3.0, sets of type `bytes32` (`Bytes32Set`), `address` (`AddressSet`) * and `uint256` (`UintSet`) are supported. * * [WARNING] * ==== * Trying to delete such a structure from storage will likely result in data corruption, rendering the structure * unusable. * See https://github.com/ethereum/solidity/pull/11843[ethereum/solidity#11843] for more info. * * In order to clean an EnumerableSet, you can either remove all elements one by one or create a fresh instance using an * array of EnumerableSet. * ==== */ library EnumerableSet { // To implement this library for multiple types with as little code // repetition as possible, we write it in terms of a generic Set type with // bytes32 values. // The Set implementation uses private functions, and user-facing // implementations (such as AddressSet) are just wrappers around the // underlying Set. // This means that we can only create new EnumerableSets for types that fit // in bytes32. struct Set { // Storage of set values bytes32[] _values; // Position of the value in the `values` array, plus 1 because index 0 // means a value is not in the set. mapping(bytes32 => uint256) _indexes; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function _add(Set storage set, bytes32 value) private returns (bool) { if (!_contains(set, value)) { set._values.push(value); // The value is stored at length-1, but we add 1 to all indexes // and use 0 as a sentinel value set._indexes[value] = set._values.length; return true; } else { return false; } } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function _remove(Set storage set, bytes32 value) private returns (bool) { // We read and store the value's index to prevent multiple reads from the same storage slot uint256 valueIndex = set._indexes[value]; if (valueIndex != 0) { // Equivalent to contains(set, value) // To delete an element from the _values array in O(1), we swap the element to delete with the last one in // the array, and then remove the last element (sometimes called as 'swap and pop'). // This modifies the order of the array, as noted in {at}. uint256 toDeleteIndex = valueIndex - 1; uint256 lastIndex = set._values.length - 1; if (lastIndex != toDeleteIndex) { bytes32 lastValue = set._values[lastIndex]; // Move the last value to the index where the value to delete is set._values[toDeleteIndex] = lastValue; // Update the index for the moved value set._indexes[lastValue] = valueIndex; // Replace lastValue's index to valueIndex } // Delete the slot where the moved value was stored set._values.pop(); // Delete the index for the deleted slot delete set._indexes[value]; return true; } else { return false; } } /** * @dev Returns true if the value is in the set. O(1). */ function _contains(Set storage set, bytes32 value) private view returns (bool) { return set._indexes[value] != 0; } /** * @dev Returns the number of values on the set. O(1). */ function _length(Set storage set) private view returns (uint256) { return set._values.length; } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function _at(Set storage set, uint256 index) private view returns (bytes32) { return set._values[index]; } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function _values(Set storage set) private view returns (bytes32[] memory) { return set._values; } // Bytes32Set struct Bytes32Set { Set _inner; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function add(Bytes32Set storage set, bytes32 value) internal returns (bool) { return _add(set._inner, value); } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function remove(Bytes32Set storage set, bytes32 value) internal returns (bool) { return _remove(set._inner, value); } /** * @dev Returns true if the value is in the set. O(1). */ function contains(Bytes32Set storage set, bytes32 value) internal view returns (bool) { return _contains(set._inner, value); } /** * @dev Returns the number of values in the set. O(1). */ function length(Bytes32Set storage set) internal view returns (uint256) { return _length(set._inner); } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(Bytes32Set storage set, uint256 index) internal view returns (bytes32) { return _at(set._inner, index); } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function values(Bytes32Set storage set) internal view returns (bytes32[] memory) { bytes32[] memory store = _values(set._inner); bytes32[] memory result; /// @solidity memory-safe-assembly assembly { result := store } return result; } // AddressSet struct AddressSet { Set _inner; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function add(AddressSet storage set, address value) internal returns (bool) { return _add(set._inner, bytes32(uint256(uint160(value)))); } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function remove(AddressSet storage set, address value) internal returns (bool) { return _remove(set._inner, bytes32(uint256(uint160(value)))); } /** * @dev Returns true if the value is in the set. O(1). */ function contains(AddressSet storage set, address value) internal view returns (bool) { return _contains(set._inner, bytes32(uint256(uint160(value)))); } /** * @dev Returns the number of values in the set. O(1). */ function length(AddressSet storage set) internal view returns (uint256) { return _length(set._inner); } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(AddressSet storage set, uint256 index) internal view returns (address) { return address(uint160(uint256(_at(set._inner, index)))); } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function values(AddressSet storage set) internal view returns (address[] memory) { bytes32[] memory store = _values(set._inner); address[] memory result; /// @solidity memory-safe-assembly assembly { result := store } return result; } // UintSet struct UintSet { Set _inner; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function add(UintSet storage set, uint256 value) internal returns (bool) { return _add(set._inner, bytes32(value)); } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function remove(UintSet storage set, uint256 value) internal returns (bool) { return _remove(set._inner, bytes32(value)); } /** * @dev Returns true if the value is in the set. O(1). */ function contains(UintSet storage set, uint256 value) internal view returns (bool) { return _contains(set._inner, bytes32(value)); } /** * @dev Returns the number of values in the set. O(1). */ function length(UintSet storage set) internal view returns (uint256) { return _length(set._inner); } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(UintSet storage set, uint256 index) internal view returns (uint256) { return uint256(_at(set._inner, index)); } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function values(UintSet storage set) internal view returns (uint256[] memory) { bytes32[] memory store = _values(set._inner); uint256[] memory result; /// @solidity memory-safe-assembly assembly { result := store } return result; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/introspection/IERC165.sol) pragma solidity 0.8.10; /** * @dev Interface of the ERC165 standard, as defined in the * https://eips.ethereum.org/EIPS/eip-165[EIP]. * * Implementers can declare support of contract interfaces, which can then be * queried by others ({ERC165Checker}). * * For an implementation, see {ERC165}. */ /* solhint-disable */ interface IERC165Upgradeable { /** * @dev Returns true if this contract implements the interface defined by * `interfaceId`. See the corresponding * https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section] * to learn more about how these ids are created. * * This function call must use less than 30 000 gas. */ function supportsInterface(bytes4 interfaceId) external view returns (bool); }
// SPDX-License-Identifier: MIT // ERC721A Contracts v4.2.3 // Creator: Chiru Labs pragma solidity 0.8.10; import "./IERC721AUpgradeable.sol"; import { ERC721AStorage } from "./ERC721AStorage.sol"; import "@openzeppelin/contracts-upgradeable/proxy/utils/Initializable.sol"; import "@openzeppelin/contracts-upgradeable/utils/ContextUpgradeable.sol"; /* solhint-disable */ /** * @dev Interface of ERC721 token receiver. */ interface ERC721A__IERC721ReceiverUpgradeable { function onERC721Received( address operator, address from, uint256 tokenId, bytes calldata data ) external returns (bytes4); } /** * @title ERC721A * @author Chiru Labs, modified by [email protected] * * @dev Implementation of the [ERC721](https://eips.ethereum.org/EIPS/eip-721) * Non-Fungible Token Standard, including the Metadata extension. * Optimized for lower gas during batch mints. * * Token IDs are minted in sequential order (e.g. 0, 1, 2, 3, ...) * starting from `_startTokenId()`. * * Assumptions: * * - An owner cannot have more than 2**64 - 1 (max value of uint64) of supply. * - The maximum token ID cannot exceed 2**256 - 1 (max value of uint256). */ contract ERC721AUpgradeable is Initializable, IERC721AUpgradeable, ContextUpgradeable { using ERC721AStorage for ERC721AStorage.Layout; // ============================================================= // CONSTANTS // ============================================================= // Mask of an entry in packed address data. uint256 private constant _BITMASK_ADDRESS_DATA_ENTRY = (1 << 64) - 1; // The bit position of `numberMinted` in packed address data. uint256 private constant _BITPOS_NUMBER_MINTED = 64; // The bit position of `numberBurned` in packed address data. uint256 private constant _BITPOS_NUMBER_BURNED = 128; // The bit position of `aux` in packed address data. uint256 private constant _BITPOS_AUX = 192; // Mask of all 256 bits in packed address data except the 64 bits for `aux`. uint256 private constant _BITMASK_AUX_COMPLEMENT = (1 << 192) - 1; // The bit position of `startTimestamp` in packed ownership. uint256 private constant _BITPOS_START_TIMESTAMP = 160; // The bit mask of the `burned` bit in packed ownership. uint256 private constant _BITMASK_BURNED = 1 << 224; // The bit position of the `nextInitialized` bit in packed ownership. uint256 private constant _BITPOS_NEXT_INITIALIZED = 225; // The bit mask of the `nextInitialized` bit in packed ownership. uint256 private constant _BITMASK_NEXT_INITIALIZED = 1 << 225; // The bit position of `extraData` in packed ownership. uint256 private constant _BITPOS_EXTRA_DATA = 232; // Mask of all 256 bits in a packed ownership except the 24 bits for `extraData`. uint256 private constant _BITMASK_EXTRA_DATA_COMPLEMENT = (1 << 232) - 1; // The mask of the lower 160 bits for addresses. uint256 private constant _BITMASK_ADDRESS = (1 << 160) - 1; // The maximum `quantity` that can be minted with {_mintERC2309}. // This limit is to prevent overflows on the address data entries. // For a limit of 5000, a total of 3.689e15 calls to {_mintERC2309} // is required to cause an overflow, which is unrealistic. uint256 private constant _MAX_MINT_ERC2309_QUANTITY_LIMIT = 5000; // The `Transfer` event signature is given by: // `keccak256(bytes("Transfer(address,address,uint256)"))`. bytes32 private constant _TRANSFER_EVENT_SIGNATURE = 0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef; // ============================================================= // CONSTRUCTOR // ============================================================= function __ERC721A_init(string memory name_, string memory symbol_) internal onlyInitializing { __ERC721A_init_unchained(name_, symbol_); } function __ERC721A_init_unchained(string memory name_, string memory symbol_) internal onlyInitializing { ERC721AStorage.layout()._name = name_; ERC721AStorage.layout()._symbol = symbol_; ERC721AStorage.layout()._currentIndex = _startTokenId(); } // ============================================================= // TOKEN COUNTING OPERATIONS // ============================================================= /** * @dev Returns the starting token ID. * To change the starting token ID, please override this function. */ function _startTokenId() internal view virtual returns (uint256) { return 1; } /** * @dev Returns the next token ID to be minted. */ function _nextTokenId() internal view virtual returns (uint256) { return ERC721AStorage.layout()._currentIndex; } /** * @dev Returns the total number of tokens in existence. * Burned tokens will reduce the count. * To get the total number of tokens minted, please see {_totalMinted}. */ function totalSupply() public view virtual override returns (uint256) { // Counter underflow is impossible as _burnCounter cannot be incremented // more than `_currentIndex - _startTokenId()` times. unchecked { return ERC721AStorage.layout()._currentIndex - ERC721AStorage.layout()._burnCounter - _startTokenId(); } } /** * @dev Returns the total amount of tokens minted in the contract. */ function _totalMinted() internal view virtual returns (uint256) { // Counter underflow is impossible as `_currentIndex` does not decrement, // and it is initialized to `_startTokenId()`. unchecked { return ERC721AStorage.layout()._currentIndex - _startTokenId(); } } /** * @dev Returns the total number of tokens burned. */ function _totalBurned() internal view virtual returns (uint256) { return ERC721AStorage.layout()._burnCounter; } // ============================================================= // ADDRESS DATA OPERATIONS // ============================================================= /** * @dev Returns the number of tokens in `owner`'s account. */ function balanceOf(address owner) public view virtual override returns (uint256) { if (owner == address(0)) _revert(BalanceQueryForZeroAddress.selector); return ERC721AStorage.layout()._packedAddressData[owner] & _BITMASK_ADDRESS_DATA_ENTRY; } /** * Returns the number of tokens minted by `owner`. */ function _numberMinted(address owner) internal view returns (uint256) { return (ERC721AStorage.layout()._packedAddressData[owner] >> _BITPOS_NUMBER_MINTED) & _BITMASK_ADDRESS_DATA_ENTRY; } /** * Returns the number of tokens burned by or on behalf of `owner`. */ function _numberBurned(address owner) internal view returns (uint256) { return (ERC721AStorage.layout()._packedAddressData[owner] >> _BITPOS_NUMBER_BURNED) & _BITMASK_ADDRESS_DATA_ENTRY; } /** * Returns the auxiliary data for `owner`. (e.g. number of whitelist mint slots used). */ function _getAux(address owner) internal view returns (uint64) { return uint64(ERC721AStorage.layout()._packedAddressData[owner] >> _BITPOS_AUX); } /** * Sets the auxiliary data for `owner`. (e.g. number of whitelist mint slots used). * If there are multiple variables, please pack them into a uint64. */ function _setAux(address owner, uint64 aux) internal virtual { uint256 packed = ERC721AStorage.layout()._packedAddressData[owner]; uint256 auxCasted; // Cast `aux` with assembly to avoid redundant masking. assembly { auxCasted := aux } packed = (packed & _BITMASK_AUX_COMPLEMENT) | (auxCasted << _BITPOS_AUX); ERC721AStorage.layout()._packedAddressData[owner] = packed; } // ============================================================= // IERC165 // ============================================================= /** * @dev Returns true if this contract implements the interface defined by * `interfaceId`. See the corresponding * [EIP section](https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified) * to learn more about how these ids are created. * * This function call must use less than 30000 gas. */ function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) { // The interface IDs are constants representing the first 4 bytes // of the XOR of all function selectors in the interface. // See: [ERC165](https://eips.ethereum.org/EIPS/eip-165) // (e.g. `bytes4(i.functionA.selector ^ i.functionB.selector ^ ...)`) return interfaceId == 0x01ffc9a7 || // ERC165 interface ID for ERC165. interfaceId == 0x80ac58cd || // ERC165 interface ID for ERC721. interfaceId == 0x5b5e139f; // ERC165 interface ID for ERC721Metadata. } // ============================================================= // IERC721Metadata // ============================================================= /** * @dev Returns the token collection name. */ function name() public view virtual override returns (string memory) { return ERC721AStorage.layout()._name; } /** * @dev Returns the token collection symbol. */ function symbol() public view virtual override returns (string memory) { return ERC721AStorage.layout()._symbol; } /** * @dev Returns the Uniform Resource Identifier (URI) for `tokenId` token. */ function tokenURI(uint256 tokenId) public view virtual override returns (string memory) { if (!_exists(tokenId)) _revert(URIQueryForNonexistentToken.selector); string memory baseURI = _baseURI(); return bytes(baseURI).length != 0 ? string(abi.encodePacked(baseURI, "/", _toString(tokenId))) : ""; } /** * @dev Base URI for computing {tokenURI}. If set, the resulting URI for each * token will be the concatenation of the `baseURI` and the `tokenId`. Empty * by default, it can be overridden in child contracts. */ function _baseURI() internal view virtual returns (string memory) { return ""; } // ============================================================= // OWNERSHIPS OPERATIONS // ============================================================= /** * @dev Returns the owner of the `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function ownerOf(uint256 tokenId) public view virtual override returns (address) { return address(uint160(_packedOwnershipOf(tokenId))); } /** * @dev Added by Highlight to facilitate updating of name and symbol */ function _setContractMetadata(string calldata newName, string calldata newSymbol) internal { ERC721AStorage.layout()._name = newName; ERC721AStorage.layout()._symbol = newSymbol; } /** * @dev Gas spent here starts off proportional to the maximum mint batch size. * It gradually moves to O(1) as tokens get transferred around over time. */ function _ownershipOf(uint256 tokenId) internal view virtual returns (TokenOwnership memory) { return _unpackedOwnership(_packedOwnershipOf(tokenId)); } /** * @dev Returns the unpacked `TokenOwnership` struct at `index`. */ function _ownershipAt(uint256 index) internal view virtual returns (TokenOwnership memory) { return _unpackedOwnership(ERC721AStorage.layout()._packedOwnerships[index]); } /** * @dev Returns whether the ownership slot at `index` is initialized. * An uninitialized slot does not necessarily mean that the slot has no owner. */ function _ownershipIsInitialized(uint256 index) internal view virtual returns (bool) { return ERC721AStorage.layout()._packedOwnerships[index] != 0; } /** * @dev Initializes the ownership slot minted at `index` for efficiency purposes. */ function _initializeOwnershipAt(uint256 index) internal virtual { if (ERC721AStorage.layout()._packedOwnerships[index] == 0) { ERC721AStorage.layout()._packedOwnerships[index] = _packedOwnershipOf(index); } } /** * Returns the packed ownership data of `tokenId`. */ function _packedOwnershipOf(uint256 tokenId) private view returns (uint256 packed) { if (_startTokenId() <= tokenId) { packed = ERC721AStorage.layout()._packedOwnerships[tokenId]; // If the data at the starting slot does not exist, start the scan. if (packed == 0) { if (tokenId >= ERC721AStorage.layout()._currentIndex) _revert(OwnerQueryForNonexistentToken.selector); // Invariant: // There will always be an initialized ownership slot // (i.e. `ownership.addr != address(0) && ownership.burned == false`) // before an unintialized ownership slot // (i.e. `ownership.addr == address(0) && ownership.burned == false`) // Hence, `tokenId` will not underflow. // // We can directly compare the packed value. // If the address is zero, packed will be zero. for (;;) { unchecked { packed = ERC721AStorage.layout()._packedOwnerships[--tokenId]; } if (packed == 0) continue; if (packed & _BITMASK_BURNED == 0) return packed; // Otherwise, the token is burned, and we must revert. // This handles the case of batch burned tokens, where only the burned bit // of the starting slot is set, and remaining slots are left uninitialized. _revert(OwnerQueryForNonexistentToken.selector); } } // Otherwise, the data exists and we can skip the scan. // This is possible because we have already achieved the target condition. // This saves 2143 gas on transfers of initialized tokens. // If the token is not burned, return `packed`. Otherwise, revert. if (packed & _BITMASK_BURNED == 0) return packed; } _revert(OwnerQueryForNonexistentToken.selector); } /** * @dev Returns the unpacked `TokenOwnership` struct from `packed`. */ function _unpackedOwnership(uint256 packed) private pure returns (TokenOwnership memory ownership) { ownership.addr = address(uint160(packed)); ownership.startTimestamp = uint64(packed >> _BITPOS_START_TIMESTAMP); ownership.burned = packed & _BITMASK_BURNED != 0; ownership.extraData = uint24(packed >> _BITPOS_EXTRA_DATA); } /** * @dev Packs ownership data into a single uint256. */ function _packOwnershipData(address owner, uint256 flags) private view returns (uint256 result) { assembly { // Mask `owner` to the lower 160 bits, in case the upper bits somehow aren't clean. owner := and(owner, _BITMASK_ADDRESS) // `owner | (block.timestamp << _BITPOS_START_TIMESTAMP) | flags`. result := or(owner, or(shl(_BITPOS_START_TIMESTAMP, timestamp()), flags)) } } /** * @dev Returns the `nextInitialized` flag set if `quantity` equals 1. */ function _nextInitializedFlag(uint256 quantity) private pure returns (uint256 result) { // For branchless setting of the `nextInitialized` flag. assembly { // `(quantity == 1) << _BITPOS_NEXT_INITIALIZED`. result := shl(_BITPOS_NEXT_INITIALIZED, eq(quantity, 1)) } } // ============================================================= // APPROVAL OPERATIONS // ============================================================= /** * @dev Gives permission to `to` to transfer `tokenId` token to another account. See {ERC721A-_approve}. * * Requirements: * * - The caller must own the token or be an approved operator. */ function approve(address to, uint256 tokenId) public payable virtual override { _approve(to, tokenId, true); } /** * @dev Returns the account approved for `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function getApproved(uint256 tokenId) public view virtual override returns (address) { if (!_exists(tokenId)) _revert(ApprovalQueryForNonexistentToken.selector); return ERC721AStorage.layout()._tokenApprovals[tokenId].value; } /** * @dev Approve or remove `operator` as an operator for the caller. * Operators can call {transferFrom} or {safeTransferFrom} * for any token owned by the caller. * * Requirements: * * - The `operator` cannot be the caller. * * Emits an {ApprovalForAll} event. */ function setApprovalForAll(address operator, bool approved) public virtual override { ERC721AStorage.layout()._operatorApprovals[_msgSenderERC721A()][operator] = approved; emit ApprovalForAll(_msgSenderERC721A(), operator, approved); } /** * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`. * * See {setApprovalForAll}. */ function isApprovedForAll(address owner, address operator) public view virtual override returns (bool) { return ERC721AStorage.layout()._operatorApprovals[owner][operator]; } /** * @dev Returns whether `tokenId` exists. * * Tokens can be managed by their owner or approved accounts via {approve} or {setApprovalForAll}. * * Tokens start existing when they are minted. See {_mint}. */ function _exists(uint256 tokenId) internal view virtual returns (bool result) { if (_startTokenId() <= tokenId) { if (tokenId < ERC721AStorage.layout()._currentIndex) { uint256 packed; while ((packed = ERC721AStorage.layout()._packedOwnerships[tokenId]) == 0) --tokenId; result = packed & _BITMASK_BURNED == 0; } } } /** * @dev Returns whether `spender` is allowed to manage `tokenId`. * * Requirements: * * - `tokenId` must exist. */ function _isApprovedOrOwner(address spender, uint256 tokenId) internal view virtual returns (bool) { address owner = ownerOf(tokenId); return (isApprovedForAll(owner, spender) || _isSenderApprovedOrOwner(getApproved(tokenId), owner, spender)); } /** * @dev Returns whether `msgSender` is equal to `approvedAddress` or `owner`. */ function _isSenderApprovedOrOwner( address approvedAddress, address owner, address msgSender ) private pure returns (bool result) { assembly { // Mask `owner` to the lower 160 bits, in case the upper bits somehow aren't clean. owner := and(owner, _BITMASK_ADDRESS) // Mask `msgSender` to the lower 160 bits, in case the upper bits somehow aren't clean. msgSender := and(msgSender, _BITMASK_ADDRESS) // `msgSender == owner || msgSender == approvedAddress`. result := or(eq(msgSender, owner), eq(msgSender, approvedAddress)) } } /** * @dev Returns the storage slot and value for the approved address of `tokenId`. */ function _getApprovedSlotAndAddress( uint256 tokenId ) private view returns (uint256 approvedAddressSlot, address approvedAddress) { ERC721AStorage.TokenApprovalRef storage tokenApproval = ERC721AStorage.layout()._tokenApprovals[tokenId]; // The following is equivalent to `approvedAddress = _tokenApprovals[tokenId].value`. assembly { approvedAddressSlot := tokenApproval.slot approvedAddress := sload(approvedAddressSlot) } } // ============================================================= // TRANSFER OPERATIONS // ============================================================= /** * @dev Transfers `tokenId` from `from` to `to`. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must be owned by `from`. * - If the caller is not `from`, it must be approved to move this token * by either {approve} or {setApprovalForAll}. * * Emits a {Transfer} event. */ function transferFrom(address from, address to, uint256 tokenId) public payable virtual override { uint256 prevOwnershipPacked = _packedOwnershipOf(tokenId); // Mask `from` to the lower 160 bits, in case the upper bits somehow aren't clean. from = address(uint160(uint256(uint160(from)) & _BITMASK_ADDRESS)); if (address(uint160(prevOwnershipPacked)) != from) _revert(TransferFromIncorrectOwner.selector); (uint256 approvedAddressSlot, address approvedAddress) = _getApprovedSlotAndAddress(tokenId); // The nested ifs save around 20+ gas over a compound boolean condition. if (!_isSenderApprovedOrOwner(approvedAddress, from, _msgSenderERC721A())) if (!isApprovedForAll(from, _msgSenderERC721A())) _revert(TransferCallerNotOwnerNorApproved.selector); // Clear approvals from the previous owner. assembly { if approvedAddress { // This is equivalent to `delete _tokenApprovals[tokenId]`. sstore(approvedAddressSlot, 0) } } // Underflow of the sender's balance is impossible because we check for // ownership above and the recipient's balance can't realistically overflow. // Counter overflow is incredibly unrealistic as `tokenId` would have to be 2**256. unchecked { // We can directly increment and decrement the balances. --ERC721AStorage.layout()._packedAddressData[from]; // Updates: `balance -= 1`. ++ERC721AStorage.layout()._packedAddressData[to]; // Updates: `balance += 1`. // Updates: // - `address` to the next owner. // - `startTimestamp` to the timestamp of transfering. // - `burned` to `false`. // - `nextInitialized` to `true`. ERC721AStorage.layout()._packedOwnerships[tokenId] = _packOwnershipData( to, _BITMASK_NEXT_INITIALIZED | _nextExtraData(from, to, prevOwnershipPacked) ); // If the next slot may not have been initialized (i.e. `nextInitialized == false`) . if (prevOwnershipPacked & _BITMASK_NEXT_INITIALIZED == 0) { uint256 nextTokenId = tokenId + 1; // If the next slot's address is zero and not burned (i.e. packed value is zero). if (ERC721AStorage.layout()._packedOwnerships[nextTokenId] == 0) { // If the next slot is within bounds. if (nextTokenId != ERC721AStorage.layout()._currentIndex) { // Initialize the next slot to maintain correctness for `ownerOf(tokenId + 1)`. ERC721AStorage.layout()._packedOwnerships[nextTokenId] = prevOwnershipPacked; } } } } // Mask `to` to the lower 160 bits, in case the upper bits somehow aren't clean. uint256 toMasked = uint256(uint160(to)) & _BITMASK_ADDRESS; assembly { // Emit the `Transfer` event. log4( 0, // Start of data (0, since no data). 0, // End of data (0, since no data). _TRANSFER_EVENT_SIGNATURE, // Signature. from, // `from`. toMasked, // `to`. tokenId // `tokenId`. ) } if (toMasked == 0) _revert(TransferToZeroAddress.selector); _afterTokenTransfers(from, to, tokenId); } /** * @dev Equivalent to `safeTransferFrom(from, to, tokenId, '')`. */ function safeTransferFrom(address from, address to, uint256 tokenId) public payable virtual override { safeTransferFrom(from, to, tokenId, ""); } /** * @dev Safely transfers `tokenId` token from `from` to `to`. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must exist and be owned by `from`. * - If the caller is not `from`, it must be approved to move this token * by either {approve} or {setApprovalForAll}. * - If `to` refers to a smart contract, it must implement * {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function safeTransferFrom( address from, address to, uint256 tokenId, bytes memory _data ) public payable virtual override { transferFrom(from, to, tokenId); if (to.code.length != 0) if (!_checkContractOnERC721Received(from, to, tokenId, _data)) { _revert(TransferToNonERC721ReceiverImplementer.selector); } } /** * @dev Hook that is called after a set of serially-ordered token IDs * have been transferred. This includes minting. * And also called after one token has been burned. * * `startTokenId` - the first token ID to be transferred. * `quantity` - the amount to be transferred. * * Calling conditions: * * - When `from` and `to` are both non-zero, `from`'s `tokenId` has been * transferred to `to`. * - When `from` is zero, `tokenId` has been minted for `to`. * - When `to` is zero, `tokenId` has been burned by `from`. * - `from` and `to` are never both zero. */ function _afterTokenTransfers(address from, address to, uint256 startTokenId) internal virtual {} /** * @dev Private function to invoke {IERC721Receiver-onERC721Received} on a target contract. * * `from` - Previous owner of the given token ID. * `to` - Target address that will receive the token. * `tokenId` - Token ID to be transferred. * `_data` - Optional data to send along with the call. * * Returns whether the call correctly returned the expected magic value. */ function _checkContractOnERC721Received( address from, address to, uint256 tokenId, bytes memory _data ) private returns (bool) { try ERC721A__IERC721ReceiverUpgradeable(to).onERC721Received(_msgSenderERC721A(), from, tokenId, _data) returns (bytes4 retval) { return retval == ERC721A__IERC721ReceiverUpgradeable(to).onERC721Received.selector; } catch (bytes memory reason) { if (reason.length == 0) { _revert(TransferToNonERC721ReceiverImplementer.selector); } assembly { revert(add(32, reason), mload(reason)) } } } // ============================================================= // MINT OPERATIONS // ============================================================= /** * @dev Mints `quantity` tokens and transfers them to `to`. * * Requirements: * * - `to` cannot be the zero address. * - `quantity` must be greater than 0. * * Emits a {Transfer} event for each mint. */ function _mint(address to, uint256 quantity) internal virtual { uint256 startTokenId = ERC721AStorage.layout()._currentIndex; if (quantity == 0) _revert(MintZeroQuantity.selector); // Overflows are incredibly unrealistic. // `balance` and `numberMinted` have a maximum limit of 2**64. // `tokenId` has a maximum limit of 2**256. unchecked { // Updates: // - `address` to the owner. // - `startTimestamp` to the timestamp of minting. // - `burned` to `false`. // - `nextInitialized` to `quantity == 1`. ERC721AStorage.layout()._packedOwnerships[startTokenId] = _packOwnershipData( to, _nextInitializedFlag(quantity) | _nextExtraData(address(0), to, 0) ); // Updates: // - `balance += quantity`. // - `numberMinted += quantity`. // // We can directly add to the `balance` and `numberMinted`. ERC721AStorage.layout()._packedAddressData[to] += quantity * ((1 << _BITPOS_NUMBER_MINTED) | 1); // Mask `to` to the lower 160 bits, in case the upper bits somehow aren't clean. uint256 toMasked = uint256(uint160(to)) & _BITMASK_ADDRESS; if (toMasked == 0) _revert(MintToZeroAddress.selector); uint256 end = startTokenId + quantity; uint256 tokenId = startTokenId; do { assembly { // Emit the `Transfer` event. log4( 0, // Start of data (0, since no data). 0, // End of data (0, since no data). _TRANSFER_EVENT_SIGNATURE, // Signature. 0, // `address(0)`. toMasked, // `to`. tokenId // `tokenId`. ) } // The `!=` check ensures that large values of `quantity` // that overflows uint256 will make the loop run out of gas. } while (++tokenId != end); ERC721AStorage.layout()._currentIndex = end; } } /** * @dev Safely mints `quantity` tokens and transfers them to `to`. * * Requirements: * * - If `to` refers to a smart contract, it must implement * {IERC721Receiver-onERC721Received}, which is called for each safe transfer. * - `quantity` must be greater than 0. * * See {_mint}. * * Emits a {Transfer} event for each mint. */ function _safeMint(address to, uint256 quantity, bytes memory _data) internal virtual { _mint(to, quantity); unchecked { if (to.code.length != 0) { uint256 end = ERC721AStorage.layout()._currentIndex; uint256 index = end - quantity; do { if (!_checkContractOnERC721Received(address(0), to, index++, _data)) { _revert(TransferToNonERC721ReceiverImplementer.selector); } } while (index < end); // Reentrancy protection. if (ERC721AStorage.layout()._currentIndex != end) _revert(bytes4(0)); } } } /** * @dev Equivalent to `_safeMint(to, quantity, '')`. */ function _safeMint(address to, uint256 quantity) internal virtual { _safeMint(to, quantity, ""); } // ============================================================= // APPROVAL OPERATIONS // ============================================================= /** * @dev Equivalent to `_approve(to, tokenId, false)`. */ function _approve(address to, uint256 tokenId) internal virtual { _approve(to, tokenId, false); } /** * @dev Gives permission to `to` to transfer `tokenId` token to another account. * The approval is cleared when the token is transferred. * * Only a single account can be approved at a time, so approving the * zero address clears previous approvals. * * Requirements: * * - `tokenId` must exist. * * Emits an {Approval} event. */ function _approve(address to, uint256 tokenId, bool approvalCheck) internal virtual { address owner = ownerOf(tokenId); if (approvalCheck && _msgSenderERC721A() != owner) if (!isApprovedForAll(owner, _msgSenderERC721A())) { _revert(ApprovalCallerNotOwnerNorApproved.selector); } ERC721AStorage.layout()._tokenApprovals[tokenId].value = to; emit Approval(owner, to, tokenId); } // ============================================================= // BURN OPERATIONS // ============================================================= /** * @dev Equivalent to `_burn(tokenId, false)`. */ function _burn(uint256 tokenId) internal virtual { _burn(tokenId, false); } /** * @dev Destroys `tokenId`. * The approval is cleared when the token is burned. * * Requirements: * * - `tokenId` must exist. * * Emits a {Transfer} event. */ function _burn(uint256 tokenId, bool approvalCheck) internal virtual { uint256 prevOwnershipPacked = _packedOwnershipOf(tokenId); address from = address(uint160(prevOwnershipPacked)); (uint256 approvedAddressSlot, address approvedAddress) = _getApprovedSlotAndAddress(tokenId); if (approvalCheck) { // The nested ifs save around 20+ gas over a compound boolean condition. if (!_isSenderApprovedOrOwner(approvedAddress, from, _msgSenderERC721A())) if (!isApprovedForAll(from, _msgSenderERC721A())) _revert(TransferCallerNotOwnerNorApproved.selector); } // Clear approvals from the previous owner. assembly { if approvedAddress { // This is equivalent to `delete _tokenApprovals[tokenId]`. sstore(approvedAddressSlot, 0) } } // Underflow of the sender's balance is impossible because we check for // ownership above and the recipient's balance can't realistically overflow. // Counter overflow is incredibly unrealistic as `tokenId` would have to be 2**256. unchecked { // Updates: // - `balance -= 1`. // - `numberBurned += 1`. // // We can directly decrement the balance, and increment the number burned. // This is equivalent to `packed -= 1; packed += 1 << _BITPOS_NUMBER_BURNED;`. ERC721AStorage.layout()._packedAddressData[from] += (1 << _BITPOS_NUMBER_BURNED) - 1; // Updates: // - `address` to the last owner. // - `startTimestamp` to the timestamp of burning. // - `burned` to `true`. // - `nextInitialized` to `true`. ERC721AStorage.layout()._packedOwnerships[tokenId] = _packOwnershipData( from, (_BITMASK_BURNED | _BITMASK_NEXT_INITIALIZED) | _nextExtraData(from, address(0), prevOwnershipPacked) ); // If the next slot may not have been initialized (i.e. `nextInitialized == false`) . if (prevOwnershipPacked & _BITMASK_NEXT_INITIALIZED == 0) { uint256 nextTokenId = tokenId + 1; // If the next slot's address is zero and not burned (i.e. packed value is zero). if (ERC721AStorage.layout()._packedOwnerships[nextTokenId] == 0) { // If the next slot is within bounds. if (nextTokenId != ERC721AStorage.layout()._currentIndex) { // Initialize the next slot to maintain correctness for `ownerOf(tokenId + 1)`. ERC721AStorage.layout()._packedOwnerships[nextTokenId] = prevOwnershipPacked; } } } } emit Transfer(from, address(0), tokenId); // Overflow not possible, as _burnCounter cannot be exceed _currentIndex times. unchecked { ERC721AStorage.layout()._burnCounter++; } } // ============================================================= // EXTRA DATA OPERATIONS // ============================================================= /** * @dev Directly sets the extra data for the ownership data `index`. */ function _setExtraDataAt(uint256 index, uint24 extraData) internal virtual { uint256 packed = ERC721AStorage.layout()._packedOwnerships[index]; if (packed == 0) _revert(OwnershipNotInitializedForExtraData.selector); uint256 extraDataCasted; // Cast `extraData` with assembly to avoid redundant masking. assembly { extraDataCasted := extraData } packed = (packed & _BITMASK_EXTRA_DATA_COMPLEMENT) | (extraDataCasted << _BITPOS_EXTRA_DATA); ERC721AStorage.layout()._packedOwnerships[index] = packed; } /** * @dev Called during each token transfer to set the 24bit `extraData` field. * Intended to be overridden by the cosumer contract. * * `previousExtraData` - the value of `extraData` before transfer. * * Calling conditions: * * - When `from` and `to` are both non-zero, `from`'s `tokenId` will be * transferred to `to`. * - When `from` is zero, `tokenId` will be minted for `to`. * - When `to` is zero, `tokenId` will be burned by `from`. * - `from` and `to` are never both zero. */ function _extraData(address from, address to, uint24 previousExtraData) internal view virtual returns (uint24) {} /** * @dev Returns the next extra data for the packed ownership data. * The returned result is shifted into position. */ function _nextExtraData(address from, address to, uint256 prevOwnershipPacked) private view returns (uint256) { uint24 extraData = uint24(prevOwnershipPacked >> _BITPOS_EXTRA_DATA); return uint256(_extraData(from, to, extraData)) << _BITPOS_EXTRA_DATA; } // ============================================================= // OTHER OPERATIONS // ============================================================= /** * @dev Returns the message sender (defaults to `msg.sender`). * * If you are writing GSN compatible contracts, you need to override this function. */ function _msgSenderERC721A() internal view virtual returns (address) { return msg.sender; } /** * @dev Converts a uint256 to its ASCII string decimal representation. */ function _toString(uint256 value) internal pure virtual returns (string memory str) { 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. Total: 5 * 0x20 = 0xa0. let m := add(mload(0x40), 0xa0) // Update the free memory pointer to allocate. mstore(0x40, m) // Assign the `str` to the end. str := sub(m, 0x20) // Zeroize the slot after the string. mstore(str, 0) // Cache the end of the memory to calculate the length later. let end := str // We write the string from rightmost digit to leftmost digit. // The following is essentially a do-while loop that also handles the zero case. // prettier-ignore for { let temp := value } 1 {} { str := sub(str, 1) // Write the character to the pointer. // The ASCII index of the '0' character is 48. mstore8(str, add(48, mod(temp, 10))) // Keep dividing `temp` until zero. temp := div(temp, 10) // prettier-ignore if iszero(temp) { break } } let length := sub(end, str) // Move the pointer 32 bytes leftwards to make room for the length. str := sub(str, 0x20) // Store the length. mstore(str, length) } } /** * @dev For more efficient reverts. */ function _revert(bytes4 errorSelector) internal pure virtual { assembly { mstore(0x00, errorSelector) revert(0x00, 0x04) } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/Context.sol) pragma solidity ^0.8.0; /** * @dev Provides information about the current execution context, including the * sender of the transaction and its data. While these are generally available * via msg.sender and msg.data, they should not be accessed in such a direct * manner, since when dealing with meta-transactions the account sending and * paying for execution may not be the actual sender (as far as an application * is concerned). * * This contract is only required for intermediate, library-like contracts. */ abstract contract Context { function _msgSender() internal view virtual returns (address) { return msg.sender; } function _msgData() internal view virtual returns (bytes calldata) { return msg.data; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (utils/Strings.sol) pragma solidity ^0.8.0; import "./math/MathUpgradeable.sol"; import "./math/SignedMathUpgradeable.sol"; /** * @dev String operations. */ library StringsUpgradeable { bytes16 private constant _SYMBOLS = "0123456789abcdef"; uint8 private constant _ADDRESS_LENGTH = 20; /** * @dev Converts a `uint256` to its ASCII `string` decimal representation. */ function toString(uint256 value) internal pure returns (string memory) { unchecked { uint256 length = MathUpgradeable.log10(value) + 1; string memory buffer = new string(length); uint256 ptr; /// @solidity memory-safe-assembly assembly { ptr := add(buffer, add(32, length)) } while (true) { ptr--; /// @solidity memory-safe-assembly assembly { mstore8(ptr, byte(mod(value, 10), _SYMBOLS)) } value /= 10; if (value == 0) break; } return buffer; } } /** * @dev Converts a `int256` to its ASCII `string` decimal representation. */ function toString(int256 value) internal pure returns (string memory) { return string(abi.encodePacked(value < 0 ? "-" : "", toString(SignedMathUpgradeable.abs(value)))); } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation. */ function toHexString(uint256 value) internal pure returns (string memory) { unchecked { return toHexString(value, MathUpgradeable.log256(value) + 1); } } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length. */ function toHexString(uint256 value, uint256 length) internal pure returns (string memory) { bytes memory buffer = new bytes(2 * length + 2); buffer[0] = "0"; buffer[1] = "x"; for (uint256 i = 2 * length + 1; i > 1; --i) { buffer[i] = _SYMBOLS[value & 0xf]; value >>= 4; } require(value == 0, "Strings: hex length insufficient"); return string(buffer); } /** * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal representation. */ function toHexString(address addr) internal pure returns (string memory) { return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH); } /** * @dev Returns true if the two strings are equal. */ function equal(string memory a, string memory b) internal pure returns (bool) { return keccak256(bytes(a)) == keccak256(bytes(b)); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (proxy/utils/Initializable.sol) pragma solidity ^0.8.2; import "../../utils/AddressUpgradeable.sol"; /** * @dev This is a base contract to aid in writing upgradeable contracts, or any kind of contract that will be deployed * behind a proxy. Since proxied contracts do not make use of a constructor, it's common to move constructor logic to an * external initializer function, usually called `initialize`. It then becomes necessary to protect this initializer * function so it can only be called once. The {initializer} modifier provided by this contract will have this effect. * * The initialization functions use a version number. Once a version number is used, it is consumed and cannot be * reused. This mechanism prevents re-execution of each "step" but allows the creation of new initialization steps in * case an upgrade adds a module that needs to be initialized. * * For example: * * [.hljs-theme-light.nopadding] * ```solidity * contract MyToken is ERC20Upgradeable { * function initialize() initializer public { * __ERC20_init("MyToken", "MTK"); * } * } * * contract MyTokenV2 is MyToken, ERC20PermitUpgradeable { * function initializeV2() reinitializer(2) public { * __ERC20Permit_init("MyToken"); * } * } * ``` * * TIP: To avoid leaving the proxy in an uninitialized state, the initializer function should be called as early as * possible by providing the encoded function call as the `_data` argument to {ERC1967Proxy-constructor}. * * CAUTION: When used with inheritance, manual care must be taken to not invoke a parent initializer twice, or to ensure * that all initializers are idempotent. This is not verified automatically as constructors are by Solidity. * * [CAUTION] * ==== * Avoid leaving a contract uninitialized. * * An uninitialized contract can be taken over by an attacker. This applies to both a proxy and its implementation * contract, which may impact the proxy. To prevent the implementation contract from being used, you should invoke * the {_disableInitializers} function in the constructor to automatically lock it when it is deployed: * * [.hljs-theme-light.nopadding] * ``` * /// @custom:oz-upgrades-unsafe-allow constructor * constructor() { * _disableInitializers(); * } * ``` * ==== */ abstract contract Initializable { /** * @dev Indicates that the contract has been initialized. * @custom:oz-retyped-from bool */ uint8 private _initialized; /** * @dev Indicates that the contract is in the process of being initialized. */ bool private _initializing; /** * @dev Triggered when the contract has been initialized or reinitialized. */ event Initialized(uint8 version); /** * @dev A modifier that defines a protected initializer function that can be invoked at most once. In its scope, * `onlyInitializing` functions can be used to initialize parent contracts. * * Similar to `reinitializer(1)`, except that functions marked with `initializer` can be nested in the context of a * constructor. * * Emits an {Initialized} event. */ modifier initializer() { bool isTopLevelCall = !_initializing; require( (isTopLevelCall && _initialized < 1) || (!AddressUpgradeable.isContract(address(this)) && _initialized == 1), "Initializable: contract is already initialized" ); _initialized = 1; if (isTopLevelCall) { _initializing = true; } _; if (isTopLevelCall) { _initializing = false; emit Initialized(1); } } /** * @dev A modifier that defines a protected reinitializer function that can be invoked at most once, and only if the * contract hasn't been initialized to a greater version before. In its scope, `onlyInitializing` functions can be * used to initialize parent contracts. * * A reinitializer may be used after the original initialization step. This is essential to configure modules that * are added through upgrades and that require initialization. * * When `version` is 1, this modifier is similar to `initializer`, except that functions marked with `reinitializer` * cannot be nested. If one is invoked in the context of another, execution will revert. * * Note that versions can jump in increments greater than 1; this implies that if multiple reinitializers coexist in * a contract, executing them in the right order is up to the developer or operator. * * WARNING: setting the version to 255 will prevent any future reinitialization. * * Emits an {Initialized} event. */ modifier reinitializer(uint8 version) { require(!_initializing && _initialized < version, "Initializable: contract is already initialized"); _initialized = version; _initializing = true; _; _initializing = false; emit Initialized(version); } /** * @dev Modifier to protect an initialization function so that it can only be invoked by functions with the * {initializer} and {reinitializer} modifiers, directly or indirectly. */ modifier onlyInitializing() { require(_initializing, "Initializable: contract is not initializing"); _; } /** * @dev Locks the contract, preventing any future reinitialization. This cannot be part of an initializer call. * Calling this in the constructor of a contract will prevent that contract from being initialized or reinitialized * to any version. It is recommended to use this to lock implementation contracts that are designed to be called * through proxies. * * Emits an {Initialized} event the first time it is successfully executed. */ function _disableInitializers() internal virtual { require(!_initializing, "Initializable: contract is initializing"); if (_initialized != type(uint8).max) { _initialized = type(uint8).max; emit Initialized(type(uint8).max); } } /** * @dev Returns the highest version that has been initialized. See {reinitializer}. */ function _getInitializedVersion() internal view returns (uint8) { return _initialized; } /** * @dev Returns `true` if the contract is currently initializing. See {onlyInitializing}. */ function _isInitializing() internal view returns (bool) { return _initializing; } }
// SPDX-License-Identifier: MIT // ERC721A Contracts v4.2.3 // Creator: Chiru Labs pragma solidity 0.8.10; /* solhint-disable */ /** * @dev Interface of ERC721A. */ interface IERC721AUpgradeable { /** * The caller must own the token or be an approved operator. */ error ApprovalCallerNotOwnerNorApproved(); /** * The token does not exist. */ error ApprovalQueryForNonexistentToken(); /** * Cannot query the balance for the zero address. */ error BalanceQueryForZeroAddress(); /** * Cannot mint to the zero address. */ error MintToZeroAddress(); /** * The quantity of tokens minted must be more than zero. */ error MintZeroQuantity(); /** * The token does not exist. */ error OwnerQueryForNonexistentToken(); /** * The caller must own the token or be an approved operator. */ error TransferCallerNotOwnerNorApproved(); /** * The token must be owned by `from`. */ error TransferFromIncorrectOwner(); /** * Cannot safely transfer to a contract that does not implement the * ERC721Receiver interface. */ error TransferToNonERC721ReceiverImplementer(); /** * Cannot transfer to the zero address. */ error TransferToZeroAddress(); /** * The token does not exist. */ error URIQueryForNonexistentToken(); /** * The `quantity` minted with ERC2309 exceeds the safety limit. */ error MintERC2309QuantityExceedsLimit(); /** * The `extraData` cannot be set on an unintialized ownership slot. */ error OwnershipNotInitializedForExtraData(); // ============================================================= // STRUCTS // ============================================================= struct TokenOwnership { // The address of the owner. address addr; // Stores the start time of ownership with minimal overhead for tokenomics. uint64 startTimestamp; // Whether the token has been burned. bool burned; // Arbitrary data similar to `startTimestamp` that can be set via {_extraData}. uint24 extraData; } // ============================================================= // TOKEN COUNTERS // ============================================================= /** * @dev Returns the total number of tokens in existence. * Burned tokens will reduce the count. * To get the total number of tokens minted, please see {_totalMinted}. */ function totalSupply() external view returns (uint256); // ============================================================= // IERC165 // ============================================================= /** * @dev Returns true if this contract implements the interface defined by * `interfaceId`. See the corresponding * [EIP section](https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified) * to learn more about how these ids are created. * * This function call must use less than 30000 gas. */ function supportsInterface(bytes4 interfaceId) external view returns (bool); // ============================================================= // IERC721 // ============================================================= /** * @dev Emitted when `tokenId` token is transferred from `from` to `to`. */ event Transfer(address indexed from, address indexed to, uint256 indexed tokenId); /** * @dev Emitted when `owner` enables `approved` to manage the `tokenId` token. */ event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId); /** * @dev Emitted when `owner` enables or disables * (`approved`) `operator` to manage all of its assets. */ event ApprovalForAll(address indexed owner, address indexed operator, bool approved); /** * @dev Returns the number of tokens in `owner`'s account. */ function balanceOf(address owner) external view returns (uint256 balance); /** * @dev Returns the owner of the `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function ownerOf(uint256 tokenId) external view returns (address owner); /** * @dev Safely transfers `tokenId` token from `from` to `to`, * checking first that contract recipients are aware of the ERC721 protocol * to prevent tokens from being forever locked. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must exist and be owned by `from`. * - If the caller is not `from`, it must be have been allowed to move * this token by either {approve} or {setApprovalForAll}. * - If `to` refers to a smart contract, it must implement * {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function safeTransferFrom(address from, address to, uint256 tokenId, bytes calldata data) external payable; /** * @dev Equivalent to `safeTransferFrom(from, to, tokenId, '')`. */ function safeTransferFrom(address from, address to, uint256 tokenId) external payable; /** * @dev Transfers `tokenId` from `from` to `to`. * * WARNING: Usage of this method is discouraged, use {safeTransferFrom} * whenever possible. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must be owned by `from`. * - If the caller is not `from`, it must be approved to move this token * by either {approve} or {setApprovalForAll}. * * Emits a {Transfer} event. */ function transferFrom(address from, address to, uint256 tokenId) external payable; /** * @dev Gives permission to `to` to transfer `tokenId` token to another account. * The approval is cleared when the token is transferred. * * Only a single account can be approved at a time, so approving the * zero address clears previous approvals. * * Requirements: * * - The caller must own the token or be an approved operator. * - `tokenId` must exist. * * Emits an {Approval} event. */ function approve(address to, uint256 tokenId) external payable; /** * @dev Approve or remove `operator` as an operator for the caller. * Operators can call {transferFrom} or {safeTransferFrom} * for any token owned by the caller. * * Requirements: * * - The `operator` cannot be the caller. * * Emits an {ApprovalForAll} event. */ function setApprovalForAll(address operator, bool _approved) external; /** * @dev Returns the account approved for `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function getApproved(uint256 tokenId) external view returns (address operator); /** * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`. * * See {setApprovalForAll}. */ function isApprovedForAll(address owner, address operator) external view returns (bool); // ============================================================= // IERC721Metadata // ============================================================= /** * @dev Returns the token collection name. */ function name() external view returns (string memory); /** * @dev Returns the token collection symbol. */ function symbol() external view returns (string memory); /** * @dev Returns the Uniform Resource Identifier (URI) for `tokenId` token. */ function tokenURI(uint256 tokenId) external view returns (string memory); }
// SPDX-License-Identifier: MIT pragma solidity 0.8.10; /* solhint-disable */ library ERC721AStorage { // Bypass for a `--via-ir` bug (https://github.com/chiru-labs/ERC721A/pull/364). struct TokenApprovalRef { address value; } struct Layout { // ============================================================= // STORAGE // ============================================================= // The next token ID to be minted. uint256 _currentIndex; // The number of tokens burned. uint256 _burnCounter; // Token name string _name; // Token symbol string _symbol; // Mapping from token ID to ownership details // An empty struct value does not necessarily mean the token is unowned. // See {_packedOwnershipOf} implementation for details. // // Bits Layout: // - [0..159] `addr` // - [160..223] `startTimestamp` // - [224] `burned` // - [225] `nextInitialized` // - [232..255] `extraData` mapping(uint256 => uint256) _packedOwnerships; // Mapping owner address to address data. // // Bits Layout: // - [0..63] `balance` // - [64..127] `numberMinted` // - [128..191] `numberBurned` // - [192..255] `aux` mapping(address => uint256) _packedAddressData; // Mapping from token ID to approved address. mapping(uint256 => ERC721AStorage.TokenApprovalRef) _tokenApprovals; // Mapping from owner to operator approvals mapping(address => mapping(address => bool)) _operatorApprovals; } bytes32 internal constant STORAGE_SLOT = keccak256("ERC721A.contracts.storage.ERC721A"); function layout() internal pure returns (Layout storage l) { bytes32 slot = STORAGE_SLOT; assembly { l.slot := slot } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.4) (utils/Context.sol) pragma solidity ^0.8.0; import {Initializable} from "../proxy/utils/Initializable.sol"; /** * @dev Provides information about the current execution context, including the * sender of the transaction and its data. While these are generally available * via msg.sender and msg.data, they should not be accessed in such a direct * manner, since when dealing with meta-transactions the account sending and * paying for execution may not be the actual sender (as far as an application * is concerned). * * This contract is only required for intermediate, library-like contracts. */ abstract contract ContextUpgradeable is Initializable { function __Context_init() internal onlyInitializing { } function __Context_init_unchained() internal onlyInitializing { } function _msgSender() internal view virtual returns (address) { return msg.sender; } function _msgData() internal view virtual returns (bytes calldata) { return msg.data; } function _contextSuffixLength() internal view virtual returns (uint256) { return 0; } /** * @dev This empty reserved space is put in place to allow future versions to add new * variables without shifting down storage in the inheritance chain. * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps */ uint256[50] private __gap; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.5.0) (interfaces/draft-IERC1822.sol) pragma solidity ^0.8.0; /** * @dev ERC1822: Universal Upgradeable Proxy Standard (UUPS) documents a method for upgradeability through a simplified * proxy whose upgrades are fully controlled by the current implementation. */ interface IERC1822Proxiable { /** * @dev Returns the storage slot that the proxiable contract assumes is being used to store the implementation * address. * * IMPORTANT: A proxy pointing at a proxiable contract should not be considered proxiable itself, because this risks * bricking a proxy that upgrades to it, by delegating to itself until out of gas. Thus it is critical that this * function revert if invoked through a proxy. */ function proxiableUUID() external view returns (bytes32); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (proxy/ERC1967/ERC1967Upgrade.sol) pragma solidity ^0.8.2; import "../beacon/IBeacon.sol"; import "../../interfaces/IERC1967.sol"; import "../../interfaces/draft-IERC1822.sol"; import "../../utils/Address.sol"; import "../../utils/StorageSlot.sol"; /** * @dev This abstract contract provides getters and event emitting update functions for * https://eips.ethereum.org/EIPS/eip-1967[EIP1967] slots. * * _Available since v4.1._ */ abstract contract ERC1967Upgrade is IERC1967 { // This is the keccak-256 hash of "eip1967.proxy.rollback" subtracted by 1 bytes32 private constant _ROLLBACK_SLOT = 0x4910fdfa16fed3260ed0e7147f7cc6da11a60208b5b9406d12a635614ffd9143; /** * @dev Storage slot with the address of the current implementation. * This is the keccak-256 hash of "eip1967.proxy.implementation" subtracted by 1, and is * validated in the constructor. */ bytes32 internal constant _IMPLEMENTATION_SLOT = 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc; /** * @dev Returns the current implementation address. */ function _getImplementation() internal view returns (address) { return StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value; } /** * @dev Stores a new address in the EIP1967 implementation slot. */ function _setImplementation(address newImplementation) private { require(Address.isContract(newImplementation), "ERC1967: new implementation is not a contract"); StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value = newImplementation; } /** * @dev Perform implementation upgrade * * Emits an {Upgraded} event. */ function _upgradeTo(address newImplementation) internal { _setImplementation(newImplementation); emit Upgraded(newImplementation); } /** * @dev Perform implementation upgrade with additional setup call. * * Emits an {Upgraded} event. */ function _upgradeToAndCall(address newImplementation, bytes memory data, bool forceCall) internal { _upgradeTo(newImplementation); if (data.length > 0 || forceCall) { Address.functionDelegateCall(newImplementation, data); } } /** * @dev Perform implementation upgrade with security checks for UUPS proxies, and additional setup call. * * Emits an {Upgraded} event. */ function _upgradeToAndCallUUPS(address newImplementation, bytes memory data, bool forceCall) internal { // Upgrades from old implementations will perform a rollback test. This test requires the new // implementation to upgrade back to the old, non-ERC1822 compliant, implementation. Removing // this special case will break upgrade paths from old UUPS implementation to new ones. if (StorageSlot.getBooleanSlot(_ROLLBACK_SLOT).value) { _setImplementation(newImplementation); } else { try IERC1822Proxiable(newImplementation).proxiableUUID() returns (bytes32 slot) { require(slot == _IMPLEMENTATION_SLOT, "ERC1967Upgrade: unsupported proxiableUUID"); } catch { revert("ERC1967Upgrade: new implementation is not UUPS"); } _upgradeToAndCall(newImplementation, data, forceCall); } } /** * @dev Storage slot with the admin of the contract. * This is the keccak-256 hash of "eip1967.proxy.admin" subtracted by 1, and is * validated in the constructor. */ bytes32 internal constant _ADMIN_SLOT = 0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103; /** * @dev Returns the current admin. */ function _getAdmin() internal view returns (address) { return StorageSlot.getAddressSlot(_ADMIN_SLOT).value; } /** * @dev Stores a new address in the EIP1967 admin slot. */ function _setAdmin(address newAdmin) private { require(newAdmin != address(0), "ERC1967: new admin is the zero address"); StorageSlot.getAddressSlot(_ADMIN_SLOT).value = newAdmin; } /** * @dev Changes the admin of the proxy. * * Emits an {AdminChanged} event. */ function _changeAdmin(address newAdmin) internal { emit AdminChanged(_getAdmin(), newAdmin); _setAdmin(newAdmin); } /** * @dev The storage slot of the UpgradeableBeacon contract which defines the implementation for this proxy. * This is bytes32(uint256(keccak256('eip1967.proxy.beacon')) - 1)) and is validated in the constructor. */ bytes32 internal constant _BEACON_SLOT = 0xa3f0ad74e5423aebfd80d3ef4346578335a9a72aeaee59ff6cb3582b35133d50; /** * @dev Returns the current beacon. */ function _getBeacon() internal view returns (address) { return StorageSlot.getAddressSlot(_BEACON_SLOT).value; } /** * @dev Stores a new beacon in the EIP1967 beacon slot. */ function _setBeacon(address newBeacon) private { require(Address.isContract(newBeacon), "ERC1967: new beacon is not a contract"); require( Address.isContract(IBeacon(newBeacon).implementation()), "ERC1967: beacon implementation is not a contract" ); StorageSlot.getAddressSlot(_BEACON_SLOT).value = newBeacon; } /** * @dev Perform beacon upgrade with additional setup call. Note: This upgrades the address of the beacon, it does * not upgrade the implementation contained in the beacon (see {UpgradeableBeacon-_setImplementation} for that). * * Emits a {BeaconUpgraded} event. */ function _upgradeBeaconToAndCall(address newBeacon, bytes memory data, bool forceCall) internal { _setBeacon(newBeacon); emit BeaconUpgraded(newBeacon); if (data.length > 0 || forceCall) { Address.functionDelegateCall(IBeacon(newBeacon).implementation(), data); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (utils/Address.sol) pragma solidity ^0.8.1; /** * @dev Collection of functions related to the address type */ library AddressUpgradeable { /** * @dev Returns true if `account` is a contract. * * [IMPORTANT] * ==== * It is unsafe to assume that an address for which this function returns * false is an externally-owned account (EOA) and not a contract. * * Among others, `isContract` will return false for the following * types of addresses: * * - an externally-owned account * - a contract in construction * - an address where a contract will be created * - an address where a contract lived, but was destroyed * * Furthermore, `isContract` will also return true if the target contract within * the same transaction is already scheduled for destruction by `SELFDESTRUCT`, * which only has an effect at the end of a transaction. * ==== * * [IMPORTANT] * ==== * You shouldn't rely on `isContract` to protect against flash loan attacks! * * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract * constructor. * ==== */ function isContract(address account) internal view returns (bool) { // This method relies on extcodesize/address.code.length, which returns 0 // for contracts in construction, since the code is only stored at the end // of the constructor execution. return account.code.length > 0; } /** * @dev Replacement for Solidity's `transfer`: sends `amount` wei to * `recipient`, forwarding all available gas and reverting on errors. * * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost * of certain opcodes, possibly making contracts go over the 2300 gas limit * imposed by `transfer`, making them unable to receive funds via * `transfer`. {sendValue} removes this limitation. * * https://consensys.net/diligence/blog/2019/09/stop-using-soliditys-transfer-now/[Learn more]. * * IMPORTANT: because control is transferred to `recipient`, care must be * taken to not create reentrancy vulnerabilities. Consider using * {ReentrancyGuard} or the * https://solidity.readthedocs.io/en/v0.8.0/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern]. */ function sendValue(address payable recipient, uint256 amount) internal { require(address(this).balance >= amount, "Address: insufficient balance"); (bool success, ) = recipient.call{value: amount}(""); require(success, "Address: unable to send value, recipient may have reverted"); } /** * @dev Performs a Solidity function call using a low level `call`. A * plain `call` is an unsafe replacement for a function call: use this * function instead. * * If `target` reverts with a revert reason, it is bubbled up by this * function (like regular Solidity function calls). * * Returns the raw returned data. To convert to the expected return value, * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`]. * * Requirements: * * - `target` must be a contract. * - calling `target` with `data` must not revert. * * _Available since v3.1._ */ function functionCall(address target, bytes memory data) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, "Address: low-level call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with * `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but also transferring `value` wei to `target`. * * Requirements: * * - the calling contract must have an ETH balance of at least `value`. * - the called Solidity function must be `payable`. * * _Available since v3.1._ */ function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) { return functionCallWithValue(target, data, value, "Address: low-level call with value failed"); } /** * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but * with `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCallWithValue( address target, bytes memory data, uint256 value, string memory errorMessage ) internal returns (bytes memory) { require(address(this).balance >= value, "Address: insufficient balance for call"); (bool success, bytes memory returndata) = target.call{value: value}(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) { return functionStaticCall(target, data, "Address: low-level static call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall( address target, bytes memory data, string memory errorMessage ) internal view returns (bytes memory) { (bool success, bytes memory returndata) = target.staticcall(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) { return functionDelegateCall(target, data, "Address: low-level delegate call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { (bool success, bytes memory returndata) = target.delegatecall(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract. * * _Available since v4.8._ */ function verifyCallResultFromTarget( address target, bool success, bytes memory returndata, string memory errorMessage ) internal view returns (bytes memory) { if (success) { if (returndata.length == 0) { // only check isContract if the call was successful and the return data is empty // otherwise we already know that it was a contract require(isContract(target), "Address: call to non-contract"); } return returndata; } else { _revert(returndata, errorMessage); } } /** * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the * revert reason or using the provided one. * * _Available since v4.3._ */ function verifyCallResult( bool success, bytes memory returndata, string memory errorMessage ) internal pure returns (bytes memory) { if (success) { return returndata; } else { _revert(returndata, errorMessage); } } function _revert(bytes memory returndata, string memory errorMessage) private pure { // Look for revert reason and bubble it up if present if (returndata.length > 0) { // The easiest way to bubble the revert reason is using memory via assembly /// @solidity memory-safe-assembly assembly { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert(errorMessage); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (utils/math/Math.sol) pragma solidity ^0.8.0; /** * @dev Standard math utilities missing in the Solidity language. */ library MathUpgradeable { enum Rounding { Down, // Toward negative infinity Up, // Toward infinity Zero // Toward zero } /** * @dev Returns the largest of two numbers. */ function max(uint256 a, uint256 b) internal pure returns (uint256) { return a > b ? a : b; } /** * @dev Returns the smallest of two numbers. */ function min(uint256 a, uint256 b) internal pure returns (uint256) { return a < b ? a : b; } /** * @dev Returns the average of two numbers. The result is rounded towards * zero. */ function average(uint256 a, uint256 b) internal pure returns (uint256) { // (a + b) / 2 can overflow. return (a & b) + (a ^ b) / 2; } /** * @dev Returns the ceiling of the division of two numbers. * * This differs from standard division with `/` in that it rounds up instead * of rounding down. */ function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) { // (a + b - 1) / b can overflow on addition, so we distribute. return a == 0 ? 0 : (a - 1) / b + 1; } /** * @notice Calculates floor(x * y / denominator) with full precision. Throws if result overflows a uint256 or denominator == 0 * @dev Original credit to Remco Bloemen under MIT license (https://xn--2-umb.com/21/muldiv) * with further edits by Uniswap Labs also under MIT license. */ function mulDiv(uint256 x, uint256 y, uint256 denominator) internal pure returns (uint256 result) { unchecked { // 512-bit multiply [prod1 prod0] = x * y. Compute the product mod 2^256 and mod 2^256 - 1, then use // use the Chinese Remainder Theorem to reconstruct the 512 bit result. The result is stored in two 256 // variables such that product = prod1 * 2^256 + prod0. uint256 prod0; // Least significant 256 bits of the product uint256 prod1; // Most significant 256 bits of the product assembly { let mm := mulmod(x, y, not(0)) prod0 := mul(x, y) prod1 := sub(sub(mm, prod0), lt(mm, prod0)) } // Handle non-overflow cases, 256 by 256 division. if (prod1 == 0) { // Solidity will revert if denominator == 0, unlike the div opcode on its own. // The surrounding unchecked block does not change this fact. // See https://docs.soliditylang.org/en/latest/control-structures.html#checked-or-unchecked-arithmetic. return prod0 / denominator; } // Make sure the result is less than 2^256. Also prevents denominator == 0. require(denominator > prod1, "Math: mulDiv overflow"); /////////////////////////////////////////////// // 512 by 256 division. /////////////////////////////////////////////// // Make division exact by subtracting the remainder from [prod1 prod0]. uint256 remainder; assembly { // Compute remainder using mulmod. remainder := mulmod(x, y, denominator) // Subtract 256 bit number from 512 bit number. prod1 := sub(prod1, gt(remainder, prod0)) prod0 := sub(prod0, remainder) } // Factor powers of two out of denominator and compute largest power of two divisor of denominator. Always >= 1. // See https://cs.stackexchange.com/q/138556/92363. // Does not overflow because the denominator cannot be zero at this stage in the function. uint256 twos = denominator & (~denominator + 1); assembly { // Divide denominator by twos. denominator := div(denominator, twos) // Divide [prod1 prod0] by twos. prod0 := div(prod0, twos) // Flip twos such that it is 2^256 / twos. If twos is zero, then it becomes one. twos := add(div(sub(0, twos), twos), 1) } // Shift in bits from prod1 into prod0. prod0 |= prod1 * twos; // Invert denominator mod 2^256. Now that denominator is an odd number, it has an inverse modulo 2^256 such // that denominator * inv = 1 mod 2^256. Compute the inverse by starting with a seed that is correct for // four bits. That is, denominator * inv = 1 mod 2^4. uint256 inverse = (3 * denominator) ^ 2; // Use the Newton-Raphson iteration to improve the precision. Thanks to Hensel's lifting lemma, this also works // in modular arithmetic, doubling the correct bits in each step. inverse *= 2 - denominator * inverse; // inverse mod 2^8 inverse *= 2 - denominator * inverse; // inverse mod 2^16 inverse *= 2 - denominator * inverse; // inverse mod 2^32 inverse *= 2 - denominator * inverse; // inverse mod 2^64 inverse *= 2 - denominator * inverse; // inverse mod 2^128 inverse *= 2 - denominator * inverse; // inverse mod 2^256 // Because the division is now exact we can divide by multiplying with the modular inverse of denominator. // This will give us the correct result modulo 2^256. Since the preconditions guarantee that the outcome is // less than 2^256, this is the final result. We don't need to compute the high bits of the result and prod1 // is no longer required. result = prod0 * inverse; return result; } } /** * @notice Calculates x * y / denominator with full precision, following the selected rounding direction. */ function mulDiv(uint256 x, uint256 y, uint256 denominator, Rounding rounding) internal pure returns (uint256) { uint256 result = mulDiv(x, y, denominator); if (rounding == Rounding.Up && mulmod(x, y, denominator) > 0) { result += 1; } return result; } /** * @dev Returns the square root of a number. If the number is not a perfect square, the value is rounded down. * * Inspired by Henry S. Warren, Jr.'s "Hacker's Delight" (Chapter 11). */ function sqrt(uint256 a) internal pure returns (uint256) { if (a == 0) { return 0; } // For our first guess, we get the biggest power of 2 which is smaller than the square root of the target. // // We know that the "msb" (most significant bit) of our target number `a` is a power of 2 such that we have // `msb(a) <= a < 2*msb(a)`. This value can be written `msb(a)=2**k` with `k=log2(a)`. // // This can be rewritten `2**log2(a) <= a < 2**(log2(a) + 1)` // → `sqrt(2**k) <= sqrt(a) < sqrt(2**(k+1))` // → `2**(k/2) <= sqrt(a) < 2**((k+1)/2) <= 2**(k/2 + 1)` // // Consequently, `2**(log2(a) / 2)` is a good first approximation of `sqrt(a)` with at least 1 correct bit. uint256 result = 1 << (log2(a) >> 1); // At this point `result` is an estimation with one bit of precision. We know the true value is a uint128, // since it is the square root of a uint256. Newton's method converges quadratically (precision doubles at // every iteration). We thus need at most 7 iteration to turn our partial result with one bit of precision // into the expected uint128 result. unchecked { result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; return min(result, a / result); } } /** * @notice Calculates sqrt(a), following the selected rounding direction. */ function sqrt(uint256 a, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = sqrt(a); return result + (rounding == Rounding.Up && result * result < a ? 1 : 0); } } /** * @dev Return the log in base 2, rounded down, of a positive value. * Returns 0 if given 0. */ function log2(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >> 128 > 0) { value >>= 128; result += 128; } if (value >> 64 > 0) { value >>= 64; result += 64; } if (value >> 32 > 0) { value >>= 32; result += 32; } if (value >> 16 > 0) { value >>= 16; result += 16; } if (value >> 8 > 0) { value >>= 8; result += 8; } if (value >> 4 > 0) { value >>= 4; result += 4; } if (value >> 2 > 0) { value >>= 2; result += 2; } if (value >> 1 > 0) { result += 1; } } return result; } /** * @dev Return the log in base 2, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log2(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log2(value); return result + (rounding == Rounding.Up && 1 << result < value ? 1 : 0); } } /** * @dev Return the log in base 10, rounded down, of a positive value. * Returns 0 if given 0. */ function log10(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >= 10 ** 64) { value /= 10 ** 64; result += 64; } if (value >= 10 ** 32) { value /= 10 ** 32; result += 32; } if (value >= 10 ** 16) { value /= 10 ** 16; result += 16; } if (value >= 10 ** 8) { value /= 10 ** 8; result += 8; } if (value >= 10 ** 4) { value /= 10 ** 4; result += 4; } if (value >= 10 ** 2) { value /= 10 ** 2; result += 2; } if (value >= 10 ** 1) { result += 1; } } return result; } /** * @dev Return the log in base 10, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log10(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log10(value); return result + (rounding == Rounding.Up && 10 ** result < value ? 1 : 0); } } /** * @dev Return the log in base 256, rounded down, of a positive value. * Returns 0 if given 0. * * Adding one to the result gives the number of pairs of hex symbols needed to represent `value` as a hex string. */ function log256(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >> 128 > 0) { value >>= 128; result += 16; } if (value >> 64 > 0) { value >>= 64; result += 8; } if (value >> 32 > 0) { value >>= 32; result += 4; } if (value >> 16 > 0) { value >>= 16; result += 2; } if (value >> 8 > 0) { result += 1; } } return result; } /** * @dev Return the log in base 256, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log256(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log256(value); return result + (rounding == Rounding.Up && 1 << (result << 3) < value ? 1 : 0); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.8.0) (utils/math/SignedMath.sol) pragma solidity ^0.8.0; /** * @dev Standard signed math utilities missing in the Solidity language. */ library SignedMathUpgradeable { /** * @dev Returns the largest of two signed numbers. */ function max(int256 a, int256 b) internal pure returns (int256) { return a > b ? a : b; } /** * @dev Returns the smallest of two signed numbers. */ function min(int256 a, int256 b) internal pure returns (int256) { return a < b ? a : b; } /** * @dev Returns the average of two signed numbers without overflow. * The result is rounded towards zero. */ function average(int256 a, int256 b) internal pure returns (int256) { // Formula from the book "Hacker's Delight" int256 x = (a & b) + ((a ^ b) >> 1); return x + (int256(uint256(x) >> 255) & (a ^ b)); } /** * @dev Returns the absolute unsigned value of a signed value. */ function abs(int256 n) internal pure returns (uint256) { unchecked { // must be unchecked in order to support `n = type(int256).min` return uint256(n >= 0 ? n : -n); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (interfaces/IERC1967.sol) pragma solidity ^0.8.0; /** * @dev ERC-1967: Proxy Storage Slots. This interface contains the events defined in the ERC. * * _Available since v4.8.3._ */ interface IERC1967 { /** * @dev Emitted when the implementation is upgraded. */ event Upgraded(address indexed implementation); /** * @dev Emitted when the admin account has changed. */ event AdminChanged(address previousAdmin, address newAdmin); /** * @dev Emitted when the beacon is changed. */ event BeaconUpgraded(address indexed beacon); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (proxy/beacon/IBeacon.sol) pragma solidity ^0.8.0; /** * @dev This is the interface that {BeaconProxy} expects of its beacon. */ interface IBeacon { /** * @dev Must return an address that can be used as a delegate call target. * * {BeaconProxy} will check that this address is a contract. */ function implementation() external view returns (address); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (utils/Address.sol) pragma solidity ^0.8.1; /** * @dev Collection of functions related to the address type */ library Address { /** * @dev Returns true if `account` is a contract. * * [IMPORTANT] * ==== * It is unsafe to assume that an address for which this function returns * false is an externally-owned account (EOA) and not a contract. * * Among others, `isContract` will return false for the following * types of addresses: * * - an externally-owned account * - a contract in construction * - an address where a contract will be created * - an address where a contract lived, but was destroyed * * Furthermore, `isContract` will also return true if the target contract within * the same transaction is already scheduled for destruction by `SELFDESTRUCT`, * which only has an effect at the end of a transaction. * ==== * * [IMPORTANT] * ==== * You shouldn't rely on `isContract` to protect against flash loan attacks! * * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract * constructor. * ==== */ function isContract(address account) internal view returns (bool) { // This method relies on extcodesize/address.code.length, which returns 0 // for contracts in construction, since the code is only stored at the end // of the constructor execution. return account.code.length > 0; } /** * @dev Replacement for Solidity's `transfer`: sends `amount` wei to * `recipient`, forwarding all available gas and reverting on errors. * * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost * of certain opcodes, possibly making contracts go over the 2300 gas limit * imposed by `transfer`, making them unable to receive funds via * `transfer`. {sendValue} removes this limitation. * * https://consensys.net/diligence/blog/2019/09/stop-using-soliditys-transfer-now/[Learn more]. * * IMPORTANT: because control is transferred to `recipient`, care must be * taken to not create reentrancy vulnerabilities. Consider using * {ReentrancyGuard} or the * https://solidity.readthedocs.io/en/v0.8.0/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern]. */ function sendValue(address payable recipient, uint256 amount) internal { require(address(this).balance >= amount, "Address: insufficient balance"); (bool success, ) = recipient.call{value: amount}(""); require(success, "Address: unable to send value, recipient may have reverted"); } /** * @dev Performs a Solidity function call using a low level `call`. A * plain `call` is an unsafe replacement for a function call: use this * function instead. * * If `target` reverts with a revert reason, it is bubbled up by this * function (like regular Solidity function calls). * * Returns the raw returned data. To convert to the expected return value, * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`]. * * Requirements: * * - `target` must be a contract. * - calling `target` with `data` must not revert. * * _Available since v3.1._ */ function functionCall(address target, bytes memory data) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, "Address: low-level call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with * `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but also transferring `value` wei to `target`. * * Requirements: * * - the calling contract must have an ETH balance of at least `value`. * - the called Solidity function must be `payable`. * * _Available since v3.1._ */ function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) { return functionCallWithValue(target, data, value, "Address: low-level call with value failed"); } /** * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but * with `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCallWithValue( address target, bytes memory data, uint256 value, string memory errorMessage ) internal returns (bytes memory) { require(address(this).balance >= value, "Address: insufficient balance for call"); (bool success, bytes memory returndata) = target.call{value: value}(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) { return functionStaticCall(target, data, "Address: low-level static call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall( address target, bytes memory data, string memory errorMessage ) internal view returns (bytes memory) { (bool success, bytes memory returndata) = target.staticcall(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) { return functionDelegateCall(target, data, "Address: low-level delegate call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { (bool success, bytes memory returndata) = target.delegatecall(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract. * * _Available since v4.8._ */ function verifyCallResultFromTarget( address target, bool success, bytes memory returndata, string memory errorMessage ) internal view returns (bytes memory) { if (success) { if (returndata.length == 0) { // only check isContract if the call was successful and the return data is empty // otherwise we already know that it was a contract require(isContract(target), "Address: call to non-contract"); } return returndata; } else { _revert(returndata, errorMessage); } } /** * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the * revert reason or using the provided one. * * _Available since v4.3._ */ function verifyCallResult( bool success, bytes memory returndata, string memory errorMessage ) internal pure returns (bytes memory) { if (success) { return returndata; } else { _revert(returndata, errorMessage); } } function _revert(bytes memory returndata, string memory errorMessage) private pure { // Look for revert reason and bubble it up if present if (returndata.length > 0) { // The easiest way to bubble the revert reason is using memory via assembly /// @solidity memory-safe-assembly assembly { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert(errorMessage); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (utils/StorageSlot.sol) // This file was procedurally generated from scripts/generate/templates/StorageSlot.js. pragma solidity ^0.8.0; /** * @dev Library for reading and writing primitive types to specific storage slots. * * Storage slots are often used to avoid storage conflict when dealing with upgradeable contracts. * This library helps with reading and writing to such slots without the need for inline assembly. * * The functions in this library return Slot structs that contain a `value` member that can be used to read or write. * * Example usage to set ERC1967 implementation slot: * ```solidity * contract ERC1967 { * bytes32 internal constant _IMPLEMENTATION_SLOT = 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc; * * function _getImplementation() internal view returns (address) { * return StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value; * } * * function _setImplementation(address newImplementation) internal { * require(Address.isContract(newImplementation), "ERC1967: new implementation is not a contract"); * StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value = newImplementation; * } * } * ``` * * _Available since v4.1 for `address`, `bool`, `bytes32`, `uint256`._ * _Available since v4.9 for `string`, `bytes`._ */ library StorageSlot { struct AddressSlot { address value; } struct BooleanSlot { bool value; } struct Bytes32Slot { bytes32 value; } struct Uint256Slot { uint256 value; } struct StringSlot { string value; } struct BytesSlot { bytes value; } /** * @dev Returns an `AddressSlot` with member `value` located at `slot`. */ function getAddressSlot(bytes32 slot) internal pure returns (AddressSlot storage r) { /// @solidity memory-safe-assembly assembly { r.slot := slot } } /** * @dev Returns an `BooleanSlot` with member `value` located at `slot`. */ function getBooleanSlot(bytes32 slot) internal pure returns (BooleanSlot storage r) { /// @solidity memory-safe-assembly assembly { r.slot := slot } } /** * @dev Returns an `Bytes32Slot` with member `value` located at `slot`. */ function getBytes32Slot(bytes32 slot) internal pure returns (Bytes32Slot storage r) { /// @solidity memory-safe-assembly assembly { r.slot := slot } } /** * @dev Returns an `Uint256Slot` with member `value` located at `slot`. */ function getUint256Slot(bytes32 slot) internal pure returns (Uint256Slot storage r) { /// @solidity memory-safe-assembly assembly { r.slot := slot } } /** * @dev Returns an `StringSlot` with member `value` located at `slot`. */ function getStringSlot(bytes32 slot) internal pure returns (StringSlot storage r) { /// @solidity memory-safe-assembly assembly { r.slot := slot } } /** * @dev Returns an `StringSlot` representation of the string storage pointer `store`. */ function getStringSlot(string storage store) internal pure returns (StringSlot storage r) { /// @solidity memory-safe-assembly assembly { r.slot := store.slot } } /** * @dev Returns an `BytesSlot` with member `value` located at `slot`. */ function getBytesSlot(bytes32 slot) internal pure returns (BytesSlot storage r) { /// @solidity memory-safe-assembly assembly { r.slot := slot } } /** * @dev Returns an `BytesSlot` representation of the bytes storage pointer `store`. */ function getBytesSlot(bytes storage store) internal pure returns (BytesSlot storage r) { /// @solidity memory-safe-assembly assembly { r.slot := store.slot } } }
{ "optimizer": { "enabled": true, "mode": "z" }, "outputSelection": { "*": { "*": [ "abi" ] } }, "detectMissingLibraries": false, "forceEVMLA": false, "enableEraVMExtensions": false, "libraries": {} }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"inputs":[],"name":"ApprovalCallerNotOwnerNorApproved","type":"error"},{"inputs":[],"name":"ApprovalQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"BalanceQueryForZeroAddress","type":"error"},{"inputs":[],"name":"EmptyString","type":"error"},{"inputs":[],"name":"InvalidManager","type":"error"},{"inputs":[],"name":"ManagerDoesNotExist","type":"error"},{"inputs":[],"name":"ManagerRemoveBlocked","type":"error"},{"inputs":[],"name":"ManagerSwapBlocked","type":"error"},{"inputs":[],"name":"MintERC2309QuantityExceedsLimit","type":"error"},{"inputs":[],"name":"MintFrozen","type":"error"},{"inputs":[],"name":"MintToZeroAddress","type":"error"},{"inputs":[],"name":"MintZeroQuantity","type":"error"},{"inputs":[],"name":"MinterRegistrationInvalid","type":"error"},{"inputs":[],"name":"MismatchedArrayLengths","type":"error"},{"inputs":[],"name":"NotMinter","type":"error"},{"inputs":[],"name":"OverLimitSupply","type":"error"},{"inputs":[],"name":"OwnerQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"OwnershipNotInitializedForExtraData","type":"error"},{"inputs":[],"name":"RoyaltyBPSInvalid","type":"error"},{"inputs":[],"name":"RoyaltySetBlocked","type":"error"},{"inputs":[],"name":"TokenDoesNotExist","type":"error"},{"inputs":[],"name":"TokenNotInRange","type":"error"},{"inputs":[],"name":"TransferCallerNotOwnerNorApproved","type":"error"},{"inputs":[],"name":"TransferFromIncorrectOwner","type":"error"},{"inputs":[],"name":"TransferToNonERC721ReceiverImplementer","type":"error"},{"inputs":[],"name":"TransferToZeroAddress","type":"error"},{"inputs":[],"name":"URIQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"Unauthorized","type":"error"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"previousAdmin","type":"address"},{"indexed":false,"internalType":"address","name":"newAdmin","type":"address"}],"name":"AdminChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"approved","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"operator","type":"address"},{"indexed":false,"internalType":"bool","name":"approved","type":"bool"}],"name":"ApprovalForAll","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"string","name":"oldBaseUri","type":"string"},{"indexed":false,"internalType":"string","name":"newBaseURI","type":"string"}],"name":"BaseURISet","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"beacon","type":"address"}],"name":"BeaconUpgraded","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"recipientAddress","type":"address"},{"indexed":true,"internalType":"uint16","name":"royaltyPercentageBPS","type":"uint16"}],"name":"DefaultRoyaltySet","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"newDefaultTokenManager","type":"address"}],"name":"DefaultTokenManagerChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256[]","name":"ids","type":"uint256[]"},{"components":[{"internalType":"address","name":"recipientAddress","type":"address"},{"internalType":"uint16","name":"royaltyPercentageBPS","type":"uint16"}],"indexed":false,"internalType":"struct IRoyaltyManager.Royalty[]","name":"_newRoyalties","type":"tuple[]"}],"name":"GranularRoyaltiesSet","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256[]","name":"_ids","type":"uint256[]"}],"name":"GranularTokenManagersRemoved","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256[]","name":"_ids","type":"uint256[]"},{"indexed":false,"internalType":"address[]","name":"_tokenManagers","type":"address[]"}],"name":"GranularTokenManagersSet","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint8","name":"version","type":"uint8"}],"name":"Initialized","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"newLimitSupply","type":"uint256"}],"name":"LimitSupplySet","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"minter","type":"address"},{"indexed":true,"internalType":"bool","name":"registered","type":"bool"}],"name":"MinterRegistrationChanged","type":"event"},{"anonymous":false,"inputs":[],"name":"MintsFrozen","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"newRoyaltyManager","type":"address"}],"name":"RoyaltyManagerChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256[]","name":"ids","type":"uint256[]"},{"indexed":false,"internalType":"string[]","name":"uris","type":"string[]"}],"name":"TokenURIsSet","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Transfer","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"implementation","type":"address"}],"name":"Upgraded","type":"event"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"baseURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"burn","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"contractURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"customRendererConfig","outputs":[{"internalType":"address","name":"renderer","type":"address"},{"internalType":"bool","name":"processMintDataOnRenderer","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"defaultManager","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"freezeMints","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"getApproved","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"creator","type":"address"},{"internalType":"string","name":"_contractURI","type":"string"},{"components":[{"internalType":"address","name":"recipientAddress","type":"address"},{"internalType":"uint16","name":"royaltyPercentageBPS","type":"uint16"}],"internalType":"struct IRoyaltyManager.Royalty","name":"defaultRoyalty","type":"tuple"},{"internalType":"address","name":"_defaultTokenManager","type":"address"},{"internalType":"string","name":"_name","type":"string"},{"internalType":"string","name":"_symbol","type":"string"},{"internalType":"address","name":"trustedForwarder","type":"address"},{"internalType":"address","name":"initialMinter","type":"address"},{"internalType":"string","name":"newBaseURI","type":"string"},{"internalType":"uint256","name":"_limitSupply","type":"uint256"},{"internalType":"bool","name":"useMarketplaceFiltererRegistry","type":"bool"},{"internalType":"address","name":"_observability","type":"address"}],"name":"initialize","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"creator","type":"address"},{"internalType":"bytes","name":"data","type":"bytes"}],"name":"initialize","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"operator","type":"address"}],"name":"isApprovedForAll","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"forwarder","type":"address"}],"name":"isTrustedForwarder","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"limitSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"mintAmountToOneRecipient","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address[]","name":"recipients","type":"address[]"}],"name":"mintOneToMultipleRecipients","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"}],"name":"mintOneToOneRecipient","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address[]","name":"recipients","type":"address[]"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"mintSameAmountToMultipleRecipients","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"minters","outputs":[{"internalType":"address[]","name":"","type":"address[]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"observability","outputs":[{"internalType":"contract IObservability","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"ownerOf","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"proxiableUUID","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"minter","type":"address"}],"name":"registerMinter","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"removeDefaultTokenManager","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"_ids","type":"uint256[]"}],"name":"removeGranularTokenManagers","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"removeRoyaltyManager","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_tokenGroupingId","type":"uint256"},{"internalType":"uint256","name":"_salePrice","type":"uint256"}],"name":"royaltyInfo","outputs":[{"internalType":"address","name":"receiver","type":"address"},{"internalType":"uint256","name":"royaltyAmount","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"royaltyManager","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"bytes","name":"_data","type":"bytes"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"operator","type":"address"},{"internalType":"bool","name":"approved","type":"bool"}],"name":"setApprovalForAll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"newBaseURI","type":"string"}],"name":"setBaseURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"newName","type":"string"},{"internalType":"string","name":"newSymbol","type":"string"},{"internalType":"string","name":"newContractUri","type":"string"}],"name":"setContractMetadata","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"components":[{"internalType":"address","name":"renderer","type":"address"},{"internalType":"bool","name":"processMintDataOnRenderer","type":"bool"}],"internalType":"struct ERC721GeneralSequenceBase.CustomRendererConfig","name":"_customRendererConfig","type":"tuple"}],"name":"setCustomRenderer","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"components":[{"internalType":"address","name":"recipientAddress","type":"address"},{"internalType":"uint16","name":"royaltyPercentageBPS","type":"uint16"}],"internalType":"struct IRoyaltyManager.Royalty","name":"_royalty","type":"tuple"}],"name":"setDefaultRoyalty","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_defaultTokenManager","type":"address"}],"name":"setDefaultTokenManager","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"ids","type":"uint256[]"},{"components":[{"internalType":"address","name":"recipientAddress","type":"address"},{"internalType":"uint16","name":"royaltyPercentageBPS","type":"uint16"}],"internalType":"struct IRoyaltyManager.Royalty[]","name":"_newRoyalties","type":"tuple[]"}],"name":"setGranularRoyalties","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"_ids","type":"uint256[]"},{"internalType":"address[]","name":"_tokenManagers","type":"address[]"}],"name":"setGranularTokenManagers","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_limitSupply","type":"uint256"}],"name":"setLimitSupply","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_royaltyManager","type":"address"}],"name":"setRoyaltyManager","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"ids","type":"uint256[]"},{"internalType":"string[]","name":"uris","type":"string[]"}],"name":"setTokenURIs","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"supply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes4","name":"interfaceId","type":"bytes4"}],"name":"supportsInterface","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"id","type":"uint256"}],"name":"tokenManager","outputs":[{"internalType":"address","name":"manager","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"tokenURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"transferFrom","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"minter","type":"address"}],"name":"unregisterMinter","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newImplementation","type":"address"}],"name":"upgradeTo","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newImplementation","type":"address"},{"internalType":"bytes","name":"data","type":"bytes"}],"name":"upgradeToAndCall","outputs":[],"stateMutability":"payable","type":"function"}]
Contract Creation Code
9c4d535b0000000000000000000000000000000000000000000000000000000000000000010007f7cd3ac6efcac2e375e2bc967e921607469f2c1fd533ccacb5bad715ef00000000000000000000000000000000000000000000000000000000000000600000000000000000000000000000000000000000000000000000000000000000
Deployed Bytecode
0x000400000000000200280000000000020000006003100270000007210030019d00000721083001970003000000810355000200000001035500000001002001900000000003000416000000970000c13d000000800b0000390000004000b0043f000000040080008c00000a760000413d000000000201043b000000e005200270000007230050009c000003890000613d000007240050009c0000039a0000613d000007250050009c000000cf07000039000000a50000613d000007260050009c000003a00000613d000007270050009c0000000402100370000003b80000613d000007280050009c0000002404100370000000e80000613d000007290050009c000003bf0000613d0000072a0050009c000000ca0a000039000001010000613d0000072b0050009c000000cb06000039000001290000613d0000072c0050009c000003c90000613d0000072d0050009c000003cd0000613d0000072e0050009c000000440610037000000107090000390000012d0000613d0000072f0050009c000001830000613d000007300050009c000003ec0000613d000007310050009c000004300000613d000007320050009c000004380000613d000007330050009c000004800000613d000007340050009c0000048e0000613d000007350050009c000001880000613d000007360050009c000001d70000613d000007370050009c000001da0000613d000007380050009c000001090c0000390000022f0000613d000007390050009c0000023b0000613d0000073a0050009c000005340000613d0000073b0050009c000005490000613d0000073c0050009c0000055b0000613d0000073d0050009c000005d20000613d0000073e0050009c000005ef0000613d0000073f0050009c000002610000613d000007400050009c000005fa0000613d000007410050009c000006040000613d000007420050009c0000060e0000613d000007430050009c000006260000613d000007440050009c000006310000613d000007450050009c000006480000613d000007460050009c0000065d0000613d000007470050009c000006610000613d000007480050009c000006740000613d000007490050009c0000069f0000613d0000074a0050009c000002c50000613d0000074b0050009c000006b00000613d0000074c0050009c000003220000613d0000074d0050009c000006cf0000613d0000074e0050009c000006e10000613d0000074f0050009c0000037b0000613d000007500050009c000006f00000613d000007510050009c000007f00000613d000007520050009c000007fa0000613d000007530050009c0000080d0000613d000007540050009c000008120000613d000007550050009c0000081a0000613d000007560050009c000008970000613d000007570050009c000008a70000613d000007580050009c000008c30000613d000007590050009c00000a760000c13d000000240080008c00000a760000413d000000000003004b00000a760000c13d000000000102043b000007ca0010019800000a760000c13d000007cb01100197000007cc0010009c00000000020000390000000102006039000007a60010009c00000001022061bf000007cd0010009c00000001022061bf000000800020043f000008170000013d000000a001000039000000400010043f000000000003004b00000a760000c13d0000000001000410000000800010043f000001400000044300000160001004430000002001000039000001000010044300000001010000390000012000100443000007220100004100001c820001042e000000000003004b00000a760000c13d0000000001000415000e00000001001d1c8113770000040f1c81142a0000040f000000cf02000039000000000102041a000007c7011001970000075e011001c7000000000012041b000000400100043d000007210010009c000f00000001001d000007210100804100000040011002100000000002000414000007210020009c0000072102008041000000c002200210000000000112019f0000076b011001c70000800d020000390000000103000039000007c8040000411c811b1d0000040f000000010020019000000a760000613d000100c30000003d00001c640000013d0026075c0010019b000100c60000003d00001b450000013d000000260440008a000000050440021000000760020000411c811aff0000040f000000000001004b00000a760000613d0000000d010000290000075c02100197000007c9010000410000000f0300002900000000001304350000000001000414000000040020008c000000db0000613d00000004040000390000000f0500002900000000060000191c8112420000040f0000000f03000029000000000001004b00000bc60000613d000100dd0000003d00001bf40000013d000007620010009c00000cd00000213d000000010020019000000cd00000c13d000000400010043f00000097010000390000000102000039000000000021041b00000000010004150000000e020000290000047d0000013d000000440080008c00000a760000413d000000000102043b000f00000001001d0000075c0010009c00000a760000213d000000000104043b000e00000001001d1c8117830000040f000d075c0010019b00000000020004110000000d0020006c00000a5b0000613d0000000d01000029000000000010043f0000078f01000041000100fa0000003d00001b4d0000013d00000000020004110000075c02200197000100fe0000003d00001c270000013d00000a5b0000c13d000007c50100004100000f190000013d000000240080008c00000a760000413d000000000003004b00000a760000c13d000000000102043b000f00000001001d0000075c0010009c00000a760000213d0000000001000415000e00000001001d0000006501000039000000000101041a0000075c021001970000000001000411000000000021004b000001130000c13d000101130000003d00001b310000013d0000075c01100197000000000010043f0000002000a0043f000101180000003d00001b610000013d000008db0000613d1c81142a0000040f000000cf01000039000000000101041a0000075d011001970000075e0010009c00000a570000613d0000075f01000041000000000201041a0000010801000039000000000101041a000000000001004b00000d3a0000613d000000000021004b00000d3a0000813d000007c20100004100000f190000013d000000000003004b00000a760000c13d000000000106041a000008150000013d000000640080008c00000a760000413d000000000003004b00000a760000c13d000000000202043b000007620020009c00000a760000213d0000002303200039000000000083004b00000a760000813d0000000403200039000000000331034f000000000303043b000f00000003001d000007620030009c00000a760000213d0000002403200039000e00000003001d0000000f02300029000000000082004b00000a760000213d000000000204043b000007620020009c00000a760000213d0000002303200039000000000083004b00000a760000813d0000000403200039000000000331034f000000000303043b000d00000003001d000007620030009c00000a760000213d0000002403200039000c00000003001d0000000d02300029000000000082004b00000a760000213d000000000206043b000007620020009c00000a760000213d0000002303200039000000000083004b00000a760000813d0000000403200039000000000131034f000000000101043b000b00000001001d000007620010009c00000a760000213d0000002402200039000a00000002001d0000000b01200029000000000081004b00000a760000213d0000000001000415000900000001001d1c8113770000040f000007bf010000410001016a0000003d00001b6d0000013d000007f60000c13d0000000f020000291c8114cd0000040f0000000f020000290000001f0020008c00000001012002100000000302200210000010fd0000a13d000007bf03000041000000000030043f000000200300008a0000000f0530017f000007c003000041000000020400036700000000060000190000000e08000029000000000056004b0000000007860019000011060000813d000000000774034f000000000707043b000000000073041b000000200660003900000001033000390000017a0000013d000000000003004b00000a760000c13d0000010801000039000000000101041a000008160000013d000000240080008c00000a760000413d000000000003004b00000a760000c13d000000000102043b000f00000001001d0000075c0010009c00000a760000213d0000000001000415000e00000001001d1c8113770000040f0000000f010000291c8113e20000040f000000000001004b0000065b0000613d000000400100043d000007210010009c000d00000001001d000007210100804100000040011002100000000002000414000007210020009c0000072102008041000000c002200210000000000112019f0000076b011001c70000800d020000390000000303000039000000010600003900000797040000410000000f050000291c811b1d0000040f000000010020019000000a760000613d000101ac0000003d00001c680000013d001f075c0010019b000101af0000003d00001b450000013d0000001f0440008a000000050440021000000760020000411c811aff0000040f000000000001004b00000a760000613d0000000c010000290000075c021001970000000d050000290000002401500039000000010300003900000000003104350000079801000041000000000015043500000004015000390000000f0300002900000000003104350000000001000414000000040020008c000001cb0000613d000000440400003900000000030500190000000d0500002900000000060000191c8112420000040f0000000d05000029000000000001004b00000bc60000613d000101cd0000003d00001c750000013d000101cf0000003d00001b490000013d000007620020009c00000cd00000213d000000010030019000000cd00000c13d000000400020043f000007630010009c000000e50000a13d00000a760000013d00000000010800191c8112f80000040f1c8113af0000040f000000000003004b00000a760000c13d00000000010800191c8113110000040f000b00000001001d000c00000002001d0000000001000415000900000001001d000101e40000003d00001c6c0000013d0000075c021001970000000001000411000000000021004b000001ea0000c13d000101ea0000003d00001b310000013d000a075c0010019b00000000020000190000000c0020006c00000beb0000813d000e00000002001d00000005012002100000000b01100029000d00000001001d0000000201100367000000000101043b000f00000001001d000000000010043f000000cc01000039000101f90000003d00001be50000013d0000075c02100198000006df0000613d000000400500043d00000024015000390000000f0300002900000000003104350000078c01000041000000000015043500000004015000390000000a0300002900000000003104350000000001000414000000040020008c000002090000c13d0000000102000031000002120000013d00000000030500190000004404000039000f00000005001d00000020060000391c8112420000040f0000000102000031000000000001004b00000d300000613d0000000f05000029000102140000003d00001c370000013d000007620010009c00000cd00000213d000000010030019000000cd00000c13d000000400010043f000007630020009c00000a760000213d000000200020008c00000a760000413d0000000001050433000102200000003d00001b370000013d00000a760000c13d000000000001004b00000a260000613d0000000d010000290000000201100367000000000101043b000000000010043f000000cc010000390001022a0000003d00001b8d0000013d0000076802200197000000000021041b0000000e020000290000000102200039000001ec0000013d000000000003004b00000a760000c13d00000000010c041a0000075c02100197000000800020043f0000075d001001980000000001000039000000010100c039000000a00010043f0000008001000039000000400200003900000bbc0000013d000000240080008c00000a760000413d000000000003004b00000a760000c13d000000000102043b000f00000001001d0000075c0010009c00000a760000213d0000000001000415000e00000001001d1c81142a0000040f0000000f010000291c811a870000040f000000000001004b000006ae0000613d0000000f010000291c811aaf0000040f000000000001004b000006ae0000c13d0000002402000039000000400100043d0000002003100039000007a60400004100000000004304350000002404100039000007a70500004100000000005404350000000000210435000007a80010009c00000cd00000213d0000006002100039000000400020043f0000000f02000029000000040020008c00000e2b0000c13d0000000001030433000000000010043f00000e2f0000013d000000440080008c00000a760000413d000000000003004b00000a760000c13d000000000102043b000f00000001001d0000075c0010009c00000a760000213d000000000104043b000e00000001001d0000000001000415000d00000001001d0000006501000039000000000101041a0000075c021001970000000001000411000000000021004b000002750000c13d000102750000003d00001b310000013d0000075c01100197000000000010043f0000002000a0043f0001027a0000003d00001b610000013d000008db0000613d1c81142a0000040f000000cf01000039000000000101041a0000075d011001970000075e0010009c00000a570000613d0000075f01000041000000000301041a000000000003004b00000d4b0000613d0000000f010000290000000e02000029000c00000003001d1c811a010000040f000000010100008a0000000e0210014f0000000c01000029000000010110008a000000000021004b00000d4b0000213d0000010802000039000000000202041a000000000002004b000002960000613d0000000e01100029000000000012004b000001270000413d000102980000003d00001c1c0000013d0000047a0000613d0000000001010433000b00000001001d001a075c0010019b0001029e0000003d00001b450000013d0000001a0440008a000000050440021000000760020000411c811aff0000040f000000000001004b00000a760000613d0000000b010000290000075c02100197000000400400043d00000044014000390000000f03000029000000000031043500000024014000390000000e0300002900000000003104350000079d010000410000000000140435000f00000004001d00000004014000390000000c0300002900000000003104350000000001000414000000040020008c000002bb0000613d00000064040000390000000f03000029000102ba0000003d00001b5c0000013d00000bc60000613d000102bd0000003d00001b3c0000013d000007620020009c00000cd00000213d000000010030019000000cd00000c13d000000400020043f000007630010009c0000047a0000a13d00000a760000013d000000000003004b00000a760000c13d0000000001000415000f00000001001d000102cb0000003d00001c6c0000013d0000075c021001970000000001000411000000000021004b000002d10000c13d000102d10000003d00001b310000013d000000cb02000039000000000202041a0000075c02200198000006df0000613d000000400500043d0000078c0300004100000000003504350000075c01100197000102db0000003d00001c700000013d0000000001000414000000040020008c000002e70000613d000000440400003900000020060000390000000003050019000e00000005001d0000000e050000291c8112420000040f0000000e05000029000000000001004b00000bc60000613d000102e90000003d00001bc60000013d000007620030009c00000cd00000213d000000010020019000000cd00000c13d0000000e02000029000000400020043f000007630010009c00000a760000213d000000200010008c00000a760000413d0000000001050433000102f60000003d00001b370000013d00000a760000c13d000000000001004b00000a260000613d000000cb02000039000000000102041a0000076801100197000000000012041b0000000e01000029000007210010009c000007210100804100000040011002100000000002000414000007210020009c0000072102008041000000c002200210000000000112019f0000076b011001c70000800d0200003900000002030000390000078d0400004100000000050000191c811b1d0000040f000000010020019000000a760000613d000103100000003d00001c640000013d0017075c0010019b000103130000003d00001b450000013d000000170440008a000000050440021000000760020000411c811aff0000040f000000000001004b00000cdf0000613d0000000d010000290000075c021001970000078e010000410000000e03000029000000000013043500000004013000390000000000010435000000000100041400000f800000013d000000440080008c00000a760000413d000000000003004b00000a760000c13d000000000202043b000007620020009c00000a760000213d0000002303200039000000000083004b00000a760000813d0000000403200039000000000331034f000000000303043b000d00000003001d000007620030009c00000a760000213d000c00240020003d0000000d0200002900000005022002100000000c02200029000000000082004b00000a760000213d000000000204043b000007620020009c00000a760000213d0000002303200039000000000083004b00000a760000813d0000000403200039000000000131034f000000000101043b000b00000001001d000007620010009c00000a760000213d000a00240020003d0000000b0100002900000006011002100000000a01100029000000000081004b00000a760000213d0000000001000415000500000001001d0001034e0000003d00001c6c0000013d0000075c021001970000000001000411000000000021004b000003540000c13d000103540000003d00001b310000013d000000cf02000039000000000202041a0009075c0020019c00000ff40000c13d0000003302000039000000000202041a000000000112013f0000075c0010019800000f180000c13d00000000030000190000000d0030006c000010a30000813d0000000b0030006c000010490000813d00000006013002100000000a0410002900000020024000390000000201000367000000000221034f000000000202043b0000ffff0020008c00000a760000213d000f00000004001d000027100020008c000008a50000213d0000000502300210000e00000003001d0000000c02200029000000000121034f000000000101043b000000000010043f000000ce01000039000103760000003d00001b520000013d0000000f020000291c81143a0000040f0000000e0300002900000001033000390000035e0000013d000000000003004b00000a760000c13d00000000010800191c8112f80000040f000f00000002001d0000000003000415000000000400041a000eff00004001930000ffff00400190000c00000003001d000d00000001001d000008f40000c13d0000000101000039000009040000013d000000000003004b00000a760000c13d000000c902000039000000000102041a000000800010043f000000000020043f0000075a0300004100000080020000390000000004000019000000000014004b000008dd0000813d0000002002200039000000000503041a000000000052043500000001044000390000000103300039000003920000013d000000000003004b00000a760000c13d0000075f01000041000000000101041a000000010110008a000008160000013d000000000003004b00000a760000c13d000007bf01000041000000000201041a000000010320019000000001012002700000007f0110618f0000001f0010008c00000000040000390000000104002039000000000043004b000007f60000c13d000000800010043f000000000003004b00000a3d0000613d000007bf02000041000000000020043f000007c0030000410000000002000019000000000012004b00000a420000813d000103b70000003d00001bab0000013d000003b30000013d000000240080008c00000a760000413d000000000003004b00000a760000c13d000000000102043b1c8117a30000040f0000060a0000013d000000000003004b00000a760000c13d000007bb01000041000000000101041a000000010200008a000000000121013f0000075f02000041000000000202041a0000000001120019000008160000013d00000000010800191c8112af0000040f1c8117d60000040f00000bba0000013d000000440080008c00000a760000413d000000000003004b00000a760000c13d000000000104043b000f00000001001d000000000102043b000000000010043f000000ce01000039000103d80000003d00001b4d0000013d000000c003000039000000400030043f000000000201041a0000075c01200198000000800010043f000000a0022002700000ffff0420018f000000a00040043f000009900000c13d0000010003000039000000400030043f0000012002000039000000cd01000039000000000401041a0000075c01400197000000c00010043f000000a0044002700000ffff0440018f000000e00040043f000009910000013d000000440080008c00000a760000413d000000000003004b00000a760000c13d000000000202043b000007620020009c00000a760000213d0000002303200039000000000083004b00000a760000813d0000000403200039000000000131034f000000000101043b000d00000001001d000007620010009c00000a760000213d000e00240020003d0000000d0100002900000005011002100000000e01100029000000000081004b00000a760000213d000000000104043b000c00000001001d0000000001000415000b00000001001d0000006501000039000000000101041a0000075c021001970000000001000411000000000021004b0000040e0000c13d0001040e0000003d00001b310000013d0000075c01100197000000000010043f0000002000a0043f000104130000003d00001b610000013d000008db0000613d1c81142a0000040f000000cf01000039000000000101041a0000075d011001970000075e0010009c00000a570000613d0000075f01000041000000000101041a000a00000001001d000000000001004b00000d4b0000613d0000000a01000029000900010010009200000000020000190000000d0020006c0000104d0000813d000f00000002001d00000005012002100000000e011000290000000201100367000000000101043b0000075c0010009c00000a760000213d0000000c020000291c811a010000040f0000000f020000290000000102200039000004220000013d000000240080008c00000a760000413d000000000003004b00000a760000c13d000000000102043b0000075c0010009c00000a760000213d1c8113af0000040f000000240080008c00000a760000413d000000000003004b00000a760000c13d000000000102043b000f00000001001d0000000001000415000d00000001001d1c8113770000040f1c81142a0000040f00000108010000390000000f05000029000000000051041b000000400100043d000007210010009c000e00000001001d000007210100804100000040011002100000000002000414000007210020009c0000072102008041000000c002200210000000000112019f0000076b011001c70000800d020000390000000203000039000007bd040000411c811b1d0000040f000000010020019000000a760000613d000104580000003d00001c680000013d0022075c0010019b0001045b0000003d00001b450000013d000000220440008a000000050440021000000760020000411c811aff0000040f000000000001004b00000a760000613d0000000c010000290000075c02100197000007be010000410000000e03000029000000000013043500000004013000390000000f0400002900000000004104350000000001000414000000040020008c000004730000613d00000024040000390000000e0500002900000000060000191c8112420000040f0000000e03000029000000000001004b00000bc60000613d000104750000003d00001bf40000013d000007620010009c00000cd00000213d000000010020019000000cd00000c13d000000400010043f0001047c0000003d00001c110000013d0000000d020000290000000001120049000000000100000200000bba0000013d000000640080008c00000a760000413d000000000102043b0000075c0010009c00000a760000213d000000000204043b0000075c0020009c00000a760000213d000000000306043b000000a004000039000000400040043f000000800000043f0000008004000039000006cd0000013d000000240080008c00000a760000413d000000000003004b00000a760000c13d000000000102043b000f00000001001d0000000001000415000c00000001001d1c81142a0000040f0000000f010000291c8114770000040f000000000301001900000000020004110000006501000039000000000101041a0000075c01100197000e00000002001d000000000012004b000004a40000c13d000104a30000003d00001be00000013d000e006000100278000000000003004b0000099e0000c13d0000000f010000291c8117830000040f0000075c01100197000b00000001001d000000000010043f0000078f01000041000104ae0000003d00001b4d0000013d0000000e020000290000075c02200197000d00000002001d000104b30000003d00001c270000013d000004bb0000c13d0000000f010000291c8117a30000040f0000000d030000290000000b0030006c000004bb0000613d0000000d0010006b00000f180000c13d0000000f010000291c8117830000040f0000000f02000029000000000020043f000007b402000041000000200020043f000d00000001001d000000400200003900000000010000191c811aeb0000040f0000000d020000290000075c03200197000000000201041a000000000002004b000004cb0000613d000000000001041b000000000030043f0000079a01000041000000200010043f00000040020000390000000001000019000b00000003001d1c811aeb0000040f000000000201041a000007b50220009a000000000021041b0000800b0100003900000004030000390000000004000415000000280440008a0000000504400210000007b6020000411c811aff0000040f000a00000001001d0000000f01000029000000000010043f000007b701000041000104e20000003d00001b520000013d0000000b050000290000000a02000029000000a002200210000000000252019f000007b8022001c7000000000021041b0000000d01000029000007b900100198000004fc0000c13d0000000f010000290000000101100039000a00000001001d000000000010043f000007b701000041000104f20000003d00001b4d0000013d0000000b05000029000000000201041a000000000002004b000004fc0000c13d0000075f02000041000000000202041a0000000a0020006b000004fc0000613d0000000d02000029000000000021041b000000400100043d000007210010009c000007210100804100000040011002100000000002000414000007210020009c0000072102008041000000c002200210000000000112019f0000076b011001c70000800d020000390000000403000039000007ba0400004100000000060000190000000f070000291c811b1d0000040f000000010020019000000a760000613d000007bb01000041000000000201041a0000000102200039000000000021041b0000000f01000029000000000010043f000000d101000039000105170000003d00001b4d0000013d000000000301041a000000010030019000000001043002700000007f0440618f0000001f0040008c00000000020000390000000102002039000000000223013f0000000100200190000007f60000c13d000000000004004b00000f240000613d000d00000004001d0000001f0040008c00000f230000a13d000b00000001001d000000000010043f000000200200003900000000010000191c811aeb0000040f0000000d020000290000001f02200039000000050220027000000000022100190000000101100039000000000021004b00000f1c0000813d000000000001041b0000052f0000013d000000440080008c00000a760000413d000000000003004b00000a760000c13d1c8113770000040f00000002020003670000000401200370000000000101043b0000075c0010009c00000a760000213d000000000001004b00000a770000c13d000000400100043d0000004402100039000007a503000041000000000032043500000024021000390000000d030000390000000000320435000007660200004100001bda0000013d000000000003004b00000a760000c13d0000000001000412002800000001001d002700000000003d000080050100003900000044030000390000000004000415000000280440008a0000000504400210000007a0020000411c811aff0000040f0000075c011001970000000002000410000000000012004b000009f50000c13d000007a301000041000008160000013d000001a40080008c00000a760000413d000000000003004b00000a760000c13d000000000202043b000f00000002001d0000075c0020009c00000a760000213d000000000304043b000007620030009c00000a760000213d0000002302300039000000000082004b00000a760000813d0000000402300039000000000121034f000000000201043b0000002401300039000d00000008001d00000000030800191c8112c00000040f000c00000001001d000000400100043d000e00000001001d0000077a0010009c00000cd00000213d0000000e010000290000004001100039000000400010043f00000002010003670000004402100370000000000202043b0000075c0020009c0000000d0300002900000a760000213d0000000e0400002900000000022404360000006404100370000000000404043b0000ffff0040008c00000a760000213d00000000004204350000008402100370000000000202043b000b00000002001d0000075c0020009c00000a760000213d000000a402100370000000000402043b000007620040009c00000a760000213d0000002302400039000000000032004b00000a760000813d000105930000003d00001bfd0000013d0000000d03000029000a00000001001d0000000202000367000000c401200370000000000101043b000007620010009c00000a760000213d0000002304100039000000000034004b00000a760000813d0000000404100039000000000242034f000000000202043b00000024011000391c8112c00000040f0000000d03000029000900000001001d0000000201000367000000e402100370000000000202043b000800000002001d0000075c0020009c00000a760000213d0000010402100370000000000202043b000700000002001d0000075c0020009c00000a760000213d0000012402100370000000000402043b000007620040009c00000a760000213d0000002302400039000000000032004b00000a760000813d000105b80000003d00001bfd0000013d000d00000001001d00000002010003670000014402100370000000000202043b000600000002001d0000016402100370000000000202043b000000000002004b0000000003000039000000010300c039000000000032004b00000a760000c13d0000018401100370000000000101043b000500000001001d0000075c0010009c00000a760000213d0000000001000415000300000001001d000000000200041a0004ff0000200193000200000002001d0000ffff00200190000012100000c13d00000001010000390000121e0000013d000000240080008c00000a760000413d000000000003004b00000a760000c13d000000000202043b000007620020009c00000a760000213d0000002303200039000000000083004b00000a760000813d000e00040020003d0000000e01100360000000000101043b000f00000001001d000007620010009c00000a760000213d0000002402200039000d00000002001d0000000f01200029000000000081004b00000a760000213d0000000001000415000c00000001001d1c81142a0000040f0000000f06000029000000000006004b00000d4f0000c13d0000079f0100004100000f190000013d000000000003004b00000a760000c13d00000000010800191c8112a40000040f0000006502000039000000000202041a000000000112013f0000075c0010019800000000020000390000000102006039000006010000013d000000240080008c00000a760000413d000000000003004b00000a760000c13d000000000102043b1c8117830000040f0000075c02100197000000400100043d0000000000210435000008180000013d000000240080008c00000a760000413d000000000003004b00000a760000c13d000000000102043b1c8114770000040f000000400300043d0000000000130435000000200200003900000a4f0000013d000000000003004b00000a760000c13d000000d503000039000000000203041a000000010420019000000001012002700000007f0110618f0000001f0010008c00000000050000390000000105002039000000000552013f0000000100500190000007f60000c13d000000800010043f000000000004004b00000a3d0000613d000000000030043f0000079c030000410000000002000019000000000012004b00000a420000813d000106250000003d00001bab0000013d000006210000013d000000240080008c00000a760000413d000000000003004b00000a760000c13d000000000102043b0000075c0010009c00000a760000213d000000000001004b00000a8c0000c13d0000079b0100004100000f190000013d000000000003004b00000a760000c13d1c8113770000040f0000003303000039000000000103041a0000076802100197000000000023041b000000400200043d000007210020009c000007210200804100000040022002100000000003000414000007210030009c0000072103008041000000c003300210000000000223019f0000075c051001970000076b012001c70000800d0200003900000003030000390000079904000041000000000600001900000a730000013d000000240080008c00000a760000413d000000000003004b00000a760000c13d000000000102043b000f00000001001d0000075c0010009c00000a760000213d0000000001000415000e00000001001d1c8113770000040f0000000f01000029000000000010043f000000ca01000039000106580000003d00001b4d0000013d000000000401041a000000000004004b00000a950000c13d000007b10100004100000f190000013d000000000003004b00000a760000c13d0000003301000039000008100000013d000000000003004b00000a760000c13d0000079201000041000000000201041a000106670000003d00001c030000013d000007f60000c13d000000800010043f000000000003004b00000a3d0000613d0000079202000041000000000020043f00000793030000410000000002000019000000000012004b00000a420000813d000106730000003d00001bab0000013d0000066f0000013d000000440080008c00000a760000413d000000000003004b00000a760000c13d000000000102043b000f00000001001d0000075c0010009c00000a760000213d000000000204043b000000000002004b0000000001000039000000010100c039000e00000002001d000000000012004b00000a760000c13d00000000010004110000075c01100197000000000010043f0000078f01000041000106890000003d00001b4d0000013d0000000f02000029000000000020043f0001068d0000003d00001b520000013d000001000200008a000000000301041a000000000223016f0000000e03000029000000000232019f000000000021041b000000800030043f0000000001000414000007210010009c0000072101008041000000c00110021000000790011001c70000800d020000390000000303000039000007910400004100000000050004110000000f06000029000006470000013d000000240080008c00000a760000413d000000000003004b00000a760000c13d000000000102043b000f00000001001d0000075c0010009c00000a760000213d0000000001000415000e00000001001d1c81142a0000040f0000000f010000291c811a540000040f000000000001004b00000ae30000c13d000007ab0100004100000f190000013d000000840080008c00000a760000413d000000000202043b000f00000002001d0000075c0020009c00000a760000213d000000000204043b000e00000002001d0000075c0020009c00000a760000213d0000006402100370000000000302043b000007620030009c00000a760000213d0000002302300039000000000082004b00000a760000813d0000000402300039000000000121034f000000000201043b000000000106043b000d00000001001d000000240130003900000000030800191c8112c00000040f00000000040100190000000f010000290000000e020000290000000d030000291c8119340000040f00000bba0000013d000000000003004b00000a760000c13d0000000001000415000f00000001001d000106d50000003d00001c6c0000013d0000075c021001970000000001000411000000000021004b000006db0000c13d000106db0000003d00001b310000013d000000cf02000039000000000202041a0000075c0220019800000a020000c13d000007b00100004100000f190000013d000000240080008c00000a760000413d000000000003004b00000a760000c13d000000000402043b00000000020c041a0000075c0220019800000a280000c13d0000000001040019000f00000004001d1c8117b60000040f000000000001004b00000b360000c13d000007830100004100000f190000013d000000000003004b00000a760000c13d00000000010800191c81134d0000040f000700000001001d000800000002001d000c00000003001d000f00000004001d0000000001000415000500000001001d1c81142a0000040f0000000f02000029000000080020006b00000a300000c13d0000000002000019000000080020006c00000dd70000813d000b00000002001d00000005042002100000000c014000290000000202000367000000000112034f000000000301043b00000000090000310000000c0590006a0000001f0550008a00000778065001970000077807300197000000000867013f000000000067004b00000000060000190000077806004041000000000053004b00000000050000190000077805008041000007780080009c000000000605c0190000000704400029000000000442034f000000000104043b000000000006004b00000a760000c13d0000000c03300029000000000232034f000000000502043b000007620050009c00000a760000213d0000000007590049000000200630003900000778027001970000077803600197000000000423013f000000000023004b00000000020000190000077802004041000e00000006001d000000000076004b00000000030000190000077803002041000007780040009c000000000203c019000000000002004b00000a760000c13d000f00000005001d000d00000001001d1c8114770000040f00000000020100190000006501000039000000000101041a0000075c011001970000000003000411000000000013004b00000000010300190000073c0000c13d0001073c0000003d00001b310000013d000000000002004b000000200300008a0000000f08000029000000000938016f000a001f0080003d000900000009001d000007890000613d000000400b00043d0000004403b00039000000600400003900000000004304350000002403b000390000000d0a0000290000000000a30435000007790300004100000000003b04350000075c011001970000000403b0003900000000001304350000006401b0003900000000008104350000008401b0003900000000039100190000000e040000290000000204400367000000000009004b0000075d0000613d000000000504034f0000000006010019000000005705043c0000000006760436000000000036004b000007590000c13d0000001f05800190000007630000613d000000000494034f0000000305500210000107630000003d00001b270000013d000000000181001900000000000104350000000001000414000000040020008c000007740000613d000000200300008a0000000a0330017f000000840430003900000000030b001900000000050b0019000000200600003900060000000b001d1c8112420000040f000000060b0000290000000d0a000029000000000001004b00000bc60000613d000107760000003d00001b570000013d0000000002b30019000107790000003d00001b490000013d000007620020009c00000cd00000213d000000010030019000000cd00000c13d000000400020043f000007630010009c00000a760000213d000000200010008c00000a760000413d00000000010b0433000107850000003d00001b370000013d00000a760000c13d000000000001004b0000078f0000c13d00000f180000013d0000003302000039000000000202041a000000000112013f0000075c001001980000000d0a00002900000f180000c13d0000000000a0043f000000d101000039000107930000003d00001b520000013d0000000005010019000000000101041a000000010010019000000001031002700000007f0330618f0000001f0030008c00000000020000390000000102002039000000000121013f0000000100100190000007f60000c13d000000200030008c0000000f04000029000d00000005001d000007b70000413d000000000050043f00000000010000190000002002000039000600000003001d1c811aeb0000040f0000000d050000290000000f040000290000000a020000290000000503200270000000200040008c000000000300401900000006020000290000001f02200039000000050220027000000000022100190000000001310019000000000021004b000007b70000813d000000000001041b0000000101100039000007b20000013d000000200040008c00000001064002100000000307400210000007cf0000413d000600000007001d000a00000006001d000000000050043f000000000100001900000020020000391c811aeb0000040f0000000f05000029000000020200036700000000030000190000000e060000290000000907000029000000000073004b0000000004630019000007d50000813d000000000442034f000000000404043b000000000041041b00000020033000390000000101100039000007c60000013d000000000004004b0000000e01000029000007e40000613d0000000201100367000000000101043b000007e50000013d000000000057004b000007e00000813d0000000603000029000000f80330018f000000010500008a000000000335022f000000000353013f000000000242034f000000000202043b000000000232016f000000000021041b0000000a0100002900000001011001bf0000000d05000029000007ea0000013d0000000001000019000000010300008a000000000273022f000000000232013f000000000121016f000000000161019f000000000015041b0000000b02000029000000010020003a000000010220003900000d4b0000413d000006ff0000013d000000000003004b00000a760000c13d000000000209041a000107f50000003d00001c030000013d00000a320000613d0000079401000041000000000010043f000000220100003900000cd30000013d000000440080008c00000a760000413d000000000003004b00000a760000c13d000000000102043b0000075c0010009c00000a760000213d000000000204043b000f00000002001d0000075c0020009c00000a760000213d1c81176d0000040f0000000f020000291c8117710000040f000000000101041a000000ff001001900000000002000039000000010200c039000006010000013d000000000003004b00000a760000c13d000000d001000039000000000101041a000008150000013d000000000003004b00000a760000c13d000000000107041a0000075c01100197000000800010043f0000008001000039000000200200003900000bbc0000013d000000000003004b00000a760000c13d00000000010800191c81134d0000040f000700000001001d000a00000002001d000800000003001d000900000004001d0000000001000415000400000001001d000108260000003d00001c6c0000013d0000075c021001970000000001000411000000000021004b0000082c0000c13d0001082c0000003d00001b310000013d0000003302000039000000000202041a000000000212013f0006075c0010019b0005075c0020019b00000000020000190000000a0020006c00000ce20000813d000000090020006c000010490000813d000e00000002001d0000000502200210000f00000002001d0000000801200029000d00000001001d0000000201100367000000000101043b0000075c0010009c00000a760000213d1c811a540000040f000000000001004b000006ae0000613d0000000f020000290000000701200029000f00000001001d0000000201100367000000000101043b1c8114770000040f000000000001004b000008820000613d000c00000001001d0000000f010000290000000201100367000000000101043b000b00000001001d0000000e01000029000000000010043f000000cc01000039000108540000003d00001be50000013d000000400500043d00000024025000390000000b0300002900000000003204350000075c0110019700000044025000390000000000120435000007700100004100000000001504350000000401500039000000060200002900000000002104350000000c020000290000000001000414000000040020008c0000086d0000613d00000000030500190000006404000039000c00000005001d0000000c0500002900000020060000391c8112420000040f0000000c05000029000000000001004b00000bc60000613d0001086f0000003d00001c750000013d000108710000003d00001b490000013d000007620020009c00000cd00000213d000000010030019000000cd00000c13d000000400020043f000007630010009c0000000f0300002900000a760000213d000000200010008c00000a760000413d00000000010504330001087e0000003d00001b370000013d00000a760000c13d000000000001004b000008850000c13d00000c7d0000013d000000050000006b0000000f0300002900000f180000c13d00000002010003670000000d02100360000000000202043b000d00000002001d0000075c0020009c00000a760000213d000000000131034f000000000101043b000000000010043f000000cc01000039000108910000003d00001b8d0000013d00000768022001970000000d022001af000000000021041b0000000e020000290000000102200039000008320000013d000000440080008c00000a760000413d000000000003004b00000a760000c13d0000000001000415000f00000001001d1c81142a0000040f00000002010003670000002402100370000000000302043b0000ffff0030008c00000a760000213d000027100030008c00000b5a0000a13d0000078b0100004100000f190000013d000000240080008c00000a760000413d000000000003004b00000a760000c13d000000000102043b000f00000001001d0000075c0010009c00000a760000213d1c8113770000040f0000000f01000029000000000001004b00000bb90000c13d000000400100043d0000006402100039000007640300004100000000003204350000004402100039000007650300004100000000003204350000002402100039000000260300003900000000003204350000076602000041000000000021043500000004021000390000002003000039000000000032043500000a000000013d000000000003004b00000a760000c13d00000000010800191c8113110000040f000e00000001001d000d00000002001d0000000001000415000c00000001001d0000006501000039000000000101041a0000075c011001970000000002000411000000000012004b0000000001020019000008d40000c13d000108d40000003d00001b310000013d0000075c01100197000000000010043f000000ca01000039000000200010043f000108da0000003d00001b610000013d00000a510000c13d000007c40100004100000f190000013d000000410120008a000000200200008a000000000321016f0000075b0030009c00000cd00000213d0000008001300039000000400010043f00000020020000390000000000210435000000a004300039000000800200043d00000000002404350000008004000039000000c0033000390000000005000019000000000025004b00000a590000813d000000200440003900000000060404330000075c0660019700000000036304360000000105500039000008ec0000013d0000000001000410001400000001001d00008002010000390000002403000039000b00000004001d0000000004000415000000140440008a000000050440021000000760020000411c811aff0000040f0000000b02000029000000ff0220018f000000010220015f00000000001201a0000000000100003900000001010060391c8114860000040f000001000100008a000000000200041a000000000112016f00000001011001bf0000ffff0200008a000000000221016f00000100022001bf0000000e0000006b000000000201c019000000000020041b0000000f010000290000000001010433000007630010009c00000a760000213d000001800010008c00000a760000413d0000000f020000290000002002200039000b00000002001d0000000004020433000007620040009c00000a760000213d0000000b02100029000a00000002001d0000000b014000291c8114950000040f0000000f0200002900000040022000390000000a03200069000900000001001d000007630030009c00000a760000213d000000400030008c00000a760000413d000000400100043d000800000001001d0000077a0010009c00000cd00000213d00000008010000290000004001100039000000400010043f00000000010204330000075c0010009c00000a760000213d000000080200002900000000011204360000000f02000029000000600220003900000000020204330000ffff0020008c00000a760000213d00000000002104350000000f0100002900000080011000390000000001010433000700000001001d0000075c0010009c00000a760000213d0000000f01000029000000a0011000390000000001010433000007620010009c00000a760000213d0000000b011000290000000a020000291c8114950000040f000600000001001d0000000f01000029000000c0011000390000000001010433000007620010009c00000a760000213d0000000b011000290000000a020000291c8114950000040f000500000001001d0000000f01000029000000e0011000390000000001010433000400000001001d0000075c0010009c00000a760000213d0000000f0100002900000100011000390000000001010433000300000001001d0000075c0010009c00000a760000213d0000000f0100002900000120011000390000000001010433000007620010009c00000a760000213d0000000b011000290000000a020000291c8114950000040f00000000090100190000000f01000029000001600110003900000000010104330001096b0000003d00001b370000013d00000a760000c13d0000000f010000290000018001100039000000000b0104330000075c00b0009c00000a760000213d0000000f010000290000014001100039000000000a0104330000000d0100002900000009020000290000000803000029000000070400002900000006050000290000000506000029000000040700002900000003080000291c8115030000040f0000000e0000006b00000f550000c13d000109810000003d00001c570000013d000007210010009c000007210100804100000040011002100000000002000414000007210020009c0000072102008041000000c002200210000000000112019f0000077b011001c70000800d020000390000077c040000411c811b1d0000040f000000010020019000000f550000c13d00000a760000013d000000e0020000390000000f06000029000000000006004b000009980000613d000000010500008a00000000056500d9000000000045004b00000d4b0000413d000000000013043500000000016400a9000027100110011a0000000000120435000000400200003900000a4f0000013d000000400500043d000007a60100004100000000001504350000000401500039000007b20200004100000000002104350000000001000414000000040030008c000b00000003001d000009b20000613d0000002404000039000000200600003900000000020300190000000003050019000d00000005001d0000000d050000291c8112660000040f0000000d05000029000000000001004b00000bc60000613d000109b40000003d00001c750000013d000109b60000003d00001b490000013d000007620020009c00000cd00000213d000000010030019000000cd00000c13d000000400020043f000007630010009c00000a760000213d000000200010008c00000a760000413d0000000001050433000109c20000003d00001b370000013d00000a760000c13d000000000001004b000004a60000613d0000000f010000291c8117830000040f0020000b0000002d000d00000001001d000109cb0000003d00001b450000013d000000200440008a000000050440021000000760020000411c811aff0000040f000000000001004b00000a760000613d000000400300043d00000044013000390000000f0200002900000000002104350000000d010000290000075c0110019700000024023000390000000000120435000007b20100004100000000001304350000000e010000290000075c01100197000d00000003001d0000000402300039000000000012043500000000010004140000000b02000029000000040020008c000009e90000613d00000064040000390000000d03000029000109e80000003d00001b5c0000013d00000bc60000613d000109eb0000003d00001c7b0000013d000109ed0000003d00001b490000013d000007620020009c00000cd00000213d000000010030019000000cd00000c13d000000400020043f000007630010009c000004bb0000a13d00000a760000013d0000076601000041000000800010043f0000002001000039000000840010043f0000003801000039000000a40010043f000007a101000041000000c40010043f000007a201000041000000e40010043f000000800100003900000084020000391c811ad90000040f000000400500043d000007840300004100000000003504350000075c01100197000000040350003900000000001304350000000001000414000000040020008c00000a140000613d000000240400003900000020060000390000000003050019000e00000005001d0000000e050000291c8112660000040f0000000e05000029000000000001004b00000bc60000613d00010a160000003d00001bc60000013d000007620030009c00000cd00000213d000000010020019000000cd00000c13d0000000e02000029000000400020043f000007630010009c00000a760000213d000000200010008c00000a760000413d000000000105043300010a230000003d00001b370000013d00000a760000c13d000000000001004b00000f580000c13d000007af0100004100000f190000013d0000077d03000041000000800030043f000000840040043f0000000003000414000000040020008c00000bbe0000c13d000000000281034f00000cb30000013d000007740100004100000f190000013d000000800010043f000000000003004b00000a3d0000613d000000000090043f00000771030000410000000002000019000000000012004b00000a420000813d00010a3c0000003d00001bab0000013d00000a380000013d000001000100008a000000000112016f000000a00010043f000000c00100003900000a490000013d000007720120009a000007730010009c00000cd00000413d0000003f01200039000000200200008a000000000121016f0000008001100039000000400010043f0000008002000039000f00000001001d1c81129e0000040f0000000f030000290000000002310049000000000103001900000bbc0000013d1c81142a0000040f000000cf01000039000000000101041a0000075d011001970000075e0010009c00000bd70000c13d000007c30100004100000f190000013d000000000213004900000bbc0000013d0000000e01000029000000000010043f000007b40100004100010a600000003d00001b930000013d00000768022001970000000f06000029000000000262019f000000000021041b000000400100043d000007210010009c000007210100804100000040011002100000000002000414000007210020009c0000072102008041000000c002200210000000000112019f0000076b011001c70000800d020000390000000403000039000007c6040000410000000d050000290000000e070000291c811b1d0000040f000000010020019000000bba0000c13d00001b770000013d0000010905000039000000000305041a0000076804300197000000000414019f000000000045041b0000002402200370000000000202043b000000000002004b0000000004000039000000010400c039000000000042004b00000a760000c13d000000000002004b00000000020000190000075e0200c041000007a403300197000000000223019f000000000112019f0000010902000039000000000012041b00000bba0000013d000000000010043f0000079a01000041000000200010043f000000400200003900000000010000191c811aeb0000040f000000000101041a0000076201100197000008160000013d000000c905000039000000000205041a000000000002004b00000d4b0000613d000000010140008a000000000042004b00000da40000c13d000000ca03000039000007960240009a000000000002041b000000000015041b0000000f01000029000000000010043f000000200030043f000000400200003900000000010000191c811aeb0000040f000000000001041b000000400100043d000007210010009c000d00000001001d000007210100804100000040011002100000000002000414000007210020009c0000072102008041000000c002200210000000000112019f0000076b011001c70000800d02000039000000030300003900000797040000410000000f0500002900000000060000191c811b1d0000040f000000010020019000000a760000613d00010abc0000003d00001c680000013d0019075c0010019b00010abf0000003d00001b450000013d000000190440008a000000050440021000000760020000411c811aff0000040f000000000001004b00000cdf0000613d0000000c010000290000075c0210019700000798010000410000000d04000029000000000014043500000004014000390000000f030000290000000000310435000000240140003900000000000104350000000001000414000000040020008c00000ad70000613d00000044040000390000000d0300002900010ad60000003d00001b5c0000013d00000bc60000613d00010ad90000003d00001c7b0000013d00010adb0000003d00001b490000013d000007620020009c00000cd00000213d000000010030019000000cd00000c13d000000400020043f000007630010009c000000e50000a13d00000cdf0000013d0000006501000039000000000101041a0000075c011001970000000003000411000000000013004b00000aec0000c13d00010aeb0000003d00001be00000013d0000006003100270000000cb01000039000000000101041a0000075c0210019800000c500000c13d0000003302000039000000000202041a000000000232013f0000075c0020019800000f180000c13d000000400300043d00000768011001970000000f05000029000000000151019f000000cb02000039000000000012041b000007210030009c000d00000003001d0000072101000041000000000103401900000040011002100000000002000414000007210020009c0000072102008041000000c002200210000000000112019f0000076b011001c70000800d0200003900000002030000390000078d040000411c811b1d0000040f000000010020019000000a760000613d00010b0e0000003d00001c680000013d0018075c0010019b00010b110000003d00001b450000013d000000180440008a000000050440021000000760020000411c811aff0000040f000000000001004b00000a760000613d0000000c010000290000075c021001970000078e010000410000000d03000029000000000013043500000004013000390000000f0300002900000000003104350000000001000414000000040020008c00000b270000613d00000024040000390000000d0300002900010b260000003d00001b5c0000013d00000bc60000613d00010b290000003d00001c7b0000013d00010b2b0000003d00001b490000013d000007620020009c00000cd00000213d000000010030019000000cd00000c13d000000400020043f000007630010009c00000a760000213d00000001010000390000009702000039000000000012041b000000e50000013d0000000f01000029000000000010043f000000d10100003900010b3b0000003d00001b930000013d000000010320019000000001052002700000007f0550618f0000001f0050008c00000000040000390000000104002039000000000442013f0000000100400190000007f60000c13d000000400400043d0000000006540436000000000003004b00000dbb0000613d000e00000004001d000000000010043f00000020020000390000000001000019000d00000005001d000c00000006001d1c811aeb0000040f0000000c060000290000000d0500002900000000020000190000000003620019000000000052004b00000dc00000813d000000000401041a00000000004304350000002002200039000000010110003900000b520000013d0000006502000039000000000202041a0000075c022001970000000004000411000000000024004b00000b650000c13d000000140200008a0000000002200031000000000221034f000000000202043b0000006004200270000000cf02000039000000000202041a0000075c0220019800000c7f0000c13d0000003302000039000000000202041a000000000242013f0000075c0020019800000f180000c13d0000000402100370000000000502043b0000075c0050009c00000a760000213d000000cd02000039000000000302041a0000076804300197000000000454019f000000000042041b0000002401100370000000000601043b0000ffff0060008c00000a760000213d0000076901300197000000a0036002100000076a03300197000000000131019f000000000151019f000000000012041b000000400100043d000007210010009c000e00000001001d000007210100804100000040011002100000000002000414000007210020009c0000072102008041000000c002200210000000000112019f0000076b011001c70000800d0200003900000003030000390000076c040000411c811b1d0000040f000000010020019000000a760000613d000000d001000039000000000101041a00000002020003670000000403200370000000000303043b000d00000003001d0000075c0030009c00000a760000213d0000002402200370000000000202043b000c00000002001d0000ffff0020008c00000a760000213d000b075c0010019b0011000b0000002d00010ba30000003d00001b450000013d000000110440008a000000050440021000000760020000411c811aff0000040f000000000001004b00000a760000613d0000000e0200002900000024012000390000000c0300002900000000003104350000076d01000041000000000012043500000004012000390000000d02000029000000000021043500000000010004140000000b02000029000000040020008c00000f870000613d00000044040000390000000b0200002900000f830000013d1c8113950000040f000000400100043d000000000200001900000000030000191c811ae10000040f000000240400003900000080050000390000000001030019000000000305001900000000060000191c8112660000040f000000000001004b00000cb10000c13d0000000304000367000000010200003100010bca0000003d00001b670000013d00000bd00000613d000000000704034f000000000801001900010bcf0000003d00001c2f0000013d00000bcd0000c13d000000000006004b00000bd60000613d000000000454034f000000030560021000010bd60000003d00001b270000013d1c811ad90000040f0000075f01000041000000000101041a000000000001004b00000d4b0000613d000a00000001001d000b00010010009200000000020000190000000d0020006c00000d470000813d000f00000002001d00000005012002100000000e011000290000000201100367000000000101043b0000075c0010009c00000a760000213d1c8119c30000040f0000000f02000029000000010220003900000bde0000013d000000400100043d000000200200003900000000022104360000000c030000290000000000320435000007ac0030009c00000a760000213d00000040031000390000000c02000029000f00050020021a0000000f0230002900000bfd0000613d0000000b040000290000000204400367000000004504043c0000000003530436000000000023004b00000bf90000c13d0000000103000039000000000003004b00000000000204350000000002120049000007210020009c00000721020080410000006002200210000007210010009c00000721010080410000004001100210000000000112019f0000000002000414000007210020009c0000072102008041000000c002200210000000000112019f0000076b011001c70000800d02000039000007ad040000411c811b1d0000040f000000010020019000000a760000613d000000d001000039000000000101041a000e00000001001d001e075c0010019b00010c190000003d00001b450000013d0000001e0440008a000000050440021000000760020000411c811aff0000040f000000000001004b00000a760000613d000000400300043d00000024013000390000000c020000290000000000210435000007ae010000410000000000130435000000040130003900000020020000390000000000210435000d00000003001d00000044013000390000000f031000290000000f0000006b00000c330000613d0000000b020000290000000202200367000000002402043c0000000001410436000000000031004b00000c2f0000c13d0000000e010000290000075c02100197000000010000008b00000000000304350000000001000414000000040020008c00000c3f0000613d0000000d05000029000000000453004900010c3e0000003d00001c0c0000013d00000bc60000613d00010c410000003d00001c7b0000013d00010c430000003d00001b490000013d000007620020009c00000cd00000213d000000010030019000000cd00000c13d000000400020043f000007630010009c00000a760000213d00000097010000390000000102000039000000000021041b000000000100041500000009020000290000047d0000013d000000400500043d00000044015000390000000f040000290000000000410435000007700100004100000000001504350000075c0130019700010c590000003d00001c700000013d0000000001000414000000040020008c00000c650000613d000000640400003900000020060000390000000003050019000d00000005001d0000000d050000291c8112420000040f0000000d05000029000000000001004b00000bc60000613d00000001010000310000001f02100039000000200300008a000000000232016f0000000004520019000000000024004b00000000020000390000000102004039000007620040009c00000cd00000213d000000010020019000000cd00000c13d0000000003040019000000400040043f000007630010009c00000a760000213d000000200010008c00000a760000413d000000000105043300010c7a0000003d00001b370000013d00000a760000c13d000000000001004b000010670000c13d000007aa0100004100000f190000013d000000400600043d0000076705000041000e00000006001d00000000005604350000000401100370000000000101043b0000075c0010009c00000a760000213d0000075c044001970000000e070000290000004405700039000000000045043500000024047000390000000000340435000000040370003900000000001304350000000001000414000000040020008c00000c9a0000613d00000064040000390000002006000039000000000307001900000000050700191c8112660000040f0000000e07000029000000000001004b00000bc60000613d00010c9c0000003d00001b570000013d000000000273001900010c9f0000003d00001b490000013d000007620020009c00000cd00000213d000000010030019000000cd00000c13d000000400020043f000007630010009c00000a760000213d000000200010008c00000a760000413d0000000e01000029000000000101043300010cac0000003d00001b370000013d00000a760000c13d000000000001004b000010950000c13d000007880100004100000f190000013d0000000302000367000000800b0000390000000101000031000000200300008a00000000043101700000001f0510018f000000800340003900000cbe0000613d000000000602034f000000006706043c000000000b7b043600000000003b004b00000cba0000c13d000000000005004b00000ccb0000613d000000000242034f0000000304500210000000000503043300000000054501cf000000000545022f000000000202043b0000010004400089000000000242022f00000000024201cf000000000252019f00000000002304350000001f02100039000000200300008a000000000232016f0000075b0020009c00000cd60000a13d0000079401000041000000000010043f0000004101000039000000040010043f000007950100004100001c83000104300000008002200039000000400020043f000007630010009c00000cdf0000213d000000200010008c00000cdf0000413d000000800300043d000007620030009c00000e250000a13d000000000100001900000000020000191c811ad90000040f000000400100043d000f00000001001d00000007020000290000000a03000029000000080400002900000009050000291c8114070000040f0000000f020000290000000001210049000007210010009c00000721010080410000006001100210000007210020009c00000721020080410000004002200210000000000121019f0000000002000414000007210020009c0000072102008041000000c002200210000000000121019f0000076b011001c70000800d0200003900000001030000390000076e040000411c811b1d0000040f000000010020019000000a760000613d000000d001000039000000000101041a000f00000001001d0012075c0010019b00010d040000003d00001b450000013d000000120440008a000000050440021000000760020000411c811aff0000040f000000000001004b00000a760000613d0000000f01000029000e075c0010019b000000400200043d0000076f010000410000000000120435000f00000002001d000000040120003900000007020000290000000a03000029000000080400002900000009050000291c8114070000040f000000000401001900000000010004140000000e03000029000000040030008c00000d210000613d00000000020300190000000f03000029000000000434004900010d200000003d00001b5c0000013d00000bc60000613d00010d230000003d00001b3c0000013d000007620020009c00000cd00000213d000000010030019000000cd00000c13d000000400020043f000007630010009c00000a760000213d00000097010000390000000102000039000000000021041b000000000100041500000004020000290000047d0000013d000000030400036700010d330000003d00001b670000013d00000bd00000613d000000000704034f000000000801001900010d380000003d00001c2f0000013d00000d360000c13d00000bd00000013d000d00000002001d0000000f010000291c8119c30000040f00010d3f0000003d00001c1c0000013d00000e720000c13d00010d420000003d00001c110000013d0000000e011000690000000001000002000000400100043d0000000d02000029000006020000013d000000010100008a0000000d0110014f0000000b0010006b00000e9e0000a13d0000079401000041000000000010043f000000110100003900000cd30000013d00000000010004110000006502000039000000000202041a0000075c04200197000000cb02000039000000000202041a0000075c02200198000b001f00600193000a001f0060003d00000ecc0000c13d000000000041004b00000d5d0000c13d00010d5d0000003d00001b310000013d0000003302000039000000000202041a000000000112013f0000075c0010019800000f180000c13d00000000030000310000000d010000290000000f020000291c8112c00000040f1c8116f70000040f00010d690000003d00001c640000013d001b075c0010019b00010d6c0000003d00001b450000013d0000001b0440008a000000050440021000000760020000411c811aff0000040f000000000001004b00000a760000613d000000400300043d0000079e01000041000000000013043500000004013000390000002002000039000000000021043500000024013000390000000f020000290000000000210435000000200100008a0000000004120170000900000003001d000000440130003900000000034100190000000e020000290000002002200039000000020520036700000d8a0000613d000000000205034f0000000006010019000000002702043c0000000006760436000000000036004b00000d860000c13d0000000d020000290000075c022001970000000b0000006b00000d930000613d000000000445034f0000000b05000029000000030550021000010d930000003d00001b270000013d0000000f0110002900000000000104350000000001000414000000040020008c00000d9f0000613d000000200300008a0000000a0330017f0000004404300039000000090300002900010d9e0000003d00001b5c0000013d00000bc60000613d00010da10000003d00001b570000013d0000000902000029000000000223001900000f490000013d000000000012004b000010490000a13d000007960120009a000000000101041a000007960240009a000000000012041b000000000010043f000000ca01000039000000200010043f00000040020000390000000001000019000d00000004001d1c811aeb0000040f0000000d02000029000000000021041b000000c901000039000000000101041a000000000001004b00000f970000c13d0000079401000041000000000010043f000000310100003900000cd30000013d000001000100008a000000000112016f0000000000160435000000400340003900000dc10000013d0000000e04000029000000000143004900010dc40000003d00001c470000013d000007620010009c00000cd00000213d000000010020019000000cd00000c13d0000000003040019000000400010043f00000000020400190000000001030433000000000001004b00000e290000c13d0000000f010000291c8117b60000040f000000000001004b00000f9c0000c13d0000078201000041000000000010043f000000040200003900000000010000191c811ad90000040f000000400100043d000f00000001001d000000070200002900000008030000290000000c0400002900000000050300191c8116890000040f0000000f020000290000000001210049000007210010009c00000721010080410000006001100210000007210020009c00000721020080410000004002200210000000000121019f0000000002000414000007210020009c0000072102008041000000c002200210000000000121019f0000076b011001c70000800d02000039000000010300003900000776040000411c811b1d0000040f000000010020019000000a760000613d000000d001000039000000000101041a000f00000001001d0013075c0010019b00010df90000003d00001b450000013d000000130440008a000000050440021000000760020000411c811aff0000040f000000000001004b00000a760000613d0000000f01000029000e075c0010019b000000400200043d00000777010000410000000000120435000f00000002001d0000000401200039000000070200002900000008030000290000000c0400002900000000050300191c8116890000040f000000000401001900000000010004140000000e03000029000000040030008c00000e160000613d00000000020300190000000f03000029000000000434004900010e150000003d00001b5c0000013d00000bc60000613d00010e180000003d00001b3c0000013d000007620020009c00000cd00000213d000000010030019000000cd00000c13d000000400020043f000007630010009c00000a760000213d00000097010000390000000102000039000000000021041b000000000100041500000005020000290000047d0000013d000000800210003900000080013000391c8114950000040f0000000002010019000000400100043d00000a4b0000013d00010e2d0000003d00001b790000013d000006ae0000613d000000000100043d0000000102000031000000200020008c000006ae0000413d000000000001004b000006ae0000613d0000006501000039000000000101041a0000075c011001970000000004000411000000000014004b00000e3d0000c13d00010e3c0000003d00001be00000013d0000006004100270000000cf01000039000000000101041a0000075c0310019800000fb80000c13d0000003302000039000000000202041a000000000242013f0000075c0020019800000f180000c13d000000400200043d000d00000002001d00000768011001970000000f05000029000000000151019f000000cf02000039000000000012041b0000000d01000029000007210010009c000007210100804100000040011002100000000002000414000007210020009c0000072102008041000000c002200210000000000112019f0000076b011001c70000800d02000039000000020300003900000785040000411c811b1d0000040f000000010020019000000a760000613d00010e5f0000003d00001c680000013d001d075c0010019b00010e620000003d00001b450000013d0000001d0440008a000000050440021000000760020000411c811aff0000040f000000000001004b00000a760000613d0000000c010000290000075c0210019700000786010000410000000d03000029000000000013043500000004013000390000000f030000290000000000310435000000000100041400000b200000013d0000000001010433000c00000001001d0025075c0010019b00010e770000003d00001b450000013d000000250440008a000000050440021000000760020000411c811aff0000040f000000000001004b00000a760000613d0000000c010000290000075c02100197000000400400043d00000044014000390000000f0300002900000000003104350000002401400039000000010300003900000000003104350000079d010000410000000000140435000f00000004001d00000004014000390000000d0300002900000000003104350000000001000414000000040020008c00000e940000613d00000064040000390000000f0300002900010e930000003d00001b5c0000013d00000bc60000613d00010e960000003d00001b3c0000013d000007620020009c00000cd00000213d000000010030019000000cd00000c13d000000400020043f000007630010009c00000a760000213d00000d400000013d0000010801000039000000000101041a000000000001004b00000ea60000613d0000000b030000290000000d02300029000000000021004b000001270000413d00010ea80000003d00001c1c0000013d00000f520000613d0000000001010433000f00000001001d0010075c0010019b00010eae0000003d00001b450000013d000000100440008a000000050440021000000760020000411c811aff0000040f000000000001004b00000a760000613d0000000f010000290000075c02100197000000400400043d000000440140003900000060030000390000000000310435000000240140003900000001030000390000000000310435000007610100004100010ec00000003d00001bb10000013d00000020033000390000000d0040006c000010970000813d0000000e05100360000000000505043b0000075c0050009c00000a760000213d000000000053043500000001044000390000000e05000029000e00200050003d00000ec00000013d0000000203000367000000000041004b00000ed40000c13d000000140100008a0000000001100031000000000113034f000000000101043b0000006001100270000000400700043d000000440470003900000060050000390000000000540435000007790400004100000000004704350000075c01100197000000040470003900000000001404350000000d043003600000006401700039000000000061043500000024017000390000000000010435000000200100008a0000000005160170000900000007001d0000008401700039000000000351001900000eee0000613d000000000604034f0000000007010019000000006806043c0000000007870436000000000037004b00000eea0000c13d0000000b0000006b00000ef50000613d000000000454034f0000000b05000029000000030550021000010ef50000003d00001b270000013d0000000f0110002900000000000104350000000001000414000000040020008c00000f030000613d000000200300008a0000000a0330017f00000084043000390000002006000039000000090300002900000000050300191c8112420000040f000000000001004b00000bc60000613d00010f050000003d00001b570000013d000000090230002900010f080000003d00001b490000013d000007620020009c00000cd00000213d000000010030019000000cd00000c13d000000400020043f000007630010009c00000a760000213d000000200010008c00000a760000413d0000000901000029000000000101043300010f150000003d00001b370000013d00000a760000c13d000000000001004b00000d620000c13d000007b301000041000000000010043f000007750100004100001c83000104300000000b01000029000000000010043f000000200200003900000000010000191c811aeb0000040f0000000b02000029000000000002041b000000000001041b00010f260000003d00001c640000013d0021075c0010019b00010f290000003d00001b450000013d000000210440008a000000050440021000000760020000411c811aff0000040f000000000001004b00000a760000613d0000000d010000290000075c02100197000000400500043d00000044015000390000000f030000290000000000310435000007bc0100004100000000001504350000000e010000290000075c0110019700010f3b0000003d00001c700000013d0000000001000414000000040020008c00000f470000613d00000064040000390000000003050019000f00000005001d0000000f0500002900000000060000191c8112420000040f0000000f05000029000000000001004b00000bc60000613d00010f490000003d00001c750000013d00010f4b0000003d00001b490000013d000007620020009c00000cd00000213d000000010030019000000cd00000c13d000000400020043f000007630010009c00000a760000213d00000001010000390000009702000039000000000012041b00000000010004150000000c020000290000047d0000013d000000cf02000039000000000102041a0000076801100197000000000012041b0000000e01000029000007210010009c000007210100804100000040011002100000000002000414000007210020009c0000072102008041000000c002200210000000000112019f0000076b011001c70000800d020000390000000203000039000007850400004100000000050000191c811b1d0000040f000000010020019000000a760000613d00010f6f0000003d00001c640000013d0015075c0010019b00010f720000003d00001b450000013d000000150440008a000000050440021000000760020000411c811aff0000040f000000000001004b00000a760000613d0000000d010000290000075c0210019700000786010000410000000e030000290000000000130435000000040130003900000000000104350000000001000414000000040020008c00000f870000613d00000024040000390000000e0300002900010f860000003d00001b5c0000013d00000bc60000613d00010f890000003d00001b570000013d0000000e0230002900010f8c0000003d00001b490000013d000007620020009c00000cd00000213d000000010030019000000cd00000c13d000000400020043f000007630010009c00000a760000213d00010f950000003d00001c110000013d0000000f020000290000047d0000013d0000000004010019000000010110008a000000ca03000039000000c90500003900000a9d0000013d000000400100043d000d00000001001d1c81132a0000040f0000000d0400002900000000014100490000001f01100039000000200200008a000000000321016f000000000243001900010fa70000003d00001b490000013d000007620020009c00000cd00000213d000000010030019000000cd00000c13d00000000010400190000000006020019000000400020043f0000000002040433000000000002004b0000106a0000c13d000007810060009c00000cd00000213d00000000020600190000002001600039000000400010043f000000000006043500000e290000013d0000075c01400197000000400500043d00000024045000390000000000140435000007a9010000410000000000150435000c00000005001d00000004015000390000000f0400002900000000004104350000000001000414000000040030008c00000fd80000613d0000004404000039000000200600003900000000020300190000000c0300002900000000050300191c8112660000040f0000000102000031000000000001004b00000fd80000c13d000000030400036700010fd10000003d00001b670000013d00000bd00000613d000000000704034f000000000801001900010fd60000003d00001c2f0000013d00000fd40000c13d00000bd00000013d0000001f01200039000000200300008a000000000131016f0000000c03100029000000000013004b00000000010000390000000101004039000d00000003001d000007620030009c00000cd00000213d000000010010019000000cd00000c13d0000000d01000029000000400010043f000007630020009c00000a760000213d000000200020008c00000a760000413d0000000c01000029000000000101043300010fee0000003d00001b370000013d00000a760000c13d000000000001004b00000c7d0000613d000000cf01000039000000000101041a00000e480000013d0008075c0010019b00000000050000190000000d0050006c000010a30000813d0000000b0050006c000010490000813d00000006015002100000000a0610002900000020016000390000000202000367000000000112034f000000000101043b0000ffff0010008c00000a760000213d000027110010008c000008a50000813d00000005035002100000000c07300029000000000372034f000000000303043b000000400800043d0000078704000041000000000048043500000004048000390000000000340435000000000262034f000000000202043b0000075c0020009c00000a760000213d000000640380003900000008040000290000000000430435000000440380003900000000001304350000002401800039000000000021043500000000010004140000000902000029000000040020008c000f00000005001d000e00000006001d0000102a0000613d00000009020000290000000003080019000000840400003900000000050800190000002006000039000700000007001d000600000008001d1c8112660000040f00000006080000290000000707000029000000000001004b00000bc60000613d0001102c0000003d00001b570000013d00000000028300190001102f0000003d00001b490000013d000007620020009c00000cd00000213d000000010030019000000cd00000c13d000000400020043f000007630010009c00000a760000213d000000200010008c00000a760000413d00000000010804330001103b0000003d00001b370000013d00000a760000c13d000000000001004b00000caf0000613d0000000201700367000000000101043b000000000010043f000000ce01000039000110440000003d00001b520000013d0000000e020000291c81143a0000040f0000000f05000029000000010550003900000ff60000013d0000079401000041000000000010043f000000320100003900000cd30000013d0000000d0000006b000010530000613d000000010100008a0000000d011000fa0000000c0010006c00000d4b0000413d0000000c020000290000000d012000b9000000010200008a000000000221013f000000090020006b00000d4b0000213d0000010802000039000000000202041a000000000002004b000010600000613d0000000901100029000000000012004b000001270000413d000110620000003d00001c1c0000013d000010da0000c13d000110650000003d00001c110000013d0000000b020000290000047d0000013d000000cb01000039000000000101041a00000af60000013d000000a003600039000000400030043f000000800260003900000000000204350000000f050000290000000007020019000000090050008c0000000a2550011a000000f803200210000000010270008a00000000040204330000077e04400197000000000343019f0000077f033001c700000000003204350000106f0000213d00000000027600490000008102200039000000210370008a000f00000003001d0000000000230435000000400200043d000e00000002001d00000020022000391c8117740000040f0000078002000041000000000021043500000001021000390000000f010000291c8117740000040f0000000e040000290000000001410049000000200210008a00000000002404350001108e0000003d00001c470000013d000007620010009c00000cd00000213d000000010020019000000cd00000c13d000000400010043f000000000204001900000e290000013d000000020100036700000b6e0000013d0000000001000414000000040020008c0000109f0000613d0000000f0500002900000000045300490001109e0000003d00001c0c0000013d00000bc60000613d000110a10000003d00001b570000013d0000000f0200002900000da20000013d000000400100043d000f00000001001d0000000c020000290000000d030000290000000a040000290000000b050000291c81144f0000040f0000000f020000290000000001210049000007210010009c00000721010080410000006001100210000007210020009c00000721020080410000004002200210000000000121019f0000000002000414000007210020009c0000072102008041000000c002200210000000000121019f0000076b011001c70000800d02000039000000010300003900000789040000411c811b1d0000040f000000010020019000000a760000613d000000d001000039000000000101041a000f00000001001d0016075c0010019b000110c50000003d00001b450000013d000000160440008a000000050440021000000760020000411c811aff0000040f000000000001004b00000a760000613d0000000f01000029000e075c0010019b000000400200043d0000078a010000410000000000120435000f00000002001d00000004012000390000000c020000290000000d030000290000000a040000290000000b050000291c81144f0000040f0000000004010019000000000100041400000e0d0000013d0000000001010433000f00000001001d0023075c0010019b000110df0000003d00001b450000013d000000230440008a000000050440021000000760020000411c811aff0000040f000000000001004b00000a760000613d0000000f010000290000075c02100197000000400400043d00000044014000390000006003000039000000000031043500000024014000390000000c0300002900000000003104350000076101000041000110f10000003d00001bb10000013d00000020033000390000000d0040006c000011530000813d0000000e05100360000000000505043b0000075c0050009c00000a760000213d000000000053043500000001044000390000000e05000029000e00200050003d000010f10000013d0000000f0000006b0000000003000019000011030000613d0000000e030000290000000203300367000000000303043b000111050000003d00001ba50000013d0000110b0000013d0000000f0050006c0000110a0000813d0001110a0000003d00001beb0000013d00000001011001bf000007bf02000041000000000012041b0000079201000041000111100000003d00001b6d0000013d000007f60000c13d0000000d020000291c8114f10000040f0000000d02000029000000200020008c00000001012002100000000302200210000011290000413d0000079203000041000000000030043f000000200300008a0000000d0530017f0000079303000041000000020400036700000000060000190000000c08000029000000000056004b0000000007860019000011320000813d000000000774034f000000000707043b000000000073041b00000020066000390000000103300039000011200000013d0000000d0000006b00000000030000190000112f0000613d0000000c030000290000000203300367000000000303043b000111310000003d00001ba50000013d000011370000013d0000000d0050006c000011360000813d000111360000003d00001beb0000013d00000001011001bf0000079202000041000000000012041b00000107010000390001113c0000003d00001b6d0000013d000007f60000c13d0000000b020000291c8114df0000040f0000000b01000029000000200010008c000011650000413d0000010701000039000000000010043f000000200100008a0000000b0310017f0000077101000041000000020200036700000000040000190000000a06000029000000000034004b0000000005640019000011740000813d000000000552034f000000000505043b000000000051041b000000200440003900000001011000390000114a0000013d0000000001000414000000040020008c0000115b0000613d0000000f0500002900000000045300490001115a0000003d00001c0c0000013d00000bc60000613d0001115d0000003d00001b3c0000013d000007620020009c00000cd00000213d000000010030019000000cd00000c13d000000400020043f000007630010009c00000a760000213d000010630000013d0000000b0000006b00000000010000190000116b0000613d0000000a010000290000000201100367000000000101043b0000000b040000290000000302400210000000010300008a000000000223022f000000000232013f000000000121016f0000000102400210000000000121019f000011830000013d0000000b0030006c000011800000813d0000000b030000290000000303300210000000f80330018f000000010400008a000000000334022f000000000343013f000000000252034f000000000202043b000000000232016f000000000021041b0000000b01000029000000010110021000000001011001bf0000010702000039000000000012041b000000d001000039000000000101041a000700000001001d0024075c0010019b0001118b0000003d00001b450000013d000000240440008a000000050440021000000760020000411c811aff0000040f000000000001004b00000a760000613d000000400400043d000007c101000041000000000014043500000064014000390000000f030000290000000000310435000000040140003900000060020000390000000000210435000000200200008a00000000052301700000001f0630018f000800000004001d0000008403400039000000000453001900000002020003670000000e07200360000011a90000613d000000000807034f0000000009030019000000008a08043c0000000009a90436000000000049004b000011a50000c13d000000000006004b000011af0000613d000000000557034f0000000306600210000111af0000003d00001bd00000013d0000000f05000029000000000453001900000000000404350000001f04500039000000200600008a000000000464016f000000000343001900000000041300490000000805000029000000240550003900000000004504350000000c052003600000000d04000029000000000343043600000000066401700000001f0740018f0000000004630019000011c70000613d000000000805034f0000000009030019000000008a08043c0000000009a90436000000000049004b000011c30000c13d000000000007004b000011cd0000613d000000000565034f0000000306700210000111cd0000003d00001bd00000013d0000000d05000029000000000453001900000000000404350000001f04500039000000200500008a000000000454016f000000000343001900000000011300490000000804000029000000440440003900000000001404350000000a042003600000000b01000029000000000313043600000000055101700000001f0610018f0000000001530019000011e50000613d000000000204034f0000000007030019000000002802043c0000000007870436000000000017004b000011e10000c13d00000007020000290000075c02200197000000000006004b000011f40000613d000000000454034f0000000305600210000000000601043300000000065601cf000000000656022f000000000404043b0000010005500089000000000454022f00000000045401cf000000000464019f00000000004104350000000b0130002900000000000104350000000001000414000000040020008c000012030000613d0000000b040000290000001f04400039000000200500008a000000000454016f000000080500002900000000045400490000000004340019000112020000003d00001c0c0000013d00000bc60000613d000112050000003d00001b570000013d0000000802300029000112080000003d00001b490000013d000007620020009c00000cd00000213d000000010030019000000cd00000c13d000000400020043f000007630010009c00000a760000213d00000c4d0000013d0000000001000410001c00000001001d000112140000003d00001b450000013d0000001c0440008a000000050440021000000760020000411c811aff0000040f0000000202000029000000ff0220018f000000010220015f00000000001201a0000000000100003900000001010060391c8114860000040f000001000100008a000000000200041a000000000112016f00000001011001bf000000000010041b000000040000006b0000123d0000c13d0000ffff0200008a000000000121016f00000100011001bf000000000010041b0001122c0000003d00001b800000013d0001122e0000003d00001c570000013d000007210010009c000007210100804100000040011002100000000002000414000007210020009c0000072102008041000000c002200210000000000112019f0000077b011001c70000800d020000390000077c040000411c811b1d0000040f00000001002001900000123f0000c13d00000a760000013d0001123f0000003d00001b800000013d000000000100041500000003020000290000047d0000013d0003000000000002000300000006001d000200000005001d000007210030009c00000721030080410000004003300210000007210040009c00000721040080410000006004400210000000000334019f000007210010009c0000072101008041000000c001100210000000000113019f1c811b1d0000040f000000020900002900000060031002700000072103300197000000030030006c000000030400002900000000040340190000001f0540018f000007ce064001980000000004690019000012600000613d000000000701034f000000007807043c0000000009890436000000000049004b0000125c0000c13d000000010220018f000000000005004b000012650000613d000112650000003d00001b990000013d00001c330000013d0003000000000002000300000006001d000200000005001d000007210030009c00000721030080410000004003300210000007210040009c00000721040080410000006004400210000000000334019f000007210010009c0000072101008041000000c001100210000000000113019f1c811b220000040f000000020900002900000060031002700000072103300197000000030030006c000000030400002900000000040340190000001f0540018f000007ce064001980000000004690019000012840000613d000000000701034f000000007807043c0000000009890436000000000049004b000012800000c13d000000010220018f000000000005004b000012890000613d000112890000003d00001b990000013d00001c330000013d0000002004100039000000000301043300000000013204360000000002000019000000000032004b000012960000813d000000000512001900000000062400190000000006060433000000000065043500000020022000390000128e0000013d000012990000a13d000000000213001900000000000204350000001f02300039000000200300008a000000000232016f0000000001210019000000000001042d00000000030200190000002002000039000000000221043600000000010300191c81128a0000040f000000000001042d000007630010009c000012ae0000213d000000230010008c000012ae0000a13d00000004010000390000000201100367000000000101043b0000075c0010009c000012ae0000213d000000000001042d00001b770000013d000007630010009c000012bf0000213d000000630010008c000012bf0000a13d00000002030003670000000401300370000000000101043b0000075c0010009c000012bf0000213d0000002402300370000000000202043b0000075c0020009c000012bf0000213d0000004403300370000000000303043b000000000001042d00001b770000013d000007cf0020009c000012f10000813d00000000040100190000001f01200039000000200600008a000000000161016f0000003f01100039000000000561016f000000400100043d0000000005510019000000000015004b00000000070000390000000107004039000007620050009c000012f10000213d0000000100700190000012f10000c13d000000400050043f00000000052104360000000007420019000000000037004b000012f70000213d00000000066201700000001f0720018f00000002044003670000000003650019000012e10000613d000000000804034f0000000009050019000000008a08043c0000000009a90436000000000039004b000012dd0000c13d000000000007004b000012ee0000613d000000000464034f0000000306700210000000000703043300000000076701cf000000000767022f000000000404043b0000010006600089000000000464022f00000000046401cf000000000474019f000000000043043500000000022500190000000000020435000000000001042d0000079401000041000000000010043f0000004101000039000000040010043f000007950100004100001c830001043000001b770000013d0002000000000002000007630010009c000013100000213d0000000003010019000000430010008c000013100000a13d00000002010003670000000402100370000000000202043b000200000002001d0000075c0020009c000013100000213d0000002402100370000000000402043b000007620040009c000013100000213d0000002302400039000000000032004b000013100000813d0001130d0000003d00001bfd0000013d00000000020100190000000201000029000000000001042d00001b770000013d000007630010009c000013290000213d000000230010008c000013290000a13d00000002020003670000000403200370000000000303043b000007620030009c000013290000213d0000002304300039000000000014004b000013290000813d0000000404300039000000000242034f000000000202043b000007620020009c000013290000213d000000240330003900000005042002100000000004340019000000000014004b000013290000213d0000000001030019000000000001042d00001b770000013d000000d505000039000000000405041a000000010640019000000001024002700000007f0220618f0000001f0020008c00000000030000390000000103002039000000000036004b000013470000c13d0000000003210436000000000006004b000013420000613d000000000050043f0000079c0400004100000000050000190000000001530019000000000025004b000013460000813d000000000604041a0000000000610435000000200550003900000001044000390000133a0000013d000001000200008a000000000224016f00000000002304350000004001100039000000000001042d0000079401000041000000000010043f0000002201000039000000040010043f000007950100004100001c8300010430000007630010009c000013760000213d000000430010008c000013760000a13d00000002030003670000000402300370000000000402043b000007620040009c000013760000213d0000002302400039000000000012004b000013760000813d0000000402400039000000000223034f000000000202043b000007620020009c000013760000213d000000240540003900000005042002100000000004540019000000000014004b000013760000213d0000002404300370000000000604043b000007620060009c000013760000213d0000002304600039000000000014004b000013760000813d0000000404600039000000000343034f000000000403043b000007620040009c000013760000213d000000240360003900000005064002100000000006360019000000000016004b000013760000213d0000000001050019000000000001042d00001b770000013d0000003301000039000000000101041a0000006502000039000000000202041a0000075c032001970000000002000411000000000032004b000013840000c13d000000140200008a00000000022000310000000202200367000000000202043b0000006002200270000000000112013f0000075c00100198000013880000c13d000000000001042d000000400100043d0000004402100039000007d0030000410000000000320435000007660200004100000000002104350000002402100039000000200300003900000000003204350000000402100039000000000032043500000064020000391c811ad90000040f00000000060100190000075c011001970000003302000039000000000302041a0000076804300197000000000114019f000000000012041b000000400100043d000007210010009c000007210100804100000040011002100000000002000414000007210020009c0000072102008041000000c002200210000000000112019f0000075c053001970000076b011001c70000800d02000039000000030300003900000799040000411c811b1d0000040f0000000100200190000013ae0000613d000000000001042d00001b770000013d00020000000000020000000001000412000200000001001d000100000000003d000080050100003900000044030000390000000004000415000000020440008a0000000504400210000007a0020000411c811aff0000040f0000076602000041000000400300043d0000000000230435000000040230003900000020040000390000000000420435000000440230003900000024043000390000075c011001970000000005000410000000000015004b000013cd0000c13d0000002c010000390000000000140435000007d10100004100000000001204350000006401300039000007d402000041000013de0000013d000007a305000041000000000505041a0000075c05500197000000000015004b000013d80000c13d0000000f010000390000000000140435000007d30100004100000000001204350000006402000039000013e00000013d0000002c010000390000000000140435000007d10100004100000000001204350000006401300039000007d2020000410000000000210435000000840200003900000000010300191c811ad90000040f0003000000000002000300000001001d000000000010043f000000ca01000039000000200010043f000113e90000003d00001b610000013d0000000001000019000013ec0000613d000000000001042d000000ca050000390000000304000029000000c901000039000000000201041a000007620020009c000014010000213d0000000103200039000000000031041b000007d50220009a000000000042041b000000000101041a000200000001001d000000000040043f000000200050043f000000400200003900000000010000191c811aeb0000040f0000000202000029000000000021041b0000000101000039000000000001042d0000079401000041000000000010043f0000004101000039000000040010043f000007950100004100001c830001043000000000060100190000004001000039000000000816043600000040016000390000000000310435000007d60030009c000014290000813d0000000207000367000000600960003900000005013002120000000001190019000014180000613d000000000227034f000000002302043c0000000009390436000000000019004b000014140000c13d000000010000008b00000000026100490000000000280435000000000051043500000000020000190000002001100039000000000052004b000014280000813d000000000347034f000000000303043b0000075c0030009c000014290000213d0000000000310435000000010220003900000020044000390000141d0000013d000000000001042d00001b770000013d0000009701000039000000000201041a000000020020008c000014310000613d0000000202000039000000000021041b000000000001042d000000400100043d0000004402100039000007d703000041000000000032043500000024021000390000001f030000390000000000320435000007660200004100001bda0000013d0000000203000367000000000423034f000000000404043b0000075e0040009c0000144e0000813d000000000501041a0000076805500197000000000445019f000000000041041b0000002002200039000000000223034f000000000202043b0000ffff0020008c0000144e0000213d000007d803400197000000a0022002100000076a02200197000000000232019f000000000021041b000000000001042d00001b770000013d0000004006000039000000000761043600000040061000390000000000360435000007d60030009c000014760000813d00000002060003670000006008100039000000050330021200000000033800190000145f0000613d000000000226034f000000002902043c0000000008980436000000000038004b0000145b0000c13d000000010000008b0000000001130049000000000017043500000000015304360000000002000019000000000052004b000014750000813d000000000346034f000000000303043b0000075c0030009c000014760000213d00000000033104360000002007400039000000000776034f000000000707043b0000ffff0070008c000014760000213d0000000000730435000000010220003900000040044000390000004001100039000014640000013d000000000001042d00001b770000013d0002000000000002000000cb02000039000000000202041a000200000002001d000000000010043f000000cc010000390001147f0000003d00001b4d0000013d00000002020000290000075c02200197000000000101041a0000075c01100198000000000201c0190000000001020019000000000001042d000000000001004b000014890000613d000000000001042d000000400100043d0000006402100039000007d90300004100000000003204350000004402100039000007da03000041000000000032043500000024021000390000002e030000390000000000320435000007660200004100001c210000013d0000001f03100039000000000023004b0000000004000019000007780400404100000778052001970000077803300197000000000653013f000000000053004b00000000030000190000077803002041000007780060009c000000000304c019000000000003004b000014cc0000613d0000000043010434000007cf0030009c000014c60000813d0000001f01300039000000200500008a000000000151016f0000003f01100039000000000551016f000000400100043d0000000005510019000000000015004b00000000060000390000000106004039000007620050009c000014c60000213d0000000100600190000014c60000c13d000000400050043f00000000053104360000000006340019000000000026004b000014cc0000213d0000000002000019000000000032004b000014c20000813d00000000065200190000000007240019000000000707043300000000007604350000002002200039000014ba0000013d000014c50000a13d00000000023500190000000000020435000000000001042d0000079401000041000000000010043f0000004101000039000000040010043f000007950100004100001c830001043000001b770000013d000000200010008c000014de0000413d000007bf03000041000000000030043f0000001f032000390000000503300270000007db0330009a000000200020008c000007c0030040410000001f011000390000000501100270000007db0110009a000000000013004b000014de0000813d000000000003041b0000000103300039000014d90000013d000000000001042d000000200010008c000014f00000413d0000010703000039000000000030043f0000001f032000390000000503300270000007dc0330009a000000200020008c00000771030040410000001f011000390000000501100270000007dc0110009a000000000013004b000014f00000813d000000000003041b0000000103300039000014eb0000013d000000000001042d000000200010008c000015020000413d0000079203000041000000000030043f0000001f032000390000000503300270000007dd0330009a000000200020008c00000793030040410000001f011000390000000501100270000007dd0110009a000000000013004b000015020000813d000000000003041b0000000103300039000014fd0000013d000000000001042d000f000000000002000c0000000b001d00040000000a001d000300000009001d000500000008001d000800000007001d000a00000006001d000b00000005001d000700000004001d000e00000003001d000900000002001d000600000001001d000000000100041a0000000801100270000000ff0110018f1c8116650000040f000000000100041a0000000801100270000000ff0110018f000d00000001001d1c8116650000040f0000000e010000290000002001100039000200000001001d00000000010104330000ffff0110018f000027110010008c000016500000813d0000000d010000291c8116650000040f0000000d010000291c8116650000040f0000006501000039000000000101041a0000075c021001970000000001000411000000000021004b0000152b0000c13d0001152b0000003d00001b310000013d1c8113950000040f000000000100041a0000000801100270000000ff0110018f000d00000001001d1c8116650000040f0000000d010000291c8116650000040f00000001020000390000009701000039000000000021041b00000006010000291c8113950000040f0000000e0100002900000000010104330000075c01100198000015460000613d000000cd02000039000000000302041a000007690330019700000002040000290000000004040433000000a0044002100000076a04400197000000000343019f000000000113019f000000000012041b00000007010000290000075c011001980000154e0000613d000000cb02000039000000000302041a0000076803300197000000000113019f000000000012041b000000000100041a0000000801100270000000ff0110018f1c8116650000040f00000008010000290000075c011001970000006503000039000000000203041a0000076802200197000000000112019f000000000013041b000000000100041a0000000801100270000000ff0110018f000e00000001001d1c8116650000040f0000000e010000291c8116650000040f0000000b010000290000000012010434000007cf0020009c000016450000813d000d00000001001d000007bf01000041000000000101041a000000010410019000000001011002700000007f0110618f0000001f0010008c00000000030000390000000103002039000000000034004b000016490000c13d000e00000002001d1c8114cd0000040f0000000e0a0000290000001f00a0008c000000200400008a0000000101a002100000000302a00210000015850000a13d000007bf03000041000000000030043f00000000044a016f0000002005000039000007c00300004100000000060000190000000a080000290000000b09000029000000000046004b00000000079500190000158a0000813d000115840000003d00001c160000013d0000157f0000013d00000000000a004b000015900000613d0000000d030000290000000003030433000015910000013d0000000000a4004b0000158e0000813d0001158e0000003d00001c3f0000013d00000001011001bf000015940000013d00000000030000190000000a08000029000115940000003d00001ba50000013d000007bf02000041000000000012041b0000000012080434000007620020009c000016450000213d000d00000001001d00000792010000410001159d0000003d00001bbc0000013d000016490000c13d000e00000002001d1c8114f10000040f0000000e09000029000000200090008c00000001019002100000000302900210000015b30000413d0000079203000041000000000030043f000000200300008a000000000439016f0000002005000039000007930300004100000000060000190000000a08000029000000000046004b0000000007850019000015b80000813d000115b20000003d00001c160000013d000015ad0000013d000000000009004b000015be0000613d0000000d030000290000000003030433000015bf0000013d000000000094004b000015bc0000813d000115bc0000003d00001c3f0000013d00000001011001bf000015c10000013d0000000003000019000115c10000003d00001ba50000013d0000079202000041000000000012041b0000075f010000410000000102000039000000000021041b00000005010000290000075c011001971c8113e20000040f00000009010000290000000012010434000007620020009c000016450000213d000d00000001001d0000010701000039000115d10000003d00001bbc0000013d000016490000c13d000e00000002001d1c8114df0000040f0000000e09000029000000200090008c000015e90000413d00000020020000390000010708000039000000000080043f000000200100008a000000000319016f000007710100004100000000040000190000000c060000290000000907000029000000000034004b0000000005720019000015f00000813d0000000005050433000000000051041b000000200440003900000020022000390000000101100039000015e00000013d000000000009004b0000000c060000290000010708000039000015fd0000613d0000000d010000290000000001010433000015fe0000013d000000000093004b000015fa0000813d0000000302900210000000f80220018f000000010300008a000000000223022f000000000232013f0000000003050433000000000223016f000000000021041b000000010190021000000001011001bf000016050000013d00000000010000190000000302900210000000010300008a000000000223022f000000000232013f000000000121016f0000000102900210000000000121019f000000000018041b000f075c0060019b000116090000003d00001b450000013d0000000f0440008a000000050440021000000760020000411c811aff0000040f000000000001004b0000164f0000613d0000000c010000290000075c02100197000000400500043d000007de01000041000000000015043500000000010004100000075c01100197000000040350003900000000001304350000000001000414000000040020008c000016260000613d00000024040000390000000003050019000d00000002001d000e00000005001d0000000e0500002900000000060000191c8112420000040f0000000e050000290000000d02000029000000000001004b000016540000613d00000001010000310000001f04100039000000200300008a000000000334016f0000000004530019000000000034004b00000000030000390000000103004039000007620040009c000016450000213d0000000100300190000016450000c13d000000400040043f000007630010009c0000164f0000213d000000d001000039000000000301041a0000076803300197000000000223019f000000000021041b00000003010000290000000002010433000000000002004b0000163f0000613d1c8116f70000040f0000000402000029000000000002004b000016440000613d0000010801000039000000000021041b000000000001042d0000079401000041000000000010043f00000041010000390000164c0000013d0000079401000041000000000010043f0000002201000039000000040010043f000007950100004100001c830001043000001b770000013d0000078b01000041000000000010043f000007750100004100001c830001043000000003040003670000000102000031000116580000003d00001b670000013d0000165e0000613d000000000704034f00000000080100190001165d0000003d00001c2f0000013d0000165b0000c13d000000000006004b000016640000613d000000000454034f0000000305600210000116640000003d00001b270000013d1c811ad90000040f000000000001004b000016680000613d000000000001042d000000400100043d0000006402100039000007df0300004100000000003204350000004402100039000007e003000041000000000032043500000024021000390000002b030000390000000000320435000007660200004100001c210000013d000000400100043d000007e10010009c000016830000813d0000004002100039000000400020043f0000010902000039000000000202041a0000075d002001980000000003000039000000010300c039000000200410003900000000003404350000075c022001970000000000210435000000000001042d0000079401000041000000000010043f0000004101000039000000040010043f000007950100004100001c83000104300003000000000002000300000005001d0000004005000039000000000751043600000040051000390000000000350435000007d60030009c000016f60000813d00000002060003670000006005100039000000050330021200000000033500190000169b0000613d000000000226034f000000002802043c0000000005850436000000000035004b000016970000c13d000000010000008b0000000001130049000000000017043500000003010000290000000000130435000000050110021000000000011300190000002001100039000100000000003500000000024000790000001f0220008a000200000002001d0000077808200197000000000a030019000000000b040019000000000c0000190000000300c0006c000016f50000813d000000200aa000390000000002310049000000200220008a00000000002a04350000000002b6034f000000000902043b0000077802900197000000000582013f000000000082004b00000000020000190000077802002041000000020090006c000000000d000019000007780d004041000007780050009c00000000020dc019000000000002004b000016f60000613d0000000009490019000000000296034f000000000d02043b0000076200d0009c000016f60000213d0000000102d00069000000000042004b00000000050000190000077805004041000007780e4001970000077802200197000000000fe2013f0000000000e2004b000000000200001900000778020020410000077800f0009c000000000205c019000000000002004b000016f60000c13d0000002002900039000000000526034f0000000002d10436000000200100008a00000000091d0170000000000e920019000016df0000613d000000000f05034f000000000102001900000000f70f043c00000000017104360000000000e1004b000016db0000c13d0000001f01d00190000016ec0000613d000000000595034f000000030110021000000000070e043300000000071701cf000000000717022f000000000505043b0000010001100089000000000515022f00000000011501cf000000000171019f00000000001e04350000001f01d00039000000200500008a000000000151016f00000000052d001900000000000504350000000001210019000000010cc00039000000200bb00039000016ab0000013d000000000001042d00001b770000013d0004000000000002000300000001001d000000400200043d000400000002001d00000040010000390000000001120436000200000001001d00000040012000391c81132a0000040f0000000002010019000000040120006a0000000203000029000000000013043500000003010000291c81128a0000040f00000004020000290000000001210049000007210020009c00000721020080410000004002200210000007210010009c00000721010080410000006001100210000000000121019f0000000002000414000007210020009c0000072102008041000000c002200210000000000121019f0000076b011001c70000800d020000390000000103000039000007e2040000411c811b1d0000040f0000000100200190000017620000613d00000003080000290000000032080434000007cf0020009c000017630000813d000000d501000039000000000401041a000000010540019000000001044002700000007f0440618f0000001f0040008c00000000060000390000000106002039000000000065004b000017670000c13d000000200040008c000017390000413d000000000010043f0000001f052000390000000505500270000007e30550009a000000200020008c0000079c050040410000001f044000390000000504400270000007e30440009a000000000045004b000017390000813d000000000005041b0000000105500039000017340000013d0000001f0020008c000017470000a13d000000000010043f000000200300008a000000000432016f00000020050000390000079c030000410000000006000019000000000046004b00000000078500190000174b0000813d000117460000003d00001c160000013d000017410000013d000000000002004b000017580000613d0000000003030433000017590000013d000000000024004b000017550000813d0000000304200210000000f80440018f000000010500008a000000000445022f000000000454013f0000000005070433000000000445016f000000000043041b000000010220021000000001022001bf000017600000013d00000000030000190000000304200210000000010500008a000000000445022f000000000454013f000000000343016f0000000102200210000000000223019f000000000021041b000000000001042d00001b770000013d0000079401000041000000000010043f00000041010000390000176a0000013d0000079401000041000000000010043f0000002201000039000000040010043f000007950100004100001c83000104300000075c01100197000000000010043f0000078f0100004100001c5f0000013d0000075c02200197000000000020043f00001c5f0000013d000000200310003900000000010104330000000004000019000000000014004b0000177f0000813d00000000052400190000000006430019000000000606043300000000006504350000002004400039000017770000013d0000000001210019000017820000a13d0000000000010435000000000001042d0002000000000002000000000001004b0000179f0000613d000000000010043f000007b702000041000000200020043f000200000001001d0001178c0000003d00001b610000013d0000179c0000c13d0000075f01000041000000000101041a0000000202000029000000000021004b0000179f0000a13d000000010220008a000200000002001d000000000020043f000007b701000041000117980000003d00001b520000013d0000000202000029000000000101041a000000000001004b000017920000613d000007e4001001980000179f0000c13d000000000001042d000007e501000041000000000010043f000007750100004100001c83000104300001000000000002000100000001001d1c8117b60000040f000000000001004b000017b20000613d0000000101000029000000000010043f000007b401000041000000200010043f000000400200003900000000010000191c811aeb0000040f000000000101041a0000075c01100197000000000001042d000007e601000041000000000010043f000007750100004100001c83000104300002000000000002000000000001004b000017d00000613d00000000020100190000075f01000041000000000101041a000000000021004b0000000001000019000017d50000a13d000200000002001d000000000020043f000007b701000041000117c40000003d00001be50000013d000000000001004b000017d20000c13d0000000202000029000000000002004b000000010220008a000017bf0000c13d0000079401000041000000000010043f0000001101000039000000040010043f000007950100004100001c83000104300000000001000019000000000001042d000007e40010019800000000010000390000000101006039000000000001042d000c000000000002000400000002001d000800000001001d000900000003001d00000000010300191c8117830000040f00000008020000290000075c02200197000600000001001d0000075c01100197000700000002001d000000000021004b000018fa0000c13d0000000901000029000000000010043f000007b401000041000117e80000003d00001b4d0000013d0000000005010019000000000401041a00000000010004110000075c011001970000000703000029000000000031004b000500000001001d000018040000613d000000000041004b000018040000613d000000000030043f0000078f01000041000000200010043f00000040020000390000000001000019000800000005001d000300000004001d1c811aeb0000040f0000000502000029000000000020043f000117fe0000003d00001b520000013d000000030400002900000008050000290000000703000029000000000101041a000000ff00100190000018fe0000613d000000000004004b000018070000613d000000000005041b000000000030043f0000079a010000410001180b0000003d00001b930000013d000000010220008a000000000021041b00000004010000290000075c01100197000800000001001d000000000010043f0000079a01000041000118140000003d00001b8d0000013d0000000102200039000000000021041b0000800b01000039000000040300003900000000040004150000000c0440008a0000000504400210000007b6020000411c811aff0000040f000400000001001d0000000901000029000000000010043f000007b701000041000118230000003d00001b520000013d0000000402000029000000a00220021000000008022001af000007b9022001c7000000000021041b0000000601000029000007b9001001980000183a0000c13d00000009010000290000000101100039000400000001001d000000000010043f000007b701000041000118320000003d00001b930000013d000000000002004b0000183a0000c13d0000075f02000041000000000202041a000000040020006b0000183a0000613d0000000602000029000000000021041b0000000001000414000007210010009c0000072101008041000000c0011002100000076b011001c70000800d020000390000000403000039000007ba040000410000000705000029000000080600002900000009070000291c811b1d0000040f0000000100200190000018f30000613d000000080000006b000018fc0000613d00000009010000291c8114770000040f000000000001004b000018c00000613d0000000007010019000000400300043d000007a60100004100000000001304350000000401300039000007e90200004100000000002104350000000001000414000000040070008c000600000007001d000018630000613d0000002404000039000000200600003900000000020700190000000005030019000400000003001d1c8112660000040f00000004030000290000000607000029000000000001004b0000190c0000613d00000001060000310000001f01600039000000200200008a000000000521016f0000000004350019000000000054004b00000000010000390000000101004039000007620040009c000018f40000213d0000000100100190000018f40000c13d000000400040043f000007630060009c000018f30000213d000000200060008c000018f30000413d0000000001030433000118770000003d00001b370000013d000018f30000c13d000000000001004b000018c00000613d000200000005001d000400000004001d000300000006001d0000006501000039000000000101041a0000075c011001970000000002000411000000000012004b000018860000c13d000118850000003d00001be00000013d0005006000100278000b00000007001d000118890000003d00001b450000013d0000000b0440008a000000050440021000000760020000411c811aff0000040f0000000606000029000000000001004b000000030500002900000004030000290000000204000029000018f30000613d0000008401300039000000a0020000390000000000210435000000640130003900000009020000290000000000210435000000440130003900000008020000290000000000210435000000240130003900000007020000290000000000210435000007ea010000410000000000130435000000040130003900000005020000290000000000210435000000a40130003900000000000104350000000001000414000000040060008c000018b50000613d000000c4040000390000000002060019000000000503001900000000060000191c8112420000040f0000000105000031000000000001004b0000191d0000613d0000001f01500039000000200200008a000000000421016f00000004030000290000000001340019000000000041004b00000000020000390000000102004039000007620010009c000018f40000213d0000000100200190000018f40000c13d000000400010043f000007630050009c000018f30000213d000000d001000039000000000101041a000600000001001d000a075c0010019b0000800201000039000000240300003900000000040004150000000a0440008a000000050440021000000760020000411c811aff0000040f000000000001004b000018f30000613d00000006010000290000075c02100197000000400500043d000000440150003900000009030000290000000000310435000000240150003900000008030000290000000000310435000007bc0100004100000000001504350000000401500039000000070300002900000000003104350000000001000414000000040020008c000018e00000c13d0000000102000031000018e90000013d00000064040000390000000003050019000900000005001d00000000060000191c8112420000040f0000000102000031000000000001004b000019020000613d0000000905000029000118eb0000003d00001c370000013d000007620010009c000018f40000213d0000000100300190000018f40000c13d000000400010043f000007630020009c000018f30000213d000000000001042d00001b770000013d0000079401000041000000000010043f0000004101000039000000040010043f000007950100004100001c8300010430000007e701000041000018ff0000013d000007eb01000041000018ff0000013d000007e801000041000000000010043f000007750100004100001c83000104300000000304000367000119050000003d00001b670000013d000019160000613d000000000704034f00000000080100190001190a0000003d00001c2f0000013d000019080000c13d000019160000013d00000003040003670000000102000031000119100000003d00001b670000013d000019160000613d000000000704034f0000000008010019000119150000003d00001c2f0000013d000019130000c13d000000000006004b0000191c0000613d000000000454034f00000003056002100001191c0000003d00001b270000013d1c811ad90000040f0000000303000367000000200100008a000000000415017000000000020500190000001f0550018f000000400100043d00000000094100190000192b0000613d000000000603034f0000000007010019000000006806043c0000000007870436000000000097004b000019270000c13d000000000005004b0000191c0000613d000000000343034f00000003045002100000000005090433000119320000003d00001c4f0000013d00000000003904351c811ad90000040f000a000000000002000600000004001d000700000002001d000400000001001d000500000003001d1c8117d60000040f000a00070000002d0000800201000039000000240300003900000000040004150000000a0440008a000000050440021000000760020000411c811aff0000040f000000000001004b000019880000613d0000000001000415000300000001001d000000400300043d00000064013000390000008002000039000200000002001d000000000021043500000044013000390000000502000029000000000021043500000004010000290000075c0110019700000024023000390000000000120435000007ec010000410000000000130435000000040130003900000000020004110000000000210435000500000003001d000000840230003900000006010000291c81128a0000040f0000000004000415000000090440008a00000005044002100000000003010019000000000100041400000007020000290000075c02200197000000040020008c0000196e0000613d00000005050000290000000004530049000000200600003900000000030500191c8112420000040f0000000004000415000000080440008a0000000504400210000000000001004b0000198a0000613d000119700000003d00001b570000013d00000005050000290000000002530019000119740000003d00001b490000013d000007620020009c000019bb0000213d0000000100300190000019bb0000c13d000000400020043f000007630010009c000019890000213d0000001f0010008c000019890000a13d0000000001050433000007ca00100198000019890000c13d0000000502400270000000000201001f000000000200041500000003022000690000000002000002000007cb01100197000007ec0010009c000019900000c13d000000000001042d00001b770000013d0000000102000032000019940000c13d00000060010000390000000002010433000000000002004b000019c10000c13d000007ed01000041000000000010043f000007750100004100001c8300010430000007620020009c000019bb0000213d0000001f01200039000000200300008a000000000131016f0000003f01100039000000000431016f000000400100043d0000000004410019000000000014004b00000000050000390000000105004039000007620040009c000019bb0000213d0000000100500190000019bb0000c13d000000400040043f000000000521043600000000033201700000001f0420018f000200000005001d00000000023500190000000305000367000019b20000613d000000000605034f0000000207000029000000006806043c0000000007870436000000000027004b000019ae0000c13d000000000004004b0000198d0000613d000000000335034f00000003044002100000000005020433000119b90000003d00001c4f0000013d00000000003204350000198d0000013d0000079401000041000000000010043f0000004101000039000000040010043f000007950100004100001c830001043000000002010000291c811ad90000040f0005000000000002000300000001001d0000075f01000041000000000101041a000400000001001d0000800b0100003900000004030000390000000004000415000000050440008a0000000504400210000007b6020000411c811aff0000040f000200000001001d0000000401000029000000000010043f000007b701000041000119d50000003d00001b4d0000013d00000003020000290000075c03200197000300000003001d0000000202000029000000a002200210000000000232019f000007b9022001c7000000000021041b000000000030043f0000079a01000041000119e10000003d00001b8d0000013d000007ee0220009a000000000021041b000000030000006b000019fd0000613d00000000010000190000000100100190000019f80000c13d0000000001000414000007210010009c0000072101008041000000c0011002100000076b011001c70000800d020000390000000403000039000007ba040000410000000005000019000000030600002900000004070000291c811b1d0000040f00000001002001900000000101000039000019e60000c13d00001b770000013d000000040100002900000001011000390000075f02000041000000000012041b000000000001042d000007ef01000041000000000010043f000007750100004100001c83000104300006000000000002000400000001001d000300000002001d000000000002004b00001a4e0000613d0000075f01000041000000000101041a000500000001001d0000800b0100003900000004030000390000000004000415000000060440008a0000000504400210000007b6020000411c811aff0000040f0000000502000029000000000020043f000007b702000041000000200020043f000200000001001d000000400200003900000000010000191c811aeb0000040f0000000202000029000000a0022002100000000303000029000000010030008c0000000003000019000007b903006041000000000223019f00000004030000290000075c03300197000400000003001d000000000232019f000000000021041b000000000030043f0000079a0100004100011a280000003d00001b520000013d0000000302000029000007f0022000d1000000000301041a0000000002230019000000000021041b000000040000006b00001a500000613d0000000301000029000300050010002d0000000001000019000000010010019000001a390000613d00000005070000290000000107700039000000030070006c00001a3a0000c13d00001a4a0000013d00000005070000290000000001000414000007210010009c0000072101008041000000c0011002100000076b011001c70000800d020000390000000403000039000007ba0400004100000000050000190000000406000029000500000007001d1c811b1d0000040f0000000100200190000000010100003900001a320000c13d00001b770000013d0000075f010000410000000302000029000000000021041b000000000001042d000007f10100004100001a510000013d000007ef01000041000000000010043f000007750100004100001c83000104300002000000000002000200000001001d1c811a870000040f000000000001004b00001a7e0000613d00000002010000291c811aaf0000040f000000000001004b000000000200001900001a7f0000c13d0000000205000029000000400100043d0000002003100039000007a60200004100000000002304350000002402100039000007f204000041000000000042043500000024020000390000000000210435000007f30010009c00001a810000813d0000006002100039000000400020043f000000040050008c00001a710000c13d0000000001030433000000000010043f00001a760000013d000000000205001900011a740000003d00001b790000013d00001a7e0000613d000000000100043d0000000102000031000000200020008c000000000200001900001a7f0000413d000000000001004b0000000002000039000000010200c03900001a7f0000013d0000000002000019000000010120018f000000000001042d0000079401000041000000000010043f0000004101000039000000040010043f000007950100004100001c830001043000010000000000020000000002010019000000400100043d0000002003100039000007a60400004100000000004304350000002405100039000000000045043500000024040000390000000000410435000007f30010009c00001aa90000813d0000006004100039000000400040043f000000040020008c00001a9a0000c13d0000000001030433000000000010043f00001a9e0000013d00011a9c0000003d00001b790000013d00001aa60000613d000000000100043d0000000102000031000000200020008c000000000200001900001aa70000413d000000000001004b0000000002000039000000010200c03900001aa70000013d0000000002000019000000010120018f000000000001042d0000079401000041000000000010043f0000004101000039000000040010043f000007950100004100001c830001043000010000000000020000000002010019000000400100043d0000002003100039000007a60400004100000000004304350000002404100039000007cb05000041000000000054043500000024040000390000000000410435000007f30010009c00001ad20000813d0000006004100039000000400040043f000000040020008c00001ac30000c13d0000000001030433000000000010043f00001ac70000013d00011ac50000003d00001b790000013d00001acf0000613d000000000100043d0000000102000031000000200020008c000000000200001900001ad00000413d000000000001004b0000000002000039000000010200c03900001ad00000013d0000000002000019000000010120018f000000000001042d0000079401000041000000000010043f0000004101000039000000040010043f000007950100004100001c8300010430000000000001042f000007210010009c00000721010080410000004001100210000007210020009c00000721020080410000006002200210000000000112019f00001c8300010430000007210010009c00000721010080410000004001100210000007210020009c00000721020080410000006002200210000000000112019f000000e002300210000000000121019f00001c820001042e000007210010009c00000721010080410000004001100210000007210020009c00000721020080410000006002200210000000000112019f0000000002000414000007210020009c0000072102008041000000c002200210000000000112019f0000076b011001c700008010020000391c811b220000040f000000010020019000001afe0000613d000000000101043b000000000001042d00001b770000013d00000000050100190000000000200443000000050030008c00001b0d0000413d000000040100003900000000020000190000000506200210000000000664001900000005066002700000000006060031000000000161043a0000000102200039000000000031004b00001b050000413d000007210030009c000007210300804100000060013002100000000002000414000007210020009c0000072102008041000000c002200210000000000112019f000007f4011001c700000000020500191c811b220000040f000000010020019000001b1c0000613d000000000101043b000000000001042d000000000001042f00001b20002104210000000102000039000000000001042d0000000002000019000000000001042d00001b25002104230000000102000039000000000001042d0000000002000019000000000001042d000000000603043300000000065601cf000000000656022f000000000404043b0000010005500089000000000454022f00000000045401cf000000000464019f0000000000430435000000010000013b000000140100008a00000000011000310000000201100367000000000101043b0000006001100270000000010000013b000000000001004b0000000002000039000000010200c039000000000021004b000000010000013b00000001010000310000001f02100039000000200300008a000000000332016f0000000f02300029000000000032004b00000000030000390000000103004039000000010000013b000080020100003900000024030000390000000004000415000000010000013b000000000032004b00000000030000390000000103004039000000010000013b000000200010043f000000400200003900000000010000191c811aeb0000040f000000010000013b000000200010043f000000000100001900000040020000391c811aeb0000040f000000010000013b00000001010000310000001f02100039000000200300008a000000000332016f000000010000013b000000000503001900000000060000191c8112420000040f000000000001004b000000010000013b000000400200003900000000010000191c811aeb0000040f000000000101041a000000000001004b000000010000013b000000200100008a00000000051201700000001f0620018f000000400100043d0000000003510019000000010000013b000000000201041a000000010020019000000001012002700000007f0110618f0000001f0010008c00000000030000390000000103002039000000000232013f0000000100200190000000010000013b000000000100001900001c830001043000000000040104330000753001000039000000200600003900000000050000191c8112660000040f000000000001004b000000010000013b0000000f010000290000000c020000290000000e030000290000000b040000290000000a050000290000000906000029000000080700002900000007080000290000000d09000029000000060a000029000000050b0000291c8115030000040f000000010000013b000000200010043f000000000100001900000040020000391c811aeb0000040f000000000201041a000000010000013b000000200010043f000000400200003900000000010000191c811aeb0000040f000000000201041a000000010000013b000000000661034f0000000305500210000000000704043300000000075701cf000000000757022f000000000606043b0000010005500089000000000656022f00000000055601cf000000000575019f0000000000540435000000010000013b000000010400008a000000000224022f000000000242013f000000000223016f000000000112019f000000010000013b000000000403041a000000a005200039000000000045043500000020022000390000000103300039000000010000013b000000000014043500000004014000390000000a030000290000000000310435000f00000004001d00000064034000390000000d01000029000000000013043500000002010003670000000004000019000000010000013b000000000401041a000000010040019000000001014002700000007f0110618f0000001f0010008c00000000030000390000000103002039000000000334013f0000000100300190000000010000013b00000001010000310000001f02100039000000200300008a000000000232016f0000000003520019000000000023004b00000000020000390000000102004039000e00000003001d000000010000013b000000000704043300000000076701cf000000000767022f000000000505043b0000010006600089000000000565022f00000000056501cf000000000575019f0000000000540435000000010000013b000000000021043500000004021000390000002003000039000000000032043500000064020000391c811ad90000040f000000140100008a00000000011000310000000201100367000000000101043b000000010000013b000000200010043f000000000100001900000040020000391c811aeb0000040f000000000101041a000000010000013b000000f80220018f000000010500008a000000000225022f000000000252013f000000000474034f000000000404043b000000000224016f000000000023041b000000010000013b0000001f010000390000000101100031000000200200008a000000000221016f0000000001320019000000000021004b00000000020000390000000102004039000000010000013b0000000402400039000000000121034f000000000201043b00000024014000391c8112c00000040f000000010000013b000000010320019000000001012002700000007f0110618f0000001f0010008c00000000040000390000000104002039000000000442013f0000000100400190000000010000013b000000000305001900000000060000191c8112420000040f000000000001004b000000010000013b00000001010000390000009702000039000000000012041b0000000001000415000000010000013b0000000007070433000000000073041b000000200660003900000020055000390000000103300039000000010000013b1c8116740000040f00000020021000390000000002020433000000000002004b000000010000013b000000000021043500000004021000390000002003000039000000000032043500000084020000391c811ad90000040f000000000020043f000000200010043f000000000100001900000040020000391c811aeb0000040f000000000101041a000000ff00100190000000010000013b000000007907043c0000000008980436000000000038004b000000010000013b000100000003001f00030000000103550000000001020019000000000001042d0000001f01200039000000200300008a000000000331016f0000000001530019000000000031004b00000000030000390000000103004039000000010000013b000000f80220018f000000010400008a000000000224022f000000000242013f0000000004070433000000000224016f000000000023041b000000010000013b0000001f01100039000000200200008a000000000221016f0000000001420019000000000021004b00000000020000390000000102004039000000010000013b00000000054501cf000000000545022f000000000303043b0000010004400089000000000343022f00000000034301cf000000000353019f000000010000013b0000ff010100008a000000000200041a000000000112016f000000000010041b0000000103000039000000400100043d0000000000310435000000010000013b000000200010043f000000400200003900000000010000191c811aeb0000040f000000000001042d000000d001000039000000000101041a000d00000001001d000000010000013b000000d001000039000000000101041a000c00000001001d000000010000013b1c81142a0000040f0000006501000039000000000101041a000000010000013b0000000403500039000000000013043500000024015000390000000000010435000000010000013b00000001010000310000001f02100039000000200300008a000000000332016f0000000002530019000000010000013b00000001010000310000001f02100039000000200300008a000000000332016f0000000d02300029000000010000013b00001c810000043200001c820001042e00001c830001043000000000000000000000000000000000000000000000000000000000ffffffff000000020000000000000000000000000000008000000100000000000000000000000000000000000000000000000000000000000000000000000000f97b57ec00000000000000000000000000000000000000000000000000000000047fc9aa00000000000000000000000000000000000000000000000000000000063a75020000000000000000000000000000000000000000000000000000000006fdde0300000000000000000000000000000000000000000000000000000000081812fc00000000000000000000000000000000000000000000000000000000095ea7b30000000000000000000000000000000000000000000000000000000018160ddd00000000000000000000000000000000000000000000000000000000184a94d50000000000000000000000000000000000000000000000000000000018bae6c80000000000000000000000000000000000000000000000000000000023b872dd000000000000000000000000000000000000000000000000000000002a55205a000000000000000000000000000000000000000000000000000000002a5ac764000000000000000000000000000000000000000000000000000000002ddcb21f000000000000000000000000000000000000000000000000000000002fc3a8d9000000000000000000000000000000000000000000000000000000003659cfe6000000000000000000000000000000000000000000000000000000003713c1a10000000000000000000000000000000000000000000000000000000042842e0e0000000000000000000000000000000000000000000000000000000042966c680000000000000000000000000000000000000000000000000000000047230dcb000000000000000000000000000000000000000000000000000000004f1ef286000000000000000000000000000000000000000000000000000000004f58122a000000000000000000000000000000000000000000000000000000004fc8416d0000000000000000000000000000000000000000000000000000000050cf5a310000000000000000000000000000000000000000000000000000000051f131670000000000000000000000000000000000000000000000000000000052d1902d00000000000000000000000000000000000000000000000000000000548205410000000000000000000000000000000000000000000000000000000055f804b300000000000000000000000000000000000000000000000000000000572b6c05000000000000000000000000000000000000000000000000000000005be95448000000000000000000000000000000000000000000000000000000006352211e000000000000000000000000000000000000000000000000000000006502abea000000000000000000000000000000000000000000000000000000006c0360eb0000000000000000000000000000000000000000000000000000000070a0823100000000000000000000000000000000000000000000000000000000715018a600000000000000000000000000000000000000000000000000000000834a20df000000000000000000000000000000000000000000000000000000008da5cb5b0000000000000000000000000000000000000000000000000000000095d89b4100000000000000000000000000000000000000000000000000000000a22cb46500000000000000000000000000000000000000000000000000000000a305f5b200000000000000000000000000000000000000000000000000000000b5c5801e00000000000000000000000000000000000000000000000000000000b88d4fde00000000000000000000000000000000000000000000000000000000bfb2995c00000000000000000000000000000000000000000000000000000000c2f5070100000000000000000000000000000000000000000000000000000000c87b56dd00000000000000000000000000000000000000000000000000000000d1f5789400000000000000000000000000000000000000000000000000000000db3e4c8400000000000000000000000000000000000000000000000000000000e8a3d48500000000000000000000000000000000000000000000000000000000e985e9c500000000000000000000000000000000000000000000000000000000eb87215000000000000000000000000000000000000000000000000000000000ee295d6200000000000000000000000000000000000000000000000000000000eeab058900000000000000000000000000000000000000000000000000000000ef60ceaf00000000000000000000000000000000000000000000000000000000f2fde38b00000000000000000000000000000000000000000000000000000000f7441c3a0000000000000000000000000000000000000000000000000000000001ffc9a766be4f155c5ef2ebd3772b228f2f00681e4ed5826cdb3b1943cc11ad15ad1d28000000000000000000000000000000000000000000000000ffffffffffffff7f000000000000000000000000ffffffffffffffffffffffffffffffffffffffff0000000000000000000000ff000000000000000000000000000000000000000000000000000000000000000100000000000000000000000000000000000000002569078dfb4b0305704d3008e7403993ae9601b85f7ae5e742de3de8f8011c401806aa1896bbf26568e884a7374b41e002500962caba6a15023a8d90e8508b8358b9164600000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffffff7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff64647265737300000000000000000000000000000000000000000000000000004f776e61626c653a206e6577206f776e657220697320746865207a65726f206108c379a00000000000000000000000000000000000000000000000000000000049a5c0f000000000000000000000000000000000000000000000000000000000ffffffffffffffffffffffff0000000000000000000000000000000000000000ffffffffffffffffffff0000000000000000000000000000000000000000000000000000000000000000ffff000000000000000000000000000000000000000002000000000000000000000000000000000000000000000000000000000000002c5ea6e4103e78cb101e796fb2dace540362fc542cbff5145eaa24af7dd8fe41d10072c000000000000000000000000000000000000000000000000000000000acc0f6922d8ccd133dbdaee1908d4d734f25a540223d55145e9fad2898471a4e15bd85bf0000000000000000000000000000000000000000000000000000000085e335b60000000000000000000000000000000000000000000000000000000047c4908e245f386bfc1825973249847f4053a761ddb4880ad63c323a7b5a2a25000000000000000000000000000000000000000000000000ffffffffffffff41ffffffffffffffffffffffffffffffffffffffffffffffff0000000000000080568efce20000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000040000000000000000000000002df7dd608aac5806395769062aca937bf08722cb45659d38c0cce82f5cd0cde374c1ace10000000000000000000000000000000000000000000000000000000080000000000000000000000000000000000000000000000000000000000000003d820a4d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffffbf02000000000000000000000000000000000000200000000000000000000000007f26b83ff96e1f2b6a682f133852f6798a09c465da95921460cefb3847402498c87b56dd0000000000000000000000000000000000000000000000000000000000ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff30000000000000000000000000000000000000000000000000000000000000002f00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffffdfa14c4b5000000000000000000000000000000000000000000000000000000000ceea21b600000000000000000000000000000000000000000000000000000000d018c0fa00000000000000000000000000000000000000000000000000000000e236d251e7da4ac8a3c275f80aebc9b74cbf356425fa04864da5043511a81219eabab42200000000000000000000000000000000000000000000000000000000ad6e40c7000000000000000000000000000000000000000000000000000000009c39c41c00000000000000000000000000000000000000000000000000000000eaf422d1ab2d8f38856ab6be8378c08d2886463a33667e896ba3a6ebf2c6260c57c31fde00000000000000000000000000000000000000000000000000000000f6bdd07c00000000000000000000000000000000000000000000000000000000ce2003a500000000000000000000000000000000000000000000000000000000c407dfe559ca4a63d7cf8439b9fce2965d7a4c8c92eb5d5047f53bb9eec295fd16365cdd000000000000000000000000000000000000000000000000000000002569078dfb4b0305704d3008e7403993ae9601b85f7ae5e742de3de8f8011c47020000000000000000000000000000000000002000000080000000000000000017307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c312569078dfb4b0305704d3008e7403993ae9601b85f7ae5e742de3de8f8011c43617167b76dcc8247761fd21f427ad8ec3be6b3be203aed34e3aac08b4d31817c4e487b710000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000240000000000000000000000009941b0eaa3a10d142c88d4dd70d0ff97e1b12a7d9324c4e6bc33ee52ea52e2d973e05f4c0a27cbf60841afa4a2f8577f5e99db3e3e9c293800d6e6b1f7b8a7aba9292a6f000000000000000000000000000000000000000000000000000000008be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e02569078dfb4b0305704d3008e7403993ae9601b85f7ae5e742de3de8f8011c458f4eb6040000000000000000000000000000000000000000000000000000000051858de9989bf7441865ebdadbf7382c8838edbf830f5d86a9a51ac773676dd6029838610000000000000000000000000000000000000000000000000000000059faca9600000000000000000000000000000000000000000000000000000000ecd7b0d100000000000000000000000000000000000000000000000000000000310ab089e4439a4c15d089f94afb7896ff553aecb10793d0ab882de59d99a32e555550535570677261646561626c653a206d757374206e6f742062652063616c6c6564207468726f7567682064656c656761746563616c6c0000000000000000360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbcffffffffffffffffffffff000000000000000000000000000000000000000000496e76616c696420696e7075740000000000000000000000000000000000000001ffc9a7000000000000000000000000000000000000000000000000000000008808898a00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffff9fbcdbc94700000000000000000000000000000000000000000000000000000000e10a05050000000000000000000000000000000000000000000000000000000034e70f7a0000000000000000000000000000000000000000000000000000000007ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff4d30c2588339a383e2c3092b5383924cf202ae9c377b76c5fe5208cd41d8d2aa80d93734000000000000000000000000000000000000000000000000000000004a3891c200000000000000000000000000000000000000000000000000000000bf317c2a00000000000000000000000000000000000000000000000000000000ea6eb5c300000000000000000000000000000000000000000000000000000000129dae8b0000000000000000000000000000000000000000000000000000000082b42900000000000000000000000000000000000000000000000000000000002569078dfb4b0305704d3008e7403993ae9601b85f7ae5e742de3de8f8011c46ffffffffffffffffffffffffffffffff00000000000000000000000000000001796b89b91644bc98cd93958e4c9038275d622183e25ac5af08cc6b5d955391322569078dfb4b0305704d3008e7403993ae9601b85f7ae5e742de3de8f8011c4400000003000000000000000000000000000000000000000000000000000000000000000200000000000000000000000000000000000000000000000000000000ddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef2569078dfb4b0305704d3008e7403993ae9601b85f7ae5e742de3de8f8011c4123de665100000000000000000000000000000000000000000000000000000000ad7735a1decc3db45c1eab757cc4a1cb4bff8e332877b55f6db4ea11d82320c62504956c000000000000000000000000000000000000000000000000000000002569078dfb4b0305704d3008e7403993ae9601b85f7ae5e742de3de8f8011c42933ecf8acb7824b680a8d16f3ff3db8864228d986aa4c2ebab1eeb2703b4beb35bf57bc300000000000000000000000000000000000000000000000000000000be1a657d0000000000000000000000000000000000000000000000000000000045a9661700000000000000000000000000000000000000000000000000000000f8d2906c00000000000000000000000000000000000000000000000000000000cfb3b942000000000000000000000000000000000000000000000000000000008c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b925ffffffffffffffffffffff00ffffffffffffffffffffffffffffffffffffffff9d4b38394dff663b46f53f52ced584161b5021180321b381f6104325eecebf49fd120bd20000000000000000000000000000000000000000000000000000000000000000ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0000000000000000000000000000000000000000000000000000000080ac58cd000000000000000000000000000000000000000000000000000000005b5e139f0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffe000000000000000000000000000000000000000000000000100000000000000004f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657246756e6374696f6e206d7573742062652063616c6c6564207468726f756768206163746976652070726f787900000000000000000000000000000000000000004e6f74207570677261646561626c65000000000000000000000000000000000064656c656761746563616c6c00000000000000000000000000000000000000009941b0eaa3a10d142c88d4dd70d0ff97e1b12a7d9324c4e6bc33ee52ea52e2d808000000000000000000000000000000000000000000000000000000000000005265656e7472616e637947756172643a207265656e7472616e742063616c6c00ffffffffffffffffffff0000ffffffffffffffffffffffffffffffffffffffff647920696e697469616c697a6564000000000000000000000000000000000000496e697469616c697a61626c653a20636f6e747261637420697320616c7265616cc130753487db497f572e90c00c24779bdd7267955b3d1454e114d8fc4b414db83b6f71dba0c79403e7da68cdb67b80bfac589e224b77f529c3cdc584a5d5db9e8e984892337db889e02de0bd852713c4194c41dfc512cb1c553f74b2ce7e842445a9b6000000000000000000000000000000000000000000000000000000006e697469616c697a696e67000000000000000000000000000000000000000000496e697469616c697a61626c653a20636f6e7472616374206973206e6f742069000000000000000000000000000000000000000000000000ffffffffffffffc02e0a5b969d96a99aee0b35787d9a60516a02ca6f528a5f66d3f936468d8f0382ae7a7216676408bbe79a14252408c7d377c712407cf0a279565ae5388c98922a0000000100000000000000000000000000000000000000000000000000000000df2d9b4200000000000000000000000000000000000000000000000000000000cf4700e400000000000000000000000000000000000000000000000000000000a11481000000000000000000000000000000000000000000000000000000000059c896be00000000000000000000000000000000000000000000000000000000f5e8d69e000000000000000000000000000000000000000000000000000000008af6791b00000000000000000000000000000000000000000000000000000000ea553b3400000000000000000000000000000000000000000000000000000000150b7a0200000000000000000000000000000000000000000000000000000000d1a57ed600000000000000000000000000000000000000000000000000000000fffffffffffffffffffffffffffffffffffffffffffffffeffffffffffffffff2e076300000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000010000000000000001b562e8dd0000000000000000000000000000000000000000000000000000000076413c5e00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffffa002000002000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000266f28977285e71829eb72bbc356282aef3f9b921dd9386f376b96d3d46c48cd
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 31 Chains
Chain | Token | Portfolio % | Price | Amount | Value |
---|
[ Download: CSV Export ]
A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.