ETH Price: $1,915.06 (+0.19%)
Gas: 0.57 Gwei

Transaction Decoder

Block:
17513324 at Jun-19-2023 10:53:23 AM +UTC
Transaction Fee:
0.00401062885869576 ETH $7.68
Gas Used:
110,040 Gas / 36.447008894 Gwei

Emitted Events:

137 Diamond.0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef( 0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef, 0x0000000000000000000000000000000000000000000000000000000000000000, 0x000000000000000000000000ec81dbb16c88328429cf4923d8a067b694e9f2fc, 0x0000000000000000000000000000000000000000000000000000000000000d9f )
138 Diamond.0x6bd5c950a8d8df17f772f5af37cb3655737899cbf903264b9795592da439661c( 0x6bd5c950a8d8df17f772f5af37cb3655737899cbf903264b9795592da439661c, 0000000000000000000000000000000000000000000000000000000000000000, 0000000000000000000000000000000000000000000000000000000000000d9e )

Account State Difference:

  Address   Before After State Difference Code
(Fee Recipient: 0x9f4...a2A)
90.190586022923251627 Eth90.190656448523251627 Eth0.0000704256
0xeC81DBB1...694e9F2FC
0.024332213121389302 Eth
Nonce: 267
0.020321584262693542 Eth
Nonce: 268
0.00401062885869576
0xFada0ef9...861E8a804

Execution Trace

Diamond.CALL( )
  • MintFacet.DELEGATECALL( )
    File 1 of 2: Diamond
    // SPDX-License-Identifier: MIT
    pragma solidity ^0.8.0;
    /******************************************************************************\\
    * Author: Nick Mudge <[email protected]> (https://twitter.com/mudgen)
    * EIP-2535 Diamonds: https://eips.ethereum.org/EIPS/eip-2535
    *
    * Implementation of a diamond.
    /******************************************************************************/
    import {LibDiamond} from "./libraries/LibDiamond.sol";
    import {IDiamondCut} from "./interfaces/IDiamondCut.sol";
    contract Diamond {
        struct Initialization {
            address initContract;
            bytes initData;
        }
        /// @notice This construct a diamond contract
        /// @param _contractOwner the owner of the contract. With default DiamondCutFacet, this is the sole address allowed to make further cuts.
        /// @param _diamondCut the list of facet to add
        /// @param _initializations the list of initialization pair to execute. This allow to setup a contract with multiple level of independent initialization.
        constructor(
            address _contractOwner,
            IDiamondCut.FacetCut[] memory _diamondCut,
            Initialization[] memory _initializations
        ) payable {
            if (_contractOwner != address(0)) {
                LibDiamond.setContractOwner(_contractOwner);
            }
            LibDiamond.diamondCut(_diamondCut, address(0), "");
            for (uint256 i = 0; i < _initializations.length; i++) {
                LibDiamond.initializeDiamondCut(_initializations[i].initContract, _initializations[i].initData);
            }
        }
        // Find facet for function that is called and execute the
        // function if a facet is found and return any value.
        fallback() external payable {
            LibDiamond.DiamondStorage storage ds;
            bytes32 position = LibDiamond.DIAMOND_STORAGE_POSITION;
            // get diamond storage
            assembly {
                ds.slot := position
            }
            // get facet from function selector
            address facet = ds.selectorToFacetAndPosition[msg.sig].facetAddress;
            require(facet != address(0), "Diamond: Function does not exist");
            // Execute external function from facet using delegatecall and return any value.
            assembly {
                // copy function selector and any arguments
                calldatacopy(0, 0, calldatasize())
                // execute function call using the facet
                let result := delegatecall(gas(), facet, 0, calldatasize(), 0, 0)
                // get any return value
                returndatacopy(0, 0, returndatasize())
                // return any return value or error back to the caller
                switch result
                case 0 {
                    revert(0, returndatasize())
                }
                default {
                    return(0, returndatasize())
                }
            }
        }
        receive() external payable {}
    }
    // SPDX-License-Identifier: MIT
    pragma solidity ^0.8.0;
    /******************************************************************************\\
    * Author: Nick Mudge <[email protected]> (https://twitter.com/mudgen)
    * EIP-2535 Diamonds: https://eips.ethereum.org/EIPS/eip-2535
    /******************************************************************************/
    interface IDiamondCut {
        enum FacetCutAction {Add, Replace, Remove}
        // Add=0, Replace=1, Remove=2
        struct FacetCut {
            address facetAddress;
            FacetCutAction action;
            bytes4[] functionSelectors;
        }
        /// @notice Add/replace/remove any number of functions and optionally execute
        ///         a function with delegatecall
        /// @param _diamondCut Contains the facet addresses and function selectors
        /// @param _init The address of the contract or facet to execute _calldata
        /// @param _calldata A function call, including function selector and arguments
        ///                  _calldata is executed with delegatecall on _init
        function diamondCut(
            FacetCut[] calldata _diamondCut,
            address _init,
            bytes calldata _calldata
        ) external;
        event DiamondCut(FacetCut[] _diamondCut, address _init, bytes _calldata);
    }
    // SPDX-License-Identifier: MIT
    pragma solidity ^0.8.0;
    /******************************************************************************\\
    * Author: Nick Mudge <[email protected]> (https://twitter.com/mudgen)
    * EIP-2535 Diamonds: https://eips.ethereum.org/EIPS/eip-2535
    /******************************************************************************/
    import { IDiamondCut } from "../interfaces/IDiamondCut.sol";
    library LibDiamond {
        bytes32 constant DIAMOND_STORAGE_POSITION = keccak256("diamond.standard.diamond.storage");
        struct FacetAddressAndPosition {
            address facetAddress;
            uint96 functionSelectorPosition; // position in facetFunctionSelectors.functionSelectors array
        }
        struct FacetFunctionSelectors {
            bytes4[] functionSelectors;
            uint256 facetAddressPosition; // position of facetAddress in facetAddresses array
        }
        struct DiamondStorage {
            // maps function selector to the facet address and
            // the position of the selector in the facetFunctionSelectors.selectors array
            mapping(bytes4 => FacetAddressAndPosition) selectorToFacetAndPosition;
            // maps facet addresses to function selectors
            mapping(address => FacetFunctionSelectors) facetFunctionSelectors;
            // facet addresses
            address[] facetAddresses;
            // Used to query if a contract implements an interface.
            // Used to implement ERC-165.
            mapping(bytes4 => bool) supportedInterfaces;
            // owner of the contract
            address contractOwner;
        }
        function diamondStorage() internal pure returns (DiamondStorage storage ds) {
            bytes32 position = DIAMOND_STORAGE_POSITION;
            assembly {
                ds.slot := position
            }
        }
        event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
        function setContractOwner(address _newOwner) internal {
            DiamondStorage storage ds = diamondStorage();
            address previousOwner = ds.contractOwner;
            ds.contractOwner = _newOwner;
            emit OwnershipTransferred(previousOwner, _newOwner);
        }
        function contractOwner() internal view returns (address contractOwner_) {
            contractOwner_ = diamondStorage().contractOwner;
        }
        function enforceIsContractOwner() internal view {
            require(msg.sender == diamondStorage().contractOwner, "LibDiamond: Must be contract owner");
        }
        event DiamondCut(IDiamondCut.FacetCut[] _diamondCut, address _init, bytes _calldata);
        // Internal function version of diamondCut
        function diamondCut(
            IDiamondCut.FacetCut[] memory _diamondCut,
            address _init,
            bytes memory _calldata
        ) internal {
            for (uint256 facetIndex; facetIndex < _diamondCut.length; facetIndex++) {
                IDiamondCut.FacetCutAction action = _diamondCut[facetIndex].action;
                if (action == IDiamondCut.FacetCutAction.Add) {
                    addFunctions(_diamondCut[facetIndex].facetAddress, _diamondCut[facetIndex].functionSelectors);
                } else if (action == IDiamondCut.FacetCutAction.Replace) {
                    replaceFunctions(_diamondCut[facetIndex].facetAddress, _diamondCut[facetIndex].functionSelectors);
                } else if (action == IDiamondCut.FacetCutAction.Remove) {
                    removeFunctions(_diamondCut[facetIndex].facetAddress, _diamondCut[facetIndex].functionSelectors);
                } else {
                    revert("LibDiamondCut: Incorrect FacetCutAction");
                }
            }
            emit DiamondCut(_diamondCut, _init, _calldata);
            initializeDiamondCut(_init, _calldata);
        }
        function addFunctions(address _facetAddress, bytes4[] memory _functionSelectors) internal {
            require(_functionSelectors.length > 0, "LibDiamondCut: No selectors in facet to cut");
            DiamondStorage storage ds = diamondStorage();        
            require(_facetAddress != address(0), "LibDiamondCut: Add facet can't be address(0)");
            uint96 selectorPosition = uint96(ds.facetFunctionSelectors[_facetAddress].functionSelectors.length);
            // add new facet address if it does not exist
            if (selectorPosition == 0) {
                addFacet(ds, _facetAddress);            
            }
            for (uint256 selectorIndex; selectorIndex < _functionSelectors.length; selectorIndex++) {
                bytes4 selector = _functionSelectors[selectorIndex];
                address oldFacetAddress = ds.selectorToFacetAndPosition[selector].facetAddress;
                require(oldFacetAddress == address(0), "LibDiamondCut: Can't add function that already exists");
                addFunction(ds, selector, selectorPosition, _facetAddress);
                selectorPosition++;
            }
        }
        function replaceFunctions(address _facetAddress, bytes4[] memory _functionSelectors) internal {
            require(_functionSelectors.length > 0, "LibDiamondCut: No selectors in facet to cut");
            DiamondStorage storage ds = diamondStorage();
            require(_facetAddress != address(0), "LibDiamondCut: Add facet can't be address(0)");
            uint96 selectorPosition = uint96(ds.facetFunctionSelectors[_facetAddress].functionSelectors.length);
            // add new facet address if it does not exist
            if (selectorPosition == 0) {
                addFacet(ds, _facetAddress);
            }
            for (uint256 selectorIndex; selectorIndex < _functionSelectors.length; selectorIndex++) {
                bytes4 selector = _functionSelectors[selectorIndex];
                address oldFacetAddress = ds.selectorToFacetAndPosition[selector].facetAddress;
                require(oldFacetAddress != _facetAddress, "LibDiamondCut: Can't replace function with same function");
                removeFunction(ds, oldFacetAddress, selector);
                addFunction(ds, selector, selectorPosition, _facetAddress);
                selectorPosition++;
            }
        }
        function removeFunctions(address _facetAddress, bytes4[] memory _functionSelectors) internal {
            require(_functionSelectors.length > 0, "LibDiamondCut: No selectors in facet to cut");
            DiamondStorage storage ds = diamondStorage();
            // if function does not exist then do nothing and return
            require(_facetAddress == address(0), "LibDiamondCut: Remove facet address must be address(0)");
            for (uint256 selectorIndex; selectorIndex < _functionSelectors.length; selectorIndex++) {
                bytes4 selector = _functionSelectors[selectorIndex];
                address oldFacetAddress = ds.selectorToFacetAndPosition[selector].facetAddress;
                removeFunction(ds, oldFacetAddress, selector);
            }
        }
        function addFacet(DiamondStorage storage ds, address _facetAddress) internal {
            enforceHasContractCode(_facetAddress, "LibDiamondCut: New facet has no code");
            ds.facetFunctionSelectors[_facetAddress].facetAddressPosition = ds.facetAddresses.length;
            ds.facetAddresses.push(_facetAddress);
        }    
        function addFunction(DiamondStorage storage ds, bytes4 _selector, uint96 _selectorPosition, address _facetAddress) internal {
            ds.selectorToFacetAndPosition[_selector].functionSelectorPosition = _selectorPosition;
            ds.facetFunctionSelectors[_facetAddress].functionSelectors.push(_selector);
            ds.selectorToFacetAndPosition[_selector].facetAddress = _facetAddress;
        }
        function removeFunction(DiamondStorage storage ds, address _facetAddress, bytes4 _selector) internal {        
            require(_facetAddress != address(0), "LibDiamondCut: Can't remove function that doesn't exist");
            // an immutable function is a function defined directly in a diamond
            require(_facetAddress != address(this), "LibDiamondCut: Can't remove immutable function");
            // replace selector with last selector, then delete last selector
            uint256 selectorPosition = ds.selectorToFacetAndPosition[_selector].functionSelectorPosition;
            uint256 lastSelectorPosition = ds.facetFunctionSelectors[_facetAddress].functionSelectors.length - 1;
            // if not the same then replace _selector with lastSelector
            if (selectorPosition != lastSelectorPosition) {
                bytes4 lastSelector = ds.facetFunctionSelectors[_facetAddress].functionSelectors[lastSelectorPosition];
                ds.facetFunctionSelectors[_facetAddress].functionSelectors[selectorPosition] = lastSelector;
                ds.selectorToFacetAndPosition[lastSelector].functionSelectorPosition = uint96(selectorPosition);
            }
            // delete the last selector
            ds.facetFunctionSelectors[_facetAddress].functionSelectors.pop();
            delete ds.selectorToFacetAndPosition[_selector];
            // if no more selectors for facet address then delete the facet address
            if (lastSelectorPosition == 0) {
                // replace facet address with last facet address and delete last facet address
                uint256 lastFacetAddressPosition = ds.facetAddresses.length - 1;
                uint256 facetAddressPosition = ds.facetFunctionSelectors[_facetAddress].facetAddressPosition;
                if (facetAddressPosition != lastFacetAddressPosition) {
                    address lastFacetAddress = ds.facetAddresses[lastFacetAddressPosition];
                    ds.facetAddresses[facetAddressPosition] = lastFacetAddress;
                    ds.facetFunctionSelectors[lastFacetAddress].facetAddressPosition = facetAddressPosition;
                }
                ds.facetAddresses.pop();
                delete ds.facetFunctionSelectors[_facetAddress].facetAddressPosition;
            }
        }
        function initializeDiamondCut(address _init, bytes memory _calldata) internal {
            if (_init == address(0)) {
                require(_calldata.length == 0, "LibDiamondCut: _init is address(0) but_calldata is not empty");
            } else {
                require(_calldata.length > 0, "LibDiamondCut: _calldata is empty but _init is not address(0)");
                if (_init != address(this)) {
                    enforceHasContractCode(_init, "LibDiamondCut: _init address has no code");
                }
                (bool success, bytes memory error) = _init.delegatecall(_calldata);
                if (!success) {
                    if (error.length > 0) {
                        // bubble up the error
                        revert(string(error));
                    } else {
                        revert("LibDiamondCut: _init function reverted");
                    }
                }
            }
        }
        function enforceHasContractCode(address _contract, string memory _errorMessage) internal view {
            uint256 contractSize;
            assembly {
                contractSize := extcodesize(_contract)
            }
            require(contractSize > 0, _errorMessage);
        }
    }
    

    File 2 of 2: MintFacet
    // SPDX-License-Identifier: MIT
    pragma solidity ^0.8.0;
    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
    // ERC721A Contracts v4.2.3
    // Creator: Chiru Labs
    pragma solidity ^0.8.4;
    import {ERC721AStorage} from './ERC721AStorage.sol';
    import {IERC721AUpgradeable} from './IERC721AUpgradeable.sol';
    interface ERC721A__IERC721ReceiverUpgradeable {
        function onERC721Received(
            address operator,
            address from,
            uint256 tokenId,
            bytes calldata data
        ) external returns (bytes4);
    }
    /**
     * @title ERC721A
     *
     * @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 ERC721AUpgradeableInternal is IERC721AUpgradeable {
        using ERC721AStorage for ERC721AStorage.Layout;
        // =============================================================
        //                           CONSTANTS
        // =============================================================
        // Mask of an entry in packed address data.
        uint256 internal constant _BITMASK_ADDRESS_DATA_ENTRY = (1 << 64) - 1;
        // The bit position of `numberMinted` in packed address data.
        uint256 internal constant _BITPOS_NUMBER_MINTED = 64;
        // The bit position of `numberBurned` in packed address data.
        uint256 internal constant _BITPOS_NUMBER_BURNED = 128;
        // The bit position of `aux` in packed address data.
        uint256 internal constant _BITPOS_AUX = 192;
        // Mask of all 256 bits in packed address data except the 64 bits for `aux`.
        uint256 internal constant _BITMASK_AUX_COMPLEMENT = (1 << 192) - 1;
        // The bit position of `startTimestamp` in packed ownership.
        uint256 internal constant _BITPOS_START_TIMESTAMP = 160;
        // The bit mask of the `burned` bit in packed ownership.
        uint256 internal constant _BITMASK_BURNED = 1 << 224;
        // The bit position of the `nextInitialized` bit in packed ownership.
        uint256 internal constant _BITPOS_NEXT_INITIALIZED = 225;
        // The bit mask of the `nextInitialized` bit in packed ownership.
        uint256 internal constant _BITMASK_NEXT_INITIALIZED = 1 << 225;
        // The bit position of `extraData` in packed ownership.
        uint256 internal constant _BITPOS_EXTRA_DATA = 232;
        // Mask of all 256 bits in a packed ownership except the 24 bits for `extraData`.
        uint256 internal constant _BITMASK_EXTRA_DATA_COMPLEMENT = (1 << 232) - 1;
        // The mask of the lower 160 bits for addresses.
        uint256 internal 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 internal constant _MAX_MINT_ERC2309_QUANTITY_LIMIT = 5000;
        // The `Transfer` event signature is given by:
        // `keccak256(bytes("Transfer(address,address,uint256)"))`.
        bytes32 internal constant _TRANSFER_EVENT_SIGNATURE =
            0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef;
        // =============================================================
        //                   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 0;
        }
        /**
         * @dev Returns the next token ID to be minted.
         */
        function _nextTokenId() internal view virtual returns (uint256) {
            return ERC721AStorage.layout()._currentIndex;
        }
        /**
         * @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
        // =============================================================
        /**
         * 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;
        }
        // =============================================================
        //                        IERC721Metadata
        // =============================================================
        /**
         * @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 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) internal 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) internal 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) internal 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) internal 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 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 `msgSender` is equal to `approvedAddress` or `owner`.
         */
        function _isSenderApprovedOrOwner(
            address approvedAddress,
            address owner,
            address msgSender
        ) internal 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)
            internal
            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 Hook that is called before a set of serially-ordered token IDs
         * are about to be transferred. This includes minting.
         * And also called before burning one token.
         *
         * `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` 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 _beforeTokenTransfers(
            address from,
            address to,
            uint256 startTokenId,
            uint256 quantity
        ) internal virtual {}
        /**
         * @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,
            uint256 quantity
        ) internal virtual {}
        /**
         * @dev internal 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
        ) internal 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);
            _beforeTokenTransfers(address(0), to, startTokenId, quantity);
            // 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;
            }
            _afterTokenTransfers(address(0), to, startTokenId, quantity);
        }
        /**
         * @dev Mints `quantity` tokens and transfers them to `to`.
         *
         * This function is intended for efficient minting only during contract creation.
         *
         * It emits only one {ConsecutiveTransfer} as defined in
         * [ERC2309](https://eips.ethereum.org/EIPS/eip-2309),
         * instead of a sequence of {Transfer} event(s).
         *
         * Calling this function outside of contract creation WILL make your contract
         * non-compliant with the ERC721 standard.
         * For full ERC721 compliance, substituting ERC721 {Transfer} event(s) with the ERC2309
         * {ConsecutiveTransfer} event is only permissible during contract creation.
         *
         * Requirements:
         *
         * - `to` cannot be the zero address.
         * - `quantity` must be greater than 0.
         *
         * Emits a {ConsecutiveTransfer} event.
         */
        function _mintERC2309(address to, uint256 quantity) internal virtual {
            uint256 startTokenId = ERC721AStorage.layout()._currentIndex;
            if (to == address(0)) _revert(MintToZeroAddress.selector);
            if (quantity == 0) _revert(MintZeroQuantity.selector);
            if (quantity > _MAX_MINT_ERC2309_QUANTITY_LIMIT) _revert(MintERC2309QuantityExceedsLimit.selector);
            _beforeTokenTransfers(address(0), to, startTokenId, quantity);
            // Overflows are unrealistic due to the above check for `quantity` to be below the limit.
            unchecked {
                // Updates:
                // - `balance += quantity`.
                // - `numberMinted += quantity`.
                //
                // We can directly add to the `balance` and `numberMinted`.
                ERC721AStorage.layout()._packedAddressData[to] += quantity * ((1 << _BITPOS_NUMBER_MINTED) | 1);
                // 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)
                );
                emit ConsecutiveTransfer(startTokenId, startTokenId + quantity - 1, address(0), to);
                ERC721AStorage.layout()._currentIndex = startTokenId + quantity;
            }
            _afterTokenTransfers(address(0), to, startTokenId, quantity);
        }
        /**
         * @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);
        }
        function _ownerOf(uint256 tokenId) internal view returns (address) {
            return address(uint160(_packedOwnershipOf(tokenId)));
        }
        
        /**
         * @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);
        }
        
        function _isApprovedForAll(address owner, address operator) internal view returns (bool) {
            return ERC721AStorage.layout()._operatorApprovals[owner][operator];
        }
        /**
         * @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);
            }
            _beforeTokenTransfers(from, address(0), tokenId, 1);
            // 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);
            _afterTokenTransfers(from, address(0), tokenId, 1);
            // 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
        ) internal 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 {
            assembly {
                mstore(0x00, errorSelector)
                revert(0x00, 0x04)
            }
        }
    }
    // SPDX-License-Identifier: MIT
    // ERC721A Contracts v4.2.3
    // Creator: Chiru Labs
    pragma solidity ^0.8.4;
    /**
     * @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;
        }
        /**
         * @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);
        event ConsecutiveTransfer(uint256 indexed fromTokenId, uint256 toTokenId, address indexed from, address indexed to);
    }
    // SPDX-License-Identifier: MIT
    pragma solidity 0.8.17;
    import {ERC721AUpgradeableInternal} from "./ERC721AUpgradeable/ERC721AUpgradeableInternal.sol";
    import {GasLoverStorage, WithStorage} from "./WithStorage.sol";
    import {SafeCastLib} from "solady/src/utils/SafeCastLib.sol";
    contract MintFacet is ERC721AUpgradeableInternal, WithStorage {
        using SafeCastLib for *;
        
        event BatchMetadataUpdate(uint256 _fromTokenId, uint256 _toTokenId);
        
        function mint() external {
            GasLoverStorage storage gs = s();
            
            uint tokenId = _nextTokenId();
            require(tokenId < gs.maxSupply, "Exceeds max supply");
            require(msg.sender == tx.origin, "Contract cannot mint");
            
            _mint(msg.sender, 1);
            
            uint packed = packTokenInfo(tx.gasprice, block.timestamp, msg.sender);
            gs.tokenIdToPackedInfo.push(packed);
            
            if (tokenId != 0) emit BatchMetadataUpdate(0, tokenId - 1);
        }
        
        function packTokenInfo(uint gasPrice, uint timestamp, address creator) internal pure returns (uint) {
            uint packedGasPrice = uint256(gasPrice.toUint64()) << 192;
            uint packedTimestamp = uint256(timestamp.toUint32()) << 160;
            uint packedCreator = uint256(uint160(creator));
            
            return packedGasPrice | packedTimestamp | packedCreator;
        }
    }
    // SPDX-License-Identifier: MIT
    pragma solidity 0.8.17;
    import {LibDiamond} from "hardhat-deploy/solc_0.8/diamond/libraries/LibDiamond.sol";
    struct GasLoverStorage {
        uint maxSupply;
        bool isInitialized;
        uint[] tokenIdToPackedInfo;
    }
    contract WithStorage {
        function s() internal pure returns (GasLoverStorage storage cs) {
            bytes32 position = keccak256("gas.lovers.nft.contract.storage");
            assembly {
               cs.slot := position
            }
        }
        
        function ds() internal pure returns (LibDiamond.DiamondStorage storage) {
            return LibDiamond.diamondStorage();
        }
    }
    // SPDX-License-Identifier: MIT
    pragma solidity ^0.8.0;
    /******************************************************************************\\
    * Author: Nick Mudge <[email protected]> (https://twitter.com/mudgen)
    * EIP-2535 Diamonds: https://eips.ethereum.org/EIPS/eip-2535
    /******************************************************************************/
    interface IDiamondCut {
        enum FacetCutAction {Add, Replace, Remove}
        // Add=0, Replace=1, Remove=2
        struct FacetCut {
            address facetAddress;
            FacetCutAction action;
            bytes4[] functionSelectors;
        }
        /// @notice Add/replace/remove any number of functions and optionally execute
        ///         a function with delegatecall
        /// @param _diamondCut Contains the facet addresses and function selectors
        /// @param _init The address of the contract or facet to execute _calldata
        /// @param _calldata A function call, including function selector and arguments
        ///                  _calldata is executed with delegatecall on _init
        function diamondCut(
            FacetCut[] calldata _diamondCut,
            address _init,
            bytes calldata _calldata
        ) external;
        event DiamondCut(FacetCut[] _diamondCut, address _init, bytes _calldata);
    }
    // SPDX-License-Identifier: MIT
    pragma solidity ^0.8.0;
    /******************************************************************************\\
    * Author: Nick Mudge <[email protected]> (https://twitter.com/mudgen)
    * EIP-2535 Diamonds: https://eips.ethereum.org/EIPS/eip-2535
    /******************************************************************************/
    import { IDiamondCut } from "../interfaces/IDiamondCut.sol";
    library LibDiamond {
        bytes32 constant DIAMOND_STORAGE_POSITION = keccak256("diamond.standard.diamond.storage");
        struct FacetAddressAndPosition {
            address facetAddress;
            uint96 functionSelectorPosition; // position in facetFunctionSelectors.functionSelectors array
        }
        struct FacetFunctionSelectors {
            bytes4[] functionSelectors;
            uint256 facetAddressPosition; // position of facetAddress in facetAddresses array
        }
        struct DiamondStorage {
            // maps function selector to the facet address and
            // the position of the selector in the facetFunctionSelectors.selectors array
            mapping(bytes4 => FacetAddressAndPosition) selectorToFacetAndPosition;
            // maps facet addresses to function selectors
            mapping(address => FacetFunctionSelectors) facetFunctionSelectors;
            // facet addresses
            address[] facetAddresses;
            // Used to query if a contract implements an interface.
            // Used to implement ERC-165.
            mapping(bytes4 => bool) supportedInterfaces;
            // owner of the contract
            address contractOwner;
        }
        function diamondStorage() internal pure returns (DiamondStorage storage ds) {
            bytes32 position = DIAMOND_STORAGE_POSITION;
            assembly {
                ds.slot := position
            }
        }
        event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
        function setContractOwner(address _newOwner) internal {
            DiamondStorage storage ds = diamondStorage();
            address previousOwner = ds.contractOwner;
            ds.contractOwner = _newOwner;
            emit OwnershipTransferred(previousOwner, _newOwner);
        }
        function contractOwner() internal view returns (address contractOwner_) {
            contractOwner_ = diamondStorage().contractOwner;
        }
        function enforceIsContractOwner() internal view {
            require(msg.sender == diamondStorage().contractOwner, "LibDiamond: Must be contract owner");
        }
        event DiamondCut(IDiamondCut.FacetCut[] _diamondCut, address _init, bytes _calldata);
        // Internal function version of diamondCut
        function diamondCut(
            IDiamondCut.FacetCut[] memory _diamondCut,
            address _init,
            bytes memory _calldata
        ) internal {
            for (uint256 facetIndex; facetIndex < _diamondCut.length; facetIndex++) {
                IDiamondCut.FacetCutAction action = _diamondCut[facetIndex].action;
                if (action == IDiamondCut.FacetCutAction.Add) {
                    addFunctions(_diamondCut[facetIndex].facetAddress, _diamondCut[facetIndex].functionSelectors);
                } else if (action == IDiamondCut.FacetCutAction.Replace) {
                    replaceFunctions(_diamondCut[facetIndex].facetAddress, _diamondCut[facetIndex].functionSelectors);
                } else if (action == IDiamondCut.FacetCutAction.Remove) {
                    removeFunctions(_diamondCut[facetIndex].facetAddress, _diamondCut[facetIndex].functionSelectors);
                } else {
                    revert("LibDiamondCut: Incorrect FacetCutAction");
                }
            }
            emit DiamondCut(_diamondCut, _init, _calldata);
            initializeDiamondCut(_init, _calldata);
        }
        function addFunctions(address _facetAddress, bytes4[] memory _functionSelectors) internal {
            require(_functionSelectors.length > 0, "LibDiamondCut: No selectors in facet to cut");
            DiamondStorage storage ds = diamondStorage();        
            require(_facetAddress != address(0), "LibDiamondCut: Add facet can't be address(0)");
            uint96 selectorPosition = uint96(ds.facetFunctionSelectors[_facetAddress].functionSelectors.length);
            // add new facet address if it does not exist
            if (selectorPosition == 0) {
                addFacet(ds, _facetAddress);            
            }
            for (uint256 selectorIndex; selectorIndex < _functionSelectors.length; selectorIndex++) {
                bytes4 selector = _functionSelectors[selectorIndex];
                address oldFacetAddress = ds.selectorToFacetAndPosition[selector].facetAddress;
                require(oldFacetAddress == address(0), "LibDiamondCut: Can't add function that already exists");
                addFunction(ds, selector, selectorPosition, _facetAddress);
                selectorPosition++;
            }
        }
        function replaceFunctions(address _facetAddress, bytes4[] memory _functionSelectors) internal {
            require(_functionSelectors.length > 0, "LibDiamondCut: No selectors in facet to cut");
            DiamondStorage storage ds = diamondStorage();
            require(_facetAddress != address(0), "LibDiamondCut: Add facet can't be address(0)");
            uint96 selectorPosition = uint96(ds.facetFunctionSelectors[_facetAddress].functionSelectors.length);
            // add new facet address if it does not exist
            if (selectorPosition == 0) {
                addFacet(ds, _facetAddress);
            }
            for (uint256 selectorIndex; selectorIndex < _functionSelectors.length; selectorIndex++) {
                bytes4 selector = _functionSelectors[selectorIndex];
                address oldFacetAddress = ds.selectorToFacetAndPosition[selector].facetAddress;
                require(oldFacetAddress != _facetAddress, "LibDiamondCut: Can't replace function with same function");
                removeFunction(ds, oldFacetAddress, selector);
                addFunction(ds, selector, selectorPosition, _facetAddress);
                selectorPosition++;
            }
        }
        function removeFunctions(address _facetAddress, bytes4[] memory _functionSelectors) internal {
            require(_functionSelectors.length > 0, "LibDiamondCut: No selectors in facet to cut");
            DiamondStorage storage ds = diamondStorage();
            // if function does not exist then do nothing and return
            require(_facetAddress == address(0), "LibDiamondCut: Remove facet address must be address(0)");
            for (uint256 selectorIndex; selectorIndex < _functionSelectors.length; selectorIndex++) {
                bytes4 selector = _functionSelectors[selectorIndex];
                address oldFacetAddress = ds.selectorToFacetAndPosition[selector].facetAddress;
                removeFunction(ds, oldFacetAddress, selector);
            }
        }
        function addFacet(DiamondStorage storage ds, address _facetAddress) internal {
            enforceHasContractCode(_facetAddress, "LibDiamondCut: New facet has no code");
            ds.facetFunctionSelectors[_facetAddress].facetAddressPosition = ds.facetAddresses.length;
            ds.facetAddresses.push(_facetAddress);
        }    
        function addFunction(DiamondStorage storage ds, bytes4 _selector, uint96 _selectorPosition, address _facetAddress) internal {
            ds.selectorToFacetAndPosition[_selector].functionSelectorPosition = _selectorPosition;
            ds.facetFunctionSelectors[_facetAddress].functionSelectors.push(_selector);
            ds.selectorToFacetAndPosition[_selector].facetAddress = _facetAddress;
        }
        function removeFunction(DiamondStorage storage ds, address _facetAddress, bytes4 _selector) internal {        
            require(_facetAddress != address(0), "LibDiamondCut: Can't remove function that doesn't exist");
            // an immutable function is a function defined directly in a diamond
            require(_facetAddress != address(this), "LibDiamondCut: Can't remove immutable function");
            // replace selector with last selector, then delete last selector
            uint256 selectorPosition = ds.selectorToFacetAndPosition[_selector].functionSelectorPosition;
            uint256 lastSelectorPosition = ds.facetFunctionSelectors[_facetAddress].functionSelectors.length - 1;
            // if not the same then replace _selector with lastSelector
            if (selectorPosition != lastSelectorPosition) {
                bytes4 lastSelector = ds.facetFunctionSelectors[_facetAddress].functionSelectors[lastSelectorPosition];
                ds.facetFunctionSelectors[_facetAddress].functionSelectors[selectorPosition] = lastSelector;
                ds.selectorToFacetAndPosition[lastSelector].functionSelectorPosition = uint96(selectorPosition);
            }
            // delete the last selector
            ds.facetFunctionSelectors[_facetAddress].functionSelectors.pop();
            delete ds.selectorToFacetAndPosition[_selector];
            // if no more selectors for facet address then delete the facet address
            if (lastSelectorPosition == 0) {
                // replace facet address with last facet address and delete last facet address
                uint256 lastFacetAddressPosition = ds.facetAddresses.length - 1;
                uint256 facetAddressPosition = ds.facetFunctionSelectors[_facetAddress].facetAddressPosition;
                if (facetAddressPosition != lastFacetAddressPosition) {
                    address lastFacetAddress = ds.facetAddresses[lastFacetAddressPosition];
                    ds.facetAddresses[facetAddressPosition] = lastFacetAddress;
                    ds.facetFunctionSelectors[lastFacetAddress].facetAddressPosition = facetAddressPosition;
                }
                ds.facetAddresses.pop();
                delete ds.facetFunctionSelectors[_facetAddress].facetAddressPosition;
            }
        }
        function initializeDiamondCut(address _init, bytes memory _calldata) internal {
            if (_init == address(0)) {
                require(_calldata.length == 0, "LibDiamondCut: _init is address(0) but_calldata is not empty");
            } else {
                require(_calldata.length > 0, "LibDiamondCut: _calldata is empty but _init is not address(0)");
                if (_init != address(this)) {
                    enforceHasContractCode(_init, "LibDiamondCut: _init address has no code");
                }
                (bool success, bytes memory error) = _init.delegatecall(_calldata);
                if (!success) {
                    if (error.length > 0) {
                        // bubble up the error
                        revert(string(error));
                    } else {
                        revert("LibDiamondCut: _init function reverted");
                    }
                }
            }
        }
        function enforceHasContractCode(address _contract, string memory _errorMessage) internal view {
            uint256 contractSize;
            assembly {
                contractSize := extcodesize(_contract)
            }
            require(contractSize > 0, _errorMessage);
        }
    }
    // SPDX-License-Identifier: MIT
    pragma solidity ^0.8.4;
    /// @notice Safe integer casting library that reverts on overflow.
    /// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/SafeCastLib.sol)
    /// @author Modified from OpenZeppelin (https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/utils/math/SafeCast.sol)
    library SafeCastLib {
        /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/
        /*                       CUSTOM ERRORS                        */
        /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/
        error Overflow();
        /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/
        /*          UNSIGNED INTEGER SAFE CASTING OPERATIONS          */
        /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/
        function toUint8(uint256 x) internal pure returns (uint8) {
            if (x >= 1 << 8) _revertOverflow();
            return uint8(x);
        }
        function toUint16(uint256 x) internal pure returns (uint16) {
            if (x >= 1 << 16) _revertOverflow();
            return uint16(x);
        }
        function toUint24(uint256 x) internal pure returns (uint24) {
            if (x >= 1 << 24) _revertOverflow();
            return uint24(x);
        }
        function toUint32(uint256 x) internal pure returns (uint32) {
            if (x >= 1 << 32) _revertOverflow();
            return uint32(x);
        }
        function toUint40(uint256 x) internal pure returns (uint40) {
            if (x >= 1 << 40) _revertOverflow();
            return uint40(x);
        }
        function toUint48(uint256 x) internal pure returns (uint48) {
            if (x >= 1 << 48) _revertOverflow();
            return uint48(x);
        }
        function toUint56(uint256 x) internal pure returns (uint56) {
            if (x >= 1 << 56) _revertOverflow();
            return uint56(x);
        }
        function toUint64(uint256 x) internal pure returns (uint64) {
            if (x >= 1 << 64) _revertOverflow();
            return uint64(x);
        }
        function toUint72(uint256 x) internal pure returns (uint72) {
            if (x >= 1 << 72) _revertOverflow();
            return uint72(x);
        }
        function toUint80(uint256 x) internal pure returns (uint80) {
            if (x >= 1 << 80) _revertOverflow();
            return uint80(x);
        }
        function toUint88(uint256 x) internal pure returns (uint88) {
            if (x >= 1 << 88) _revertOverflow();
            return uint88(x);
        }
        function toUint96(uint256 x) internal pure returns (uint96) {
            if (x >= 1 << 96) _revertOverflow();
            return uint96(x);
        }
        function toUint104(uint256 x) internal pure returns (uint104) {
            if (x >= 1 << 104) _revertOverflow();
            return uint104(x);
        }
        function toUint112(uint256 x) internal pure returns (uint112) {
            if (x >= 1 << 112) _revertOverflow();
            return uint112(x);
        }
        function toUint120(uint256 x) internal pure returns (uint120) {
            if (x >= 1 << 120) _revertOverflow();
            return uint120(x);
        }
        function toUint128(uint256 x) internal pure returns (uint128) {
            if (x >= 1 << 128) _revertOverflow();
            return uint128(x);
        }
        function toUint136(uint256 x) internal pure returns (uint136) {
            if (x >= 1 << 136) _revertOverflow();
            return uint136(x);
        }
        function toUint144(uint256 x) internal pure returns (uint144) {
            if (x >= 1 << 144) _revertOverflow();
            return uint144(x);
        }
        function toUint152(uint256 x) internal pure returns (uint152) {
            if (x >= 1 << 152) _revertOverflow();
            return uint152(x);
        }
        function toUint160(uint256 x) internal pure returns (uint160) {
            if (x >= 1 << 160) _revertOverflow();
            return uint160(x);
        }
        function toUint168(uint256 x) internal pure returns (uint168) {
            if (x >= 1 << 168) _revertOverflow();
            return uint168(x);
        }
        function toUint176(uint256 x) internal pure returns (uint176) {
            if (x >= 1 << 176) _revertOverflow();
            return uint176(x);
        }
        function toUint184(uint256 x) internal pure returns (uint184) {
            if (x >= 1 << 184) _revertOverflow();
            return uint184(x);
        }
        function toUint192(uint256 x) internal pure returns (uint192) {
            if (x >= 1 << 192) _revertOverflow();
            return uint192(x);
        }
        function toUint200(uint256 x) internal pure returns (uint200) {
            if (x >= 1 << 200) _revertOverflow();
            return uint200(x);
        }
        function toUint208(uint256 x) internal pure returns (uint208) {
            if (x >= 1 << 208) _revertOverflow();
            return uint208(x);
        }
        function toUint216(uint256 x) internal pure returns (uint216) {
            if (x >= 1 << 216) _revertOverflow();
            return uint216(x);
        }
        function toUint224(uint256 x) internal pure returns (uint224) {
            if (x >= 1 << 224) _revertOverflow();
            return uint224(x);
        }
        function toUint232(uint256 x) internal pure returns (uint232) {
            if (x >= 1 << 232) _revertOverflow();
            return uint232(x);
        }
        function toUint240(uint256 x) internal pure returns (uint240) {
            if (x >= 1 << 240) _revertOverflow();
            return uint240(x);
        }
        function toUint248(uint256 x) internal pure returns (uint248) {
            if (x >= 1 << 248) _revertOverflow();
            return uint248(x);
        }
        /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/
        /*           SIGNED INTEGER SAFE CASTING OPERATIONS           */
        /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/
        function toInt8(int256 x) internal pure returns (int8) {
            int8 y = int8(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt16(int256 x) internal pure returns (int16) {
            int16 y = int16(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt24(int256 x) internal pure returns (int24) {
            int24 y = int24(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt32(int256 x) internal pure returns (int32) {
            int32 y = int32(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt40(int256 x) internal pure returns (int40) {
            int40 y = int40(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt48(int256 x) internal pure returns (int48) {
            int48 y = int48(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt56(int256 x) internal pure returns (int56) {
            int56 y = int56(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt64(int256 x) internal pure returns (int64) {
            int64 y = int64(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt72(int256 x) internal pure returns (int72) {
            int72 y = int72(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt80(int256 x) internal pure returns (int80) {
            int80 y = int80(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt88(int256 x) internal pure returns (int88) {
            int88 y = int88(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt96(int256 x) internal pure returns (int96) {
            int96 y = int96(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt104(int256 x) internal pure returns (int104) {
            int104 y = int104(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt112(int256 x) internal pure returns (int112) {
            int112 y = int112(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt120(int256 x) internal pure returns (int120) {
            int120 y = int120(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt128(int256 x) internal pure returns (int128) {
            int128 y = int128(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt136(int256 x) internal pure returns (int136) {
            int136 y = int136(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt144(int256 x) internal pure returns (int144) {
            int144 y = int144(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt152(int256 x) internal pure returns (int152) {
            int152 y = int152(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt160(int256 x) internal pure returns (int160) {
            int160 y = int160(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt168(int256 x) internal pure returns (int168) {
            int168 y = int168(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt176(int256 x) internal pure returns (int176) {
            int176 y = int176(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt184(int256 x) internal pure returns (int184) {
            int184 y = int184(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt192(int256 x) internal pure returns (int192) {
            int192 y = int192(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt200(int256 x) internal pure returns (int200) {
            int200 y = int200(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt208(int256 x) internal pure returns (int208) {
            int208 y = int208(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt216(int256 x) internal pure returns (int216) {
            int216 y = int216(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt224(int256 x) internal pure returns (int224) {
            int224 y = int224(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt232(int256 x) internal pure returns (int232) {
            int232 y = int232(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt240(int256 x) internal pure returns (int240) {
            int240 y = int240(x);
            if (x != y) _revertOverflow();
            return y;
        }
        function toInt248(int256 x) internal pure returns (int248) {
            int248 y = int248(x);
            if (x != y) _revertOverflow();
            return y;
        }
        /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/
        /*                      PRIVATE HELPERS                       */
        /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/
        function _revertOverflow() private pure {
            /// @solidity memory-safe-assembly
            assembly {
                // Store the function selector of `Overflow()`.
                mstore(0x00, 0x35278d12)
                // Revert with (offset, size).
                revert(0x1c, 0x04)
            }
        }
    }