ETH Price: $2,540.44 (+0.38%)

Transaction Decoder

Block:
20137008 at Jun-21-2024 02:07:11 AM +UTC
Transaction Fee:
0.000654458538063776 ETH $1.66
Gas Used:
218,464 Gas / 2.995727159 Gwei

Emitted Events:

197 ERC1967Proxy.0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef( 0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef, 0x000000000000000000000000f396126d4ce9e8b8f8c70b5173b46834aeef66a2, 0x000000000000000000000000a033ff09f2da45f0e9ae495f525363722df42b2a, 0000000000000000000000000000000000000000000000004578ffa469f27434 )
198 GnosisSafeProxy.0x3d0ce9bfc3ed7d6862dbb28b2dea94561fe714a1b4d019aa8af39730d1ad7c3d( 0x3d0ce9bfc3ed7d6862dbb28b2dea94561fe714a1b4d019aa8af39730d1ad7c3d, 0x0000000000000000000000006774bcbd5cecef1336b5300fb5186a12ddd8b367, 00000000000000000000000000000000000000000000000000003110989ca340 )
199 TransparentUpgradeableProxy.0x69cfcb8e6d4192b8aba9902243912587f37e550d75c1fa801491fce26717f37e( 0x69cfcb8e6d4192b8aba9902243912587f37e550d75c1fa801491fce26717f37e, 0x0000000000000000000000007885bcbd5cecef1336b5300fb5186a12ddd8c478, 0x000000000000000000000000781e90f1c8fc4611c9b7497c3b47f99ef6969cbc, 0000000000000000000000000000000000000000000000000000000000000000, 00000000000000000000000000000000000000000000000000000000000d56d6, 000000000000000000000000000000000000000000000000000000000002bf20, 0000000000000000000000000000000000000000000000000000000000000080, 00000000000000000000000000000000000000000000000000000000000001c4, 8ef1332e000000000000000000000000a033ff09f2da45f0e9ae495f52536372, 2df42b2a0000000000000000000000009ebf2f33526cd571f8b2ad312492cb65, 0870cfd600000000000000000000000000000000000000000000000000000000, 0000000000000000000000000000000000000000000000000000000000000000, 000d56d600000000000000000000000000000000000000000000000000000000, 000000a000000000000000000000000000000000000000000000000000000000, 000000e48431f5c1000000000000000000000000d9a442856c234a39a81a089c, 06451ebaa4306a72000000000000000000000000c4d46e8402f476f269c37967, 7c99f18e22ea030e000000000000000000000000f396126d4ce9e8b8f8c70b51, 73b46834aeef66a2000000000000000000000000f396126d4ce9e8b8f8c70b51, 73b46834aeef66a2000000000000000000000000000000000000000000000000, 4578ffa469f27434000000000000000000000000000000000000000000000000, 00000000000000c0000000000000000000000000000000000000000000000000, 0000000000000000000000000000000000000000000000000000000000000000, 0000000000000000000000000000000000000000000000000000000000000000 )
200 TransparentUpgradeableProxy.0x104371f3b442861a2a7b82a070afbbaab748bb13757bf47769e170e37809ec1e( 0x104371f3b442861a2a7b82a070afbbaab748bb13757bf47769e170e37809ec1e, 0x000000000000000000000000a033ff09f2da45f0e9ae495f525363722df42b2a, 0x0000000000000000000000009ebf2f33526cd571f8b2ad312492cb650870cfd6, 0000000000000000000000000000000000000000000000000000000000000000, 00000000000000000000000000000000000000000000000000000000000d56d6, 000000000000000000000000000000000000000000000000000000000002bf20, 0000000000000000000000000000000000000000000000000000000000000080, 00000000000000000000000000000000000000000000000000000000000000e4, 8431f5c1000000000000000000000000d9a442856c234a39a81a089c06451eba, a4306a72000000000000000000000000c4d46e8402f476f269c379677c99f18e, 22ea030e000000000000000000000000f396126d4ce9e8b8f8c70b5173b46834, aeef66a2000000000000000000000000f396126d4ce9e8b8f8c70b5173b46834, aeef66a20000000000000000000000000000000000000000000000004578ffa4, 69f2743400000000000000000000000000000000000000000000000000000000, 000000c000000000000000000000000000000000000000000000000000000000, 0000000000000000000000000000000000000000000000000000000000000000 )
201 TransparentUpgradeableProxy.0x31cd3b976e4d654022bf95c68a2ce53f1d5d94afabe0454d2832208eeb40af25( 0x31cd3b976e4d654022bf95c68a2ce53f1d5d94afabe0454d2832208eeb40af25, 0x000000000000000000000000d9a442856c234a39a81a089c06451ebaa4306a72, 0x000000000000000000000000c4d46e8402f476f269c379677c99f18e22ea030e, 0x000000000000000000000000f396126d4ce9e8b8f8c70b5173b46834aeef66a2, 000000000000000000000000f396126d4ce9e8b8f8c70b5173b46834aeef66a2, 0000000000000000000000000000000000000000000000004578ffa469f27434, 0000000000000000000000000000000000000000000000000000000000000060, 0000000000000000000000000000000000000000000000000000000000000000 )

Account State Difference:

  Address   Before After State Difference Code
0x0d7E906B...54E39AF9B
(Scroll: L1 Message Queue Proxy)
4.017534876321738712 Eth4.017545799521738712 Eth0.0000109232
0x6774Bcbd...2DDD8b367
(Scroll: L1 Scroll Messenger Proxy)
0x8FA3b457...5867eEB48 992.420577347215584627 Eth992.420631294565224627 Eth0.00005394734964
0xD9A44285...Aa4306a72
0xF396126d...4AEeF66A2
0.106686391314443109 Eth
Nonce: 925
0.105977985426739333 Eth
Nonce: 926
0.000708405887703776

Execution Trace

ETH 0.00006473681946 TransparentUpgradeableProxy.21425ee0( )
  • ETH 0.00006473681946 L1GatewayRouter.depositERC20( _token=0xD9A442856C234a39a81a089C06451EBAa4306a72, _amount=5006032067439522868, _gasLimit=180000 )
    • ETH 0.00006473681946 TransparentUpgradeableProxy.0aea8c26( )
      • ETH 0.00006473681946 L1CustomERC20Gateway.depositERC20AndCall( _token=0xD9A442856C234a39a81a089C06451EBAa4306a72, _to=0xF396126d4ce9e8b8f8c70b5173b46834AEeF66A2, _amount=5006032067439522868, _data=0x000000000000000000000000F396126D4CE9E8B8F8C70B5173B46834AEEF66A200000000000000000000000000000000000000000000000000000000000000400000000000000000000000000000000000000000000000000000000000000000, _gasLimit=180000 )
        • TransparentUpgradeableProxy.c52a3bbc( )
          • L1GatewayRouter.requestERC20( _sender=0xF396126d4ce9e8b8f8c70b5173b46834AEeF66A2, _token=0xD9A442856C234a39a81a089C06451EBAa4306a72, _amount=5006032067439522868 ) => ( 5006032067439522868 )
            • ERC1967Proxy.70a08231( )
            • ERC1967Proxy.23b872dd( )
            • ERC1967Proxy.70a08231( )
            • ETH 0.00006473681946 TransparentUpgradeableProxy.5f7b1577( )
              • ETH 0.00006473681946 L1ScrollMessenger.sendMessage( _to=0x9eBf2f33526CD571f8b2ad312492cb650870CFd6, _value=0, _message=0x8431F5C1000000000000000000000000D9A442856C234A39A81A089C06451EBAA4306A72000000000000000000000000C4D46E8402F476F269C379677C99F18E22EA030E000000000000000000000000F396126D4CE9E8B8F8C70B5173B46834AEEF66A2000000000000000000000000F396126D4CE9E8B8F8C70B5173B46834AEEF66A20000000000000000000000000000000000000000000000004578FFA469F2743400000000000000000000000000000000000000000000000000000000000000C00000000000000000000000000000000000000000000000000000000000000000, _gasLimit=180000, _refundAddress=0xF396126d4ce9e8b8f8c70b5173b46834AEeF66A2 )
                • TransparentUpgradeableProxy.STATICCALL( )
                • TransparentUpgradeableProxy.d7704bae( )
                • ETH 0.00005394734964 GnosisSafeProxy.CALL( )
                • TransparentUpgradeableProxy.9b159782( )
                • ETH 0.00001078946982 0xf396126d4ce9e8b8f8c70b5173b46834aeef66a2.CALL( )
                  File 1 of 9: TransparentUpgradeableProxy
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.9.0) (proxy/transparent/TransparentUpgradeableProxy.sol)
                  pragma solidity ^0.8.0;
                  import "../ERC1967/ERC1967Proxy.sol";
                  /**
                   * @dev Interface for {TransparentUpgradeableProxy}. In order to implement transparency, {TransparentUpgradeableProxy}
                   * does not implement this interface directly, and some of its functions are implemented by an internal dispatch
                   * mechanism. The compiler is unaware that these functions are implemented by {TransparentUpgradeableProxy} and will not
                   * include them in the ABI so this interface must be used to interact with it.
                   */
                  interface ITransparentUpgradeableProxy is IERC1967 {
                      function admin() external view returns (address);
                      function implementation() external view returns (address);
                      function changeAdmin(address) external;
                      function upgradeTo(address) external;
                      function upgradeToAndCall(address, bytes memory) external payable;
                  }
                  /**
                   * @dev This contract implements a proxy that is upgradeable by an admin.
                   *
                   * To avoid https://medium.com/nomic-labs-blog/malicious-backdoors-in-ethereum-proxies-62629adf3357[proxy selector
                   * clashing], which can potentially be used in an attack, this contract uses the
                   * https://blog.openzeppelin.com/the-transparent-proxy-pattern/[transparent proxy pattern]. This pattern implies two
                   * things that go hand in hand:
                   *
                   * 1. If any account other than the admin calls the proxy, the call will be forwarded to the implementation, even if
                   * that call matches one of the admin functions exposed by the proxy itself.
                   * 2. If the admin calls the proxy, it can access the admin functions, but its calls will never be forwarded to the
                   * implementation. If the admin tries to call a function on the implementation it will fail with an error that says
                   * "admin cannot fallback to proxy target".
                   *
                   * These properties mean that the admin account can only be used for admin actions like upgrading the proxy or changing
                   * the admin, so it's best if it's a dedicated account that is not used for anything else. This will avoid headaches due
                   * to sudden errors when trying to call a function from the proxy implementation.
                   *
                   * Our recommendation is for the dedicated account to be an instance of the {ProxyAdmin} contract. If set up this way,
                   * you should think of the `ProxyAdmin` instance as the real administrative interface of your proxy.
                   *
                   * NOTE: The real interface of this proxy is that defined in `ITransparentUpgradeableProxy`. This contract does not
                   * inherit from that interface, and instead the admin functions are implicitly implemented using a custom dispatch
                   * mechanism in `_fallback`. Consequently, the compiler will not produce an ABI for this contract. This is necessary to
                   * fully implement transparency without decoding reverts caused by selector clashes between the proxy and the
                   * implementation.
                   *
                   * WARNING: It is not recommended to extend this contract to add additional external functions. If you do so, the compiler
                   * will not check that there are no selector conflicts, due to the note above. A selector clash between any new function
                   * and the functions declared in {ITransparentUpgradeableProxy} will be resolved in favor of the new one. This could
                   * render the admin operations inaccessible, which could prevent upgradeability. Transparency may also be compromised.
                   */
                  contract TransparentUpgradeableProxy is ERC1967Proxy {
                      /**
                       * @dev Initializes an upgradeable proxy managed by `_admin`, backed by the implementation at `_logic`, and
                       * optionally initialized with `_data` as explained in {ERC1967Proxy-constructor}.
                       */
                      constructor(address _logic, address admin_, bytes memory _data) payable ERC1967Proxy(_logic, _data) {
                          _changeAdmin(admin_);
                      }
                      /**
                       * @dev Modifier used internally that will delegate the call to the implementation unless the sender is the admin.
                       *
                       * CAUTION: This modifier is deprecated, as it could cause issues if the modified function has arguments, and the
                       * implementation provides a function with the same selector.
                       */
                      modifier ifAdmin() {
                          if (msg.sender == _getAdmin()) {
                              _;
                          } else {
                              _fallback();
                          }
                      }
                      /**
                       * @dev If caller is the admin process the call internally, otherwise transparently fallback to the proxy behavior
                       */
                      function _fallback() internal virtual override {
                          if (msg.sender == _getAdmin()) {
                              bytes memory ret;
                              bytes4 selector = msg.sig;
                              if (selector == ITransparentUpgradeableProxy.upgradeTo.selector) {
                                  ret = _dispatchUpgradeTo();
                              } else if (selector == ITransparentUpgradeableProxy.upgradeToAndCall.selector) {
                                  ret = _dispatchUpgradeToAndCall();
                              } else if (selector == ITransparentUpgradeableProxy.changeAdmin.selector) {
                                  ret = _dispatchChangeAdmin();
                              } else if (selector == ITransparentUpgradeableProxy.admin.selector) {
                                  ret = _dispatchAdmin();
                              } else if (selector == ITransparentUpgradeableProxy.implementation.selector) {
                                  ret = _dispatchImplementation();
                              } else {
                                  revert("TransparentUpgradeableProxy: admin cannot fallback to proxy target");
                              }
                              assembly {
                                  return(add(ret, 0x20), mload(ret))
                              }
                          } else {
                              super._fallback();
                          }
                      }
                      /**
                       * @dev Returns the current admin.
                       *
                       * TIP: To get this value clients can read directly from the storage slot shown below (specified by EIP1967) using the
                       * https://eth.wiki/json-rpc/API#eth_getstorageat[`eth_getStorageAt`] RPC call.
                       * `0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103`
                       */
                      function _dispatchAdmin() private returns (bytes memory) {
                          _requireZeroValue();
                          address admin = _getAdmin();
                          return abi.encode(admin);
                      }
                      /**
                       * @dev Returns the current implementation.
                       *
                       * TIP: To get this value clients can read directly from the storage slot shown below (specified by EIP1967) using the
                       * https://eth.wiki/json-rpc/API#eth_getstorageat[`eth_getStorageAt`] RPC call.
                       * `0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc`
                       */
                      function _dispatchImplementation() private returns (bytes memory) {
                          _requireZeroValue();
                          address implementation = _implementation();
                          return abi.encode(implementation);
                      }
                      /**
                       * @dev Changes the admin of the proxy.
                       *
                       * Emits an {AdminChanged} event.
                       */
                      function _dispatchChangeAdmin() private returns (bytes memory) {
                          _requireZeroValue();
                          address newAdmin = abi.decode(msg.data[4:], (address));
                          _changeAdmin(newAdmin);
                          return "";
                      }
                      /**
                       * @dev Upgrade the implementation of the proxy.
                       */
                      function _dispatchUpgradeTo() private returns (bytes memory) {
                          _requireZeroValue();
                          address newImplementation = abi.decode(msg.data[4:], (address));
                          _upgradeToAndCall(newImplementation, bytes(""), false);
                          return "";
                      }
                      /**
                       * @dev Upgrade the implementation of the proxy, and then call a function from the new implementation as specified
                       * by `data`, which should be an encoded function call. This is useful to initialize new storage variables in the
                       * proxied contract.
                       */
                      function _dispatchUpgradeToAndCall() private returns (bytes memory) {
                          (address newImplementation, bytes memory data) = abi.decode(msg.data[4:], (address, bytes));
                          _upgradeToAndCall(newImplementation, data, true);
                          return "";
                      }
                      /**
                       * @dev Returns the current admin.
                       *
                       * CAUTION: This function is deprecated. Use {ERC1967Upgrade-_getAdmin} instead.
                       */
                      function _admin() internal view virtual returns (address) {
                          return _getAdmin();
                      }
                      /**
                       * @dev To keep this contract fully transparent, all `ifAdmin` functions must be payable. This helper is here to
                       * emulate some proxy functions being non-payable while still allowing value to pass through.
                       */
                      function _requireZeroValue() private {
                          require(msg.value == 0);
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.7.0) (proxy/ERC1967/ERC1967Proxy.sol)
                  pragma solidity ^0.8.0;
                  import "../Proxy.sol";
                  import "./ERC1967Upgrade.sol";
                  /**
                   * @dev This contract implements an upgradeable proxy. It is upgradeable because calls are delegated to an
                   * implementation address that can be changed. This address is stored in storage in the location specified by
                   * https://eips.ethereum.org/EIPS/eip-1967[EIP1967], so that it doesn't conflict with the storage layout of the
                   * implementation behind the proxy.
                   */
                  contract ERC1967Proxy is Proxy, ERC1967Upgrade {
                      /**
                       * @dev Initializes the upgradeable proxy with an initial implementation specified by `_logic`.
                       *
                       * If `_data` is nonempty, it's used as data in a delegate call to `_logic`. This will typically be an encoded
                       * function call, and allows initializing the storage of the proxy like a Solidity constructor.
                       */
                      constructor(address _logic, bytes memory _data) payable {
                          _upgradeToAndCall(_logic, _data, false);
                      }
                      /**
                       * @dev Returns the current implementation address.
                       */
                      function _implementation() internal view virtual override returns (address impl) {
                          return ERC1967Upgrade._getImplementation();
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.6.0) (proxy/Proxy.sol)
                  pragma solidity ^0.8.0;
                  /**
                   * @dev This abstract contract provides a fallback function that delegates all calls to another contract using the EVM
                   * instruction `delegatecall`. We refer to the second contract as the _implementation_ behind the proxy, and it has to
                   * be specified by overriding the virtual {_implementation} function.
                   *
                   * Additionally, delegation to the implementation can be triggered manually through the {_fallback} function, or to a
                   * different contract through the {_delegate} function.
                   *
                   * The success and return data of the delegated call will be returned back to the caller of the proxy.
                   */
                  abstract contract Proxy {
                      /**
                       * @dev Delegates the current call to `implementation`.
                       *
                       * This function does not return to its internal call site, it will return directly to the external caller.
                       */
                      function _delegate(address implementation) internal virtual {
                          assembly {
                              // Copy msg.data. We take full control of memory in this inline assembly
                              // block because it will not return to Solidity code. We overwrite the
                              // Solidity scratch pad at memory position 0.
                              calldatacopy(0, 0, calldatasize())
                              // Call the implementation.
                              // out and outsize are 0 because we don't know the size yet.
                              let result := delegatecall(gas(), implementation, 0, calldatasize(), 0, 0)
                              // Copy the returned data.
                              returndatacopy(0, 0, returndatasize())
                              switch result
                              // delegatecall returns 0 on error.
                              case 0 {
                                  revert(0, returndatasize())
                              }
                              default {
                                  return(0, returndatasize())
                              }
                          }
                      }
                      /**
                       * @dev This is a virtual function that should be overridden so it returns the address to which the fallback function
                       * and {_fallback} should delegate.
                       */
                      function _implementation() internal view virtual returns (address);
                      /**
                       * @dev Delegates the current call to the address returned by `_implementation()`.
                       *
                       * This function does not return to its internal call site, it will return directly to the external caller.
                       */
                      function _fallback() internal virtual {
                          _beforeFallback();
                          _delegate(_implementation());
                      }
                      /**
                       * @dev Fallback function that delegates calls to the address returned by `_implementation()`. Will run if no other
                       * function in the contract matches the call data.
                       */
                      fallback() external payable virtual {
                          _fallback();
                      }
                      /**
                       * @dev Fallback function that delegates calls to the address returned by `_implementation()`. Will run if call data
                       * is empty.
                       */
                      receive() external payable virtual {
                          _fallback();
                      }
                      /**
                       * @dev Hook that is called before falling back to the implementation. Can happen as part of a manual `_fallback`
                       * call, or as part of the Solidity `fallback` or `receive` functions.
                       *
                       * If overridden should call `super._beforeFallback()`.
                       */
                      function _beforeFallback() internal virtual {}
                  }
                  // 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 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) (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 (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) (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
                          }
                      }
                  }
                  

                  File 2 of 9: ERC1967Proxy
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v5.0.0) (proxy/ERC1967/ERC1967Proxy.sol)
                  pragma solidity ^0.8.20;
                  import {Proxy} from "../Proxy.sol";
                  import {ERC1967Utils} from "./ERC1967Utils.sol";
                  /**
                   * @dev This contract implements an upgradeable proxy. It is upgradeable because calls are delegated to an
                   * implementation address that can be changed. This address is stored in storage in the location specified by
                   * https://eips.ethereum.org/EIPS/eip-1967[EIP1967], so that it doesn't conflict with the storage layout of the
                   * implementation behind the proxy.
                   */
                  contract ERC1967Proxy is Proxy {
                      /**
                       * @dev Initializes the upgradeable proxy with an initial implementation specified by `implementation`.
                       *
                       * If `_data` is nonempty, it's used as data in a delegate call to `implementation`. This will typically be an
                       * encoded function call, and allows initializing the storage of the proxy like a Solidity constructor.
                       *
                       * Requirements:
                       *
                       * - If `data` is empty, `msg.value` must be zero.
                       */
                      constructor(address implementation, bytes memory _data) payable {
                          ERC1967Utils.upgradeToAndCall(implementation, _data);
                      }
                      /**
                       * @dev Returns the current implementation address.
                       *
                       * TIP: To get this value clients can read directly from the storage slot shown below (specified by EIP1967) using
                       * the https://eth.wiki/json-rpc/API#eth_getstorageat[`eth_getStorageAt`] RPC call.
                       * `0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc`
                       */
                      function _implementation() internal view virtual override returns (address) {
                          return ERC1967Utils.getImplementation();
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v5.0.0) (proxy/Proxy.sol)
                  pragma solidity ^0.8.20;
                  /**
                   * @dev This abstract contract provides a fallback function that delegates all calls to another contract using the EVM
                   * instruction `delegatecall`. We refer to the second contract as the _implementation_ behind the proxy, and it has to
                   * be specified by overriding the virtual {_implementation} function.
                   *
                   * Additionally, delegation to the implementation can be triggered manually through the {_fallback} function, or to a
                   * different contract through the {_delegate} function.
                   *
                   * The success and return data of the delegated call will be returned back to the caller of the proxy.
                   */
                  abstract contract Proxy {
                      /**
                       * @dev Delegates the current call to `implementation`.
                       *
                       * This function does not return to its internal call site, it will return directly to the external caller.
                       */
                      function _delegate(address implementation) internal virtual {
                          assembly {
                              // Copy msg.data. We take full control of memory in this inline assembly
                              // block because it will not return to Solidity code. We overwrite the
                              // Solidity scratch pad at memory position 0.
                              calldatacopy(0, 0, calldatasize())
                              // Call the implementation.
                              // out and outsize are 0 because we don't know the size yet.
                              let result := delegatecall(gas(), implementation, 0, calldatasize(), 0, 0)
                              // Copy the returned data.
                              returndatacopy(0, 0, returndatasize())
                              switch result
                              // delegatecall returns 0 on error.
                              case 0 {
                                  revert(0, returndatasize())
                              }
                              default {
                                  return(0, returndatasize())
                              }
                          }
                      }
                      /**
                       * @dev This is a virtual function that should be overridden so it returns the address to which the fallback
                       * function and {_fallback} should delegate.
                       */
                      function _implementation() internal view virtual returns (address);
                      /**
                       * @dev Delegates the current call to the address returned by `_implementation()`.
                       *
                       * This function does not return to its internal call site, it will return directly to the external caller.
                       */
                      function _fallback() internal virtual {
                          _delegate(_implementation());
                      }
                      /**
                       * @dev Fallback function that delegates calls to the address returned by `_implementation()`. Will run if no other
                       * function in the contract matches the call data.
                       */
                      fallback() external payable virtual {
                          _fallback();
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v5.0.0) (proxy/ERC1967/ERC1967Utils.sol)
                  pragma solidity ^0.8.20;
                  import {IBeacon} from "../beacon/IBeacon.sol";
                  import {Address} from "../../utils/Address.sol";
                  import {StorageSlot} from "../../utils/StorageSlot.sol";
                  /**
                   * @dev This abstract contract provides getters and event emitting update functions for
                   * https://eips.ethereum.org/EIPS/eip-1967[EIP1967] slots.
                   */
                  library ERC1967Utils {
                      // We re-declare ERC-1967 events here because they can't be used directly from IERC1967.
                      // This will be fixed in Solidity 0.8.21. At that point we should remove these events.
                      /**
                       * @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);
                      /**
                       * @dev Storage slot with the address of the current implementation.
                       * This is the keccak-256 hash of "eip1967.proxy.implementation" subtracted by 1.
                       */
                      // solhint-disable-next-line private-vars-leading-underscore
                      bytes32 internal constant IMPLEMENTATION_SLOT = 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc;
                      /**
                       * @dev The `implementation` of the proxy is invalid.
                       */
                      error ERC1967InvalidImplementation(address implementation);
                      /**
                       * @dev The `admin` of the proxy is invalid.
                       */
                      error ERC1967InvalidAdmin(address admin);
                      /**
                       * @dev The `beacon` of the proxy is invalid.
                       */
                      error ERC1967InvalidBeacon(address beacon);
                      /**
                       * @dev An upgrade function sees `msg.value > 0` that may be lost.
                       */
                      error ERC1967NonPayable();
                      /**
                       * @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 {
                          if (newImplementation.code.length == 0) {
                              revert ERC1967InvalidImplementation(newImplementation);
                          }
                          StorageSlot.getAddressSlot(IMPLEMENTATION_SLOT).value = newImplementation;
                      }
                      /**
                       * @dev Performs implementation upgrade with additional setup call if data is nonempty.
                       * This function is payable only if the setup call is performed, otherwise `msg.value` is rejected
                       * to avoid stuck value in the contract.
                       *
                       * Emits an {IERC1967-Upgraded} event.
                       */
                      function upgradeToAndCall(address newImplementation, bytes memory data) internal {
                          _setImplementation(newImplementation);
                          emit Upgraded(newImplementation);
                          if (data.length > 0) {
                              Address.functionDelegateCall(newImplementation, data);
                          } else {
                              _checkNonPayable();
                          }
                      }
                      /**
                       * @dev Storage slot with the admin of the contract.
                       * This is the keccak-256 hash of "eip1967.proxy.admin" subtracted by 1.
                       */
                      // solhint-disable-next-line private-vars-leading-underscore
                      bytes32 internal constant ADMIN_SLOT = 0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103;
                      /**
                       * @dev Returns the current admin.
                       *
                       * TIP: To get this value clients can read directly from the storage slot shown below (specified by EIP1967) using
                       * the https://eth.wiki/json-rpc/API#eth_getstorageat[`eth_getStorageAt`] RPC call.
                       * `0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103`
                       */
                      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 {
                          if (newAdmin == address(0)) {
                              revert ERC1967InvalidAdmin(address(0));
                          }
                          StorageSlot.getAddressSlot(ADMIN_SLOT).value = newAdmin;
                      }
                      /**
                       * @dev Changes the admin of the proxy.
                       *
                       * Emits an {IERC1967-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 the keccak-256 hash of "eip1967.proxy.beacon" subtracted by 1.
                       */
                      // solhint-disable-next-line private-vars-leading-underscore
                      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 {
                          if (newBeacon.code.length == 0) {
                              revert ERC1967InvalidBeacon(newBeacon);
                          }
                          StorageSlot.getAddressSlot(BEACON_SLOT).value = newBeacon;
                          address beaconImplementation = IBeacon(newBeacon).implementation();
                          if (beaconImplementation.code.length == 0) {
                              revert ERC1967InvalidImplementation(beaconImplementation);
                          }
                      }
                      /**
                       * @dev Change the beacon and trigger a setup call if data is nonempty.
                       * This function is payable only if the setup call is performed, otherwise `msg.value` is rejected
                       * to avoid stuck value in the contract.
                       *
                       * Emits an {IERC1967-BeaconUpgraded} event.
                       *
                       * CAUTION: Invoking this function has no effect on an instance of {BeaconProxy} since v5, since
                       * it uses an immutable beacon without looking at the value of the ERC-1967 beacon slot for
                       * efficiency.
                       */
                      function upgradeBeaconToAndCall(address newBeacon, bytes memory data) internal {
                          _setBeacon(newBeacon);
                          emit BeaconUpgraded(newBeacon);
                          if (data.length > 0) {
                              Address.functionDelegateCall(IBeacon(newBeacon).implementation(), data);
                          } else {
                              _checkNonPayable();
                          }
                      }
                      /**
                       * @dev Reverts if `msg.value` is not zero. It can be used to avoid `msg.value` stuck in the contract
                       * if an upgrade doesn't perform an initialization call.
                       */
                      function _checkNonPayable() private {
                          if (msg.value > 0) {
                              revert ERC1967NonPayable();
                          }
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v5.0.0) (proxy/beacon/IBeacon.sol)
                  pragma solidity ^0.8.20;
                  /**
                   * @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.
                       *
                       * {UpgradeableBeacon} will check that this address is a contract.
                       */
                      function implementation() external view returns (address);
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v5.0.0) (utils/Address.sol)
                  pragma solidity ^0.8.20;
                  /**
                   * @dev Collection of functions related to the address type
                   */
                  library Address {
                      /**
                       * @dev The ETH balance of the account is not enough to perform the operation.
                       */
                      error AddressInsufficientBalance(address account);
                      /**
                       * @dev There's no code at `target` (it is not a contract).
                       */
                      error AddressEmptyCode(address target);
                      /**
                       * @dev A call to an address target failed. The target may have reverted.
                       */
                      error FailedInnerCall();
                      /**
                       * @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.20/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
                       */
                      function sendValue(address payable recipient, uint256 amount) internal {
                          if (address(this).balance < amount) {
                              revert AddressInsufficientBalance(address(this));
                          }
                          (bool success, ) = recipient.call{value: amount}("");
                          if (!success) {
                              revert FailedInnerCall();
                          }
                      }
                      /**
                       * @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 or custom error, it is bubbled
                       * up by this function (like regular Solidity function calls). However, if
                       * the call reverted with no returned reason, this function reverts with a
                       * {FailedInnerCall} error.
                       *
                       * 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.
                       */
                      function functionCall(address target, bytes memory data) internal returns (bytes memory) {
                          return functionCallWithValue(target, data, 0);
                      }
                      /**
                       * @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`.
                       */
                      function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
                          if (address(this).balance < value) {
                              revert AddressInsufficientBalance(address(this));
                          }
                          (bool success, bytes memory returndata) = target.call{value: value}(data);
                          return verifyCallResultFromTarget(target, success, returndata);
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                       * but performing a static call.
                       */
                      function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
                          (bool success, bytes memory returndata) = target.staticcall(data);
                          return verifyCallResultFromTarget(target, success, returndata);
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                       * but performing a delegate call.
                       */
                      function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
                          (bool success, bytes memory returndata) = target.delegatecall(data);
                          return verifyCallResultFromTarget(target, success, returndata);
                      }
                      /**
                       * @dev Tool to verify that a low level call to smart-contract was successful, and reverts if the target
                       * was not a contract or bubbling up the revert reason (falling back to {FailedInnerCall}) in case of an
                       * unsuccessful call.
                       */
                      function verifyCallResultFromTarget(
                          address target,
                          bool success,
                          bytes memory returndata
                      ) internal view returns (bytes memory) {
                          if (!success) {
                              _revert(returndata);
                          } else {
                              // only check if target is a contract if the call was successful and the return data is empty
                              // otherwise we already know that it was a contract
                              if (returndata.length == 0 && target.code.length == 0) {
                                  revert AddressEmptyCode(target);
                              }
                              return returndata;
                          }
                      }
                      /**
                       * @dev Tool to verify that a low level call was successful, and reverts if it wasn't, either by bubbling the
                       * revert reason or with a default {FailedInnerCall} error.
                       */
                      function verifyCallResult(bool success, bytes memory returndata) internal pure returns (bytes memory) {
                          if (!success) {
                              _revert(returndata);
                          } else {
                              return returndata;
                          }
                      }
                      /**
                       * @dev Reverts with returndata if present. Otherwise reverts with {FailedInnerCall}.
                       */
                      function _revert(bytes memory returndata) 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 FailedInnerCall();
                          }
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v5.0.0) (utils/StorageSlot.sol)
                  // This file was procedurally generated from scripts/generate/templates/StorageSlot.js.
                  pragma solidity ^0.8.20;
                  /**
                   * @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(newImplementation.code.length > 0);
                   *         StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value = newImplementation;
                   *     }
                   * }
                   * ```
                   */
                  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
                          }
                      }
                  }
                  

                  File 3 of 9: GnosisSafeProxy
                  // SPDX-License-Identifier: LGPL-3.0-only
                  pragma solidity >=0.7.0 <0.9.0;
                  
                  /// @title IProxy - Helper interface to access masterCopy of the Proxy on-chain
                  /// @author Richard Meissner - <[email protected]>
                  interface IProxy {
                      function masterCopy() external view returns (address);
                  }
                  
                  /// @title GnosisSafeProxy - Generic proxy contract allows to execute all transactions applying the code of a master contract.
                  /// @author Stefan George - <[email protected]>
                  /// @author Richard Meissner - <[email protected]>
                  contract GnosisSafeProxy {
                      // singleton always needs to be first declared variable, to ensure that it is at the same location in the contracts to which calls are delegated.
                      // To reduce deployment costs this variable is internal and needs to be retrieved via `getStorageAt`
                      address internal singleton;
                  
                      /// @dev Constructor function sets address of singleton contract.
                      /// @param _singleton Singleton address.
                      constructor(address _singleton) {
                          require(_singleton != address(0), "Invalid singleton address provided");
                          singleton = _singleton;
                      }
                  
                      /// @dev Fallback function forwards all transactions and returns all received return data.
                      fallback() external payable {
                          // solhint-disable-next-line no-inline-assembly
                          assembly {
                              let _singleton := and(sload(0), 0xffffffffffffffffffffffffffffffffffffffff)
                              // 0xa619486e == keccak("masterCopy()"). The value is right padded to 32-bytes with 0s
                              if eq(calldataload(0), 0xa619486e00000000000000000000000000000000000000000000000000000000) {
                                  mstore(0, _singleton)
                                  return(0, 0x20)
                              }
                              calldatacopy(0, 0, calldatasize())
                              let success := delegatecall(gas(), _singleton, 0, calldatasize(), 0, 0)
                              returndatacopy(0, 0, returndatasize())
                              if eq(success, 0) {
                                  revert(0, returndatasize())
                              }
                              return(0, returndatasize())
                          }
                      }
                  }
                  
                  /// @title Proxy Factory - Allows to create new proxy contact and execute a message call to the new proxy within one transaction.
                  /// @author Stefan George - <[email protected]>
                  contract GnosisSafeProxyFactory {
                      event ProxyCreation(GnosisSafeProxy proxy, address singleton);
                  
                      /// @dev Allows to create new proxy contact and execute a message call to the new proxy within one transaction.
                      /// @param singleton Address of singleton contract.
                      /// @param data Payload for message call sent to new proxy contract.
                      function createProxy(address singleton, bytes memory data) public returns (GnosisSafeProxy proxy) {
                          proxy = new GnosisSafeProxy(singleton);
                          if (data.length > 0)
                              // solhint-disable-next-line no-inline-assembly
                              assembly {
                                  if eq(call(gas(), proxy, 0, add(data, 0x20), mload(data), 0, 0), 0) {
                                      revert(0, 0)
                                  }
                              }
                          emit ProxyCreation(proxy, singleton);
                      }
                  
                      /// @dev Allows to retrieve the runtime code of a deployed Proxy. This can be used to check that the expected Proxy was deployed.
                      function proxyRuntimeCode() public pure returns (bytes memory) {
                          return type(GnosisSafeProxy).runtimeCode;
                      }
                  
                      /// @dev Allows to retrieve the creation code used for the Proxy deployment. With this it is easily possible to calculate predicted address.
                      function proxyCreationCode() public pure returns (bytes memory) {
                          return type(GnosisSafeProxy).creationCode;
                      }
                  
                      /// @dev Allows to create new proxy contact using CREATE2 but it doesn't run the initializer.
                      ///      This method is only meant as an utility to be called from other methods
                      /// @param _singleton Address of singleton contract.
                      /// @param initializer Payload for message call sent to new proxy contract.
                      /// @param saltNonce Nonce that will be used to generate the salt to calculate the address of the new proxy contract.
                      function deployProxyWithNonce(
                          address _singleton,
                          bytes memory initializer,
                          uint256 saltNonce
                      ) internal returns (GnosisSafeProxy proxy) {
                          // If the initializer changes the proxy address should change too. Hashing the initializer data is cheaper than just concatinating it
                          bytes32 salt = keccak256(abi.encodePacked(keccak256(initializer), saltNonce));
                          bytes memory deploymentData = abi.encodePacked(type(GnosisSafeProxy).creationCode, uint256(uint160(_singleton)));
                          // solhint-disable-next-line no-inline-assembly
                          assembly {
                              proxy := create2(0x0, add(0x20, deploymentData), mload(deploymentData), salt)
                          }
                          require(address(proxy) != address(0), "Create2 call failed");
                      }
                  
                      /// @dev Allows to create new proxy contact and execute a message call to the new proxy within one transaction.
                      /// @param _singleton Address of singleton contract.
                      /// @param initializer Payload for message call sent to new proxy contract.
                      /// @param saltNonce Nonce that will be used to generate the salt to calculate the address of the new proxy contract.
                      function createProxyWithNonce(
                          address _singleton,
                          bytes memory initializer,
                          uint256 saltNonce
                      ) public returns (GnosisSafeProxy proxy) {
                          proxy = deployProxyWithNonce(_singleton, initializer, saltNonce);
                          if (initializer.length > 0)
                              // solhint-disable-next-line no-inline-assembly
                              assembly {
                                  if eq(call(gas(), proxy, 0, add(initializer, 0x20), mload(initializer), 0, 0), 0) {
                                      revert(0, 0)
                                  }
                              }
                          emit ProxyCreation(proxy, _singleton);
                      }
                  
                      /// @dev Allows to create new proxy contact, execute a message call to the new proxy and call a specified callback within one transaction
                      /// @param _singleton Address of singleton contract.
                      /// @param initializer Payload for message call sent to new proxy contract.
                      /// @param saltNonce Nonce that will be used to generate the salt to calculate the address of the new proxy contract.
                      /// @param callback Callback that will be invoced after the new proxy contract has been successfully deployed and initialized.
                      function createProxyWithCallback(
                          address _singleton,
                          bytes memory initializer,
                          uint256 saltNonce,
                          IProxyCreationCallback callback
                      ) public returns (GnosisSafeProxy proxy) {
                          uint256 saltNonceWithCallback = uint256(keccak256(abi.encodePacked(saltNonce, callback)));
                          proxy = createProxyWithNonce(_singleton, initializer, saltNonceWithCallback);
                          if (address(callback) != address(0)) callback.proxyCreated(proxy, _singleton, initializer, saltNonce);
                      }
                  
                      /// @dev Allows to get the address for a new proxy contact created via `createProxyWithNonce`
                      ///      This method is only meant for address calculation purpose when you use an initializer that would revert,
                      ///      therefore the response is returned with a revert. When calling this method set `from` to the address of the proxy factory.
                      /// @param _singleton Address of singleton contract.
                      /// @param initializer Payload for message call sent to new proxy contract.
                      /// @param saltNonce Nonce that will be used to generate the salt to calculate the address of the new proxy contract.
                      function calculateCreateProxyWithNonceAddress(
                          address _singleton,
                          bytes calldata initializer,
                          uint256 saltNonce
                      ) external returns (GnosisSafeProxy proxy) {
                          proxy = deployProxyWithNonce(_singleton, initializer, saltNonce);
                          revert(string(abi.encodePacked(proxy)));
                      }
                  }
                  
                  interface IProxyCreationCallback {
                      function proxyCreated(
                          GnosisSafeProxy proxy,
                          address _singleton,
                          bytes calldata initializer,
                          uint256 saltNonce
                      ) external;
                  }

                  File 4 of 9: TransparentUpgradeableProxy
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.9.0) (proxy/transparent/TransparentUpgradeableProxy.sol)
                  pragma solidity ^0.8.0;
                  import "../ERC1967/ERC1967Proxy.sol";
                  /**
                   * @dev Interface for {TransparentUpgradeableProxy}. In order to implement transparency, {TransparentUpgradeableProxy}
                   * does not implement this interface directly, and some of its functions are implemented by an internal dispatch
                   * mechanism. The compiler is unaware that these functions are implemented by {TransparentUpgradeableProxy} and will not
                   * include them in the ABI so this interface must be used to interact with it.
                   */
                  interface ITransparentUpgradeableProxy is IERC1967 {
                      function admin() external view returns (address);
                      function implementation() external view returns (address);
                      function changeAdmin(address) external;
                      function upgradeTo(address) external;
                      function upgradeToAndCall(address, bytes memory) external payable;
                  }
                  /**
                   * @dev This contract implements a proxy that is upgradeable by an admin.
                   *
                   * To avoid https://medium.com/nomic-labs-blog/malicious-backdoors-in-ethereum-proxies-62629adf3357[proxy selector
                   * clashing], which can potentially be used in an attack, this contract uses the
                   * https://blog.openzeppelin.com/the-transparent-proxy-pattern/[transparent proxy pattern]. This pattern implies two
                   * things that go hand in hand:
                   *
                   * 1. If any account other than the admin calls the proxy, the call will be forwarded to the implementation, even if
                   * that call matches one of the admin functions exposed by the proxy itself.
                   * 2. If the admin calls the proxy, it can access the admin functions, but its calls will never be forwarded to the
                   * implementation. If the admin tries to call a function on the implementation it will fail with an error that says
                   * "admin cannot fallback to proxy target".
                   *
                   * These properties mean that the admin account can only be used for admin actions like upgrading the proxy or changing
                   * the admin, so it's best if it's a dedicated account that is not used for anything else. This will avoid headaches due
                   * to sudden errors when trying to call a function from the proxy implementation.
                   *
                   * Our recommendation is for the dedicated account to be an instance of the {ProxyAdmin} contract. If set up this way,
                   * you should think of the `ProxyAdmin` instance as the real administrative interface of your proxy.
                   *
                   * NOTE: The real interface of this proxy is that defined in `ITransparentUpgradeableProxy`. This contract does not
                   * inherit from that interface, and instead the admin functions are implicitly implemented using a custom dispatch
                   * mechanism in `_fallback`. Consequently, the compiler will not produce an ABI for this contract. This is necessary to
                   * fully implement transparency without decoding reverts caused by selector clashes between the proxy and the
                   * implementation.
                   *
                   * WARNING: It is not recommended to extend this contract to add additional external functions. If you do so, the compiler
                   * will not check that there are no selector conflicts, due to the note above. A selector clash between any new function
                   * and the functions declared in {ITransparentUpgradeableProxy} will be resolved in favor of the new one. This could
                   * render the admin operations inaccessible, which could prevent upgradeability. Transparency may also be compromised.
                   */
                  contract TransparentUpgradeableProxy is ERC1967Proxy {
                      /**
                       * @dev Initializes an upgradeable proxy managed by `_admin`, backed by the implementation at `_logic`, and
                       * optionally initialized with `_data` as explained in {ERC1967Proxy-constructor}.
                       */
                      constructor(address _logic, address admin_, bytes memory _data) payable ERC1967Proxy(_logic, _data) {
                          _changeAdmin(admin_);
                      }
                      /**
                       * @dev Modifier used internally that will delegate the call to the implementation unless the sender is the admin.
                       *
                       * CAUTION: This modifier is deprecated, as it could cause issues if the modified function has arguments, and the
                       * implementation provides a function with the same selector.
                       */
                      modifier ifAdmin() {
                          if (msg.sender == _getAdmin()) {
                              _;
                          } else {
                              _fallback();
                          }
                      }
                      /**
                       * @dev If caller is the admin process the call internally, otherwise transparently fallback to the proxy behavior
                       */
                      function _fallback() internal virtual override {
                          if (msg.sender == _getAdmin()) {
                              bytes memory ret;
                              bytes4 selector = msg.sig;
                              if (selector == ITransparentUpgradeableProxy.upgradeTo.selector) {
                                  ret = _dispatchUpgradeTo();
                              } else if (selector == ITransparentUpgradeableProxy.upgradeToAndCall.selector) {
                                  ret = _dispatchUpgradeToAndCall();
                              } else if (selector == ITransparentUpgradeableProxy.changeAdmin.selector) {
                                  ret = _dispatchChangeAdmin();
                              } else if (selector == ITransparentUpgradeableProxy.admin.selector) {
                                  ret = _dispatchAdmin();
                              } else if (selector == ITransparentUpgradeableProxy.implementation.selector) {
                                  ret = _dispatchImplementation();
                              } else {
                                  revert("TransparentUpgradeableProxy: admin cannot fallback to proxy target");
                              }
                              assembly {
                                  return(add(ret, 0x20), mload(ret))
                              }
                          } else {
                              super._fallback();
                          }
                      }
                      /**
                       * @dev Returns the current admin.
                       *
                       * TIP: To get this value clients can read directly from the storage slot shown below (specified by EIP1967) using the
                       * https://eth.wiki/json-rpc/API#eth_getstorageat[`eth_getStorageAt`] RPC call.
                       * `0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103`
                       */
                      function _dispatchAdmin() private returns (bytes memory) {
                          _requireZeroValue();
                          address admin = _getAdmin();
                          return abi.encode(admin);
                      }
                      /**
                       * @dev Returns the current implementation.
                       *
                       * TIP: To get this value clients can read directly from the storage slot shown below (specified by EIP1967) using the
                       * https://eth.wiki/json-rpc/API#eth_getstorageat[`eth_getStorageAt`] RPC call.
                       * `0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc`
                       */
                      function _dispatchImplementation() private returns (bytes memory) {
                          _requireZeroValue();
                          address implementation = _implementation();
                          return abi.encode(implementation);
                      }
                      /**
                       * @dev Changes the admin of the proxy.
                       *
                       * Emits an {AdminChanged} event.
                       */
                      function _dispatchChangeAdmin() private returns (bytes memory) {
                          _requireZeroValue();
                          address newAdmin = abi.decode(msg.data[4:], (address));
                          _changeAdmin(newAdmin);
                          return "";
                      }
                      /**
                       * @dev Upgrade the implementation of the proxy.
                       */
                      function _dispatchUpgradeTo() private returns (bytes memory) {
                          _requireZeroValue();
                          address newImplementation = abi.decode(msg.data[4:], (address));
                          _upgradeToAndCall(newImplementation, bytes(""), false);
                          return "";
                      }
                      /**
                       * @dev Upgrade the implementation of the proxy, and then call a function from the new implementation as specified
                       * by `data`, which should be an encoded function call. This is useful to initialize new storage variables in the
                       * proxied contract.
                       */
                      function _dispatchUpgradeToAndCall() private returns (bytes memory) {
                          (address newImplementation, bytes memory data) = abi.decode(msg.data[4:], (address, bytes));
                          _upgradeToAndCall(newImplementation, data, true);
                          return "";
                      }
                      /**
                       * @dev Returns the current admin.
                       *
                       * CAUTION: This function is deprecated. Use {ERC1967Upgrade-_getAdmin} instead.
                       */
                      function _admin() internal view virtual returns (address) {
                          return _getAdmin();
                      }
                      /**
                       * @dev To keep this contract fully transparent, all `ifAdmin` functions must be payable. This helper is here to
                       * emulate some proxy functions being non-payable while still allowing value to pass through.
                       */
                      function _requireZeroValue() private {
                          require(msg.value == 0);
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.7.0) (proxy/ERC1967/ERC1967Proxy.sol)
                  pragma solidity ^0.8.0;
                  import "../Proxy.sol";
                  import "./ERC1967Upgrade.sol";
                  /**
                   * @dev This contract implements an upgradeable proxy. It is upgradeable because calls are delegated to an
                   * implementation address that can be changed. This address is stored in storage in the location specified by
                   * https://eips.ethereum.org/EIPS/eip-1967[EIP1967], so that it doesn't conflict with the storage layout of the
                   * implementation behind the proxy.
                   */
                  contract ERC1967Proxy is Proxy, ERC1967Upgrade {
                      /**
                       * @dev Initializes the upgradeable proxy with an initial implementation specified by `_logic`.
                       *
                       * If `_data` is nonempty, it's used as data in a delegate call to `_logic`. This will typically be an encoded
                       * function call, and allows initializing the storage of the proxy like a Solidity constructor.
                       */
                      constructor(address _logic, bytes memory _data) payable {
                          _upgradeToAndCall(_logic, _data, false);
                      }
                      /**
                       * @dev Returns the current implementation address.
                       */
                      function _implementation() internal view virtual override returns (address impl) {
                          return ERC1967Upgrade._getImplementation();
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.6.0) (proxy/Proxy.sol)
                  pragma solidity ^0.8.0;
                  /**
                   * @dev This abstract contract provides a fallback function that delegates all calls to another contract using the EVM
                   * instruction `delegatecall`. We refer to the second contract as the _implementation_ behind the proxy, and it has to
                   * be specified by overriding the virtual {_implementation} function.
                   *
                   * Additionally, delegation to the implementation can be triggered manually through the {_fallback} function, or to a
                   * different contract through the {_delegate} function.
                   *
                   * The success and return data of the delegated call will be returned back to the caller of the proxy.
                   */
                  abstract contract Proxy {
                      /**
                       * @dev Delegates the current call to `implementation`.
                       *
                       * This function does not return to its internal call site, it will return directly to the external caller.
                       */
                      function _delegate(address implementation) internal virtual {
                          assembly {
                              // Copy msg.data. We take full control of memory in this inline assembly
                              // block because it will not return to Solidity code. We overwrite the
                              // Solidity scratch pad at memory position 0.
                              calldatacopy(0, 0, calldatasize())
                              // Call the implementation.
                              // out and outsize are 0 because we don't know the size yet.
                              let result := delegatecall(gas(), implementation, 0, calldatasize(), 0, 0)
                              // Copy the returned data.
                              returndatacopy(0, 0, returndatasize())
                              switch result
                              // delegatecall returns 0 on error.
                              case 0 {
                                  revert(0, returndatasize())
                              }
                              default {
                                  return(0, returndatasize())
                              }
                          }
                      }
                      /**
                       * @dev This is a virtual function that should be overridden so it returns the address to which the fallback function
                       * and {_fallback} should delegate.
                       */
                      function _implementation() internal view virtual returns (address);
                      /**
                       * @dev Delegates the current call to the address returned by `_implementation()`.
                       *
                       * This function does not return to its internal call site, it will return directly to the external caller.
                       */
                      function _fallback() internal virtual {
                          _beforeFallback();
                          _delegate(_implementation());
                      }
                      /**
                       * @dev Fallback function that delegates calls to the address returned by `_implementation()`. Will run if no other
                       * function in the contract matches the call data.
                       */
                      fallback() external payable virtual {
                          _fallback();
                      }
                      /**
                       * @dev Fallback function that delegates calls to the address returned by `_implementation()`. Will run if call data
                       * is empty.
                       */
                      receive() external payable virtual {
                          _fallback();
                      }
                      /**
                       * @dev Hook that is called before falling back to the implementation. Can happen as part of a manual `_fallback`
                       * call, or as part of the Solidity `fallback` or `receive` functions.
                       *
                       * If overridden should call `super._beforeFallback()`.
                       */
                      function _beforeFallback() internal virtual {}
                  }
                  // 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 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) (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 (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) (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
                          }
                      }
                  }
                  

                  File 5 of 9: TransparentUpgradeableProxy
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.9.0) (proxy/transparent/TransparentUpgradeableProxy.sol)
                  pragma solidity ^0.8.0;
                  import "../ERC1967/ERC1967Proxy.sol";
                  /**
                   * @dev Interface for {TransparentUpgradeableProxy}. In order to implement transparency, {TransparentUpgradeableProxy}
                   * does not implement this interface directly, and some of its functions are implemented by an internal dispatch
                   * mechanism. The compiler is unaware that these functions are implemented by {TransparentUpgradeableProxy} and will not
                   * include them in the ABI so this interface must be used to interact with it.
                   */
                  interface ITransparentUpgradeableProxy is IERC1967 {
                      function admin() external view returns (address);
                      function implementation() external view returns (address);
                      function changeAdmin(address) external;
                      function upgradeTo(address) external;
                      function upgradeToAndCall(address, bytes memory) external payable;
                  }
                  /**
                   * @dev This contract implements a proxy that is upgradeable by an admin.
                   *
                   * To avoid https://medium.com/nomic-labs-blog/malicious-backdoors-in-ethereum-proxies-62629adf3357[proxy selector
                   * clashing], which can potentially be used in an attack, this contract uses the
                   * https://blog.openzeppelin.com/the-transparent-proxy-pattern/[transparent proxy pattern]. This pattern implies two
                   * things that go hand in hand:
                   *
                   * 1. If any account other than the admin calls the proxy, the call will be forwarded to the implementation, even if
                   * that call matches one of the admin functions exposed by the proxy itself.
                   * 2. If the admin calls the proxy, it can access the admin functions, but its calls will never be forwarded to the
                   * implementation. If the admin tries to call a function on the implementation it will fail with an error that says
                   * "admin cannot fallback to proxy target".
                   *
                   * These properties mean that the admin account can only be used for admin actions like upgrading the proxy or changing
                   * the admin, so it's best if it's a dedicated account that is not used for anything else. This will avoid headaches due
                   * to sudden errors when trying to call a function from the proxy implementation.
                   *
                   * Our recommendation is for the dedicated account to be an instance of the {ProxyAdmin} contract. If set up this way,
                   * you should think of the `ProxyAdmin` instance as the real administrative interface of your proxy.
                   *
                   * NOTE: The real interface of this proxy is that defined in `ITransparentUpgradeableProxy`. This contract does not
                   * inherit from that interface, and instead the admin functions are implicitly implemented using a custom dispatch
                   * mechanism in `_fallback`. Consequently, the compiler will not produce an ABI for this contract. This is necessary to
                   * fully implement transparency without decoding reverts caused by selector clashes between the proxy and the
                   * implementation.
                   *
                   * WARNING: It is not recommended to extend this contract to add additional external functions. If you do so, the compiler
                   * will not check that there are no selector conflicts, due to the note above. A selector clash between any new function
                   * and the functions declared in {ITransparentUpgradeableProxy} will be resolved in favor of the new one. This could
                   * render the admin operations inaccessible, which could prevent upgradeability. Transparency may also be compromised.
                   */
                  contract TransparentUpgradeableProxy is ERC1967Proxy {
                      /**
                       * @dev Initializes an upgradeable proxy managed by `_admin`, backed by the implementation at `_logic`, and
                       * optionally initialized with `_data` as explained in {ERC1967Proxy-constructor}.
                       */
                      constructor(address _logic, address admin_, bytes memory _data) payable ERC1967Proxy(_logic, _data) {
                          _changeAdmin(admin_);
                      }
                      /**
                       * @dev Modifier used internally that will delegate the call to the implementation unless the sender is the admin.
                       *
                       * CAUTION: This modifier is deprecated, as it could cause issues if the modified function has arguments, and the
                       * implementation provides a function with the same selector.
                       */
                      modifier ifAdmin() {
                          if (msg.sender == _getAdmin()) {
                              _;
                          } else {
                              _fallback();
                          }
                      }
                      /**
                       * @dev If caller is the admin process the call internally, otherwise transparently fallback to the proxy behavior
                       */
                      function _fallback() internal virtual override {
                          if (msg.sender == _getAdmin()) {
                              bytes memory ret;
                              bytes4 selector = msg.sig;
                              if (selector == ITransparentUpgradeableProxy.upgradeTo.selector) {
                                  ret = _dispatchUpgradeTo();
                              } else if (selector == ITransparentUpgradeableProxy.upgradeToAndCall.selector) {
                                  ret = _dispatchUpgradeToAndCall();
                              } else if (selector == ITransparentUpgradeableProxy.changeAdmin.selector) {
                                  ret = _dispatchChangeAdmin();
                              } else if (selector == ITransparentUpgradeableProxy.admin.selector) {
                                  ret = _dispatchAdmin();
                              } else if (selector == ITransparentUpgradeableProxy.implementation.selector) {
                                  ret = _dispatchImplementation();
                              } else {
                                  revert("TransparentUpgradeableProxy: admin cannot fallback to proxy target");
                              }
                              assembly {
                                  return(add(ret, 0x20), mload(ret))
                              }
                          } else {
                              super._fallback();
                          }
                      }
                      /**
                       * @dev Returns the current admin.
                       *
                       * TIP: To get this value clients can read directly from the storage slot shown below (specified by EIP1967) using the
                       * https://eth.wiki/json-rpc/API#eth_getstorageat[`eth_getStorageAt`] RPC call.
                       * `0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103`
                       */
                      function _dispatchAdmin() private returns (bytes memory) {
                          _requireZeroValue();
                          address admin = _getAdmin();
                          return abi.encode(admin);
                      }
                      /**
                       * @dev Returns the current implementation.
                       *
                       * TIP: To get this value clients can read directly from the storage slot shown below (specified by EIP1967) using the
                       * https://eth.wiki/json-rpc/API#eth_getstorageat[`eth_getStorageAt`] RPC call.
                       * `0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc`
                       */
                      function _dispatchImplementation() private returns (bytes memory) {
                          _requireZeroValue();
                          address implementation = _implementation();
                          return abi.encode(implementation);
                      }
                      /**
                       * @dev Changes the admin of the proxy.
                       *
                       * Emits an {AdminChanged} event.
                       */
                      function _dispatchChangeAdmin() private returns (bytes memory) {
                          _requireZeroValue();
                          address newAdmin = abi.decode(msg.data[4:], (address));
                          _changeAdmin(newAdmin);
                          return "";
                      }
                      /**
                       * @dev Upgrade the implementation of the proxy.
                       */
                      function _dispatchUpgradeTo() private returns (bytes memory) {
                          _requireZeroValue();
                          address newImplementation = abi.decode(msg.data[4:], (address));
                          _upgradeToAndCall(newImplementation, bytes(""), false);
                          return "";
                      }
                      /**
                       * @dev Upgrade the implementation of the proxy, and then call a function from the new implementation as specified
                       * by `data`, which should be an encoded function call. This is useful to initialize new storage variables in the
                       * proxied contract.
                       */
                      function _dispatchUpgradeToAndCall() private returns (bytes memory) {
                          (address newImplementation, bytes memory data) = abi.decode(msg.data[4:], (address, bytes));
                          _upgradeToAndCall(newImplementation, data, true);
                          return "";
                      }
                      /**
                       * @dev Returns the current admin.
                       *
                       * CAUTION: This function is deprecated. Use {ERC1967Upgrade-_getAdmin} instead.
                       */
                      function _admin() internal view virtual returns (address) {
                          return _getAdmin();
                      }
                      /**
                       * @dev To keep this contract fully transparent, all `ifAdmin` functions must be payable. This helper is here to
                       * emulate some proxy functions being non-payable while still allowing value to pass through.
                       */
                      function _requireZeroValue() private {
                          require(msg.value == 0);
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.7.0) (proxy/ERC1967/ERC1967Proxy.sol)
                  pragma solidity ^0.8.0;
                  import "../Proxy.sol";
                  import "./ERC1967Upgrade.sol";
                  /**
                   * @dev This contract implements an upgradeable proxy. It is upgradeable because calls are delegated to an
                   * implementation address that can be changed. This address is stored in storage in the location specified by
                   * https://eips.ethereum.org/EIPS/eip-1967[EIP1967], so that it doesn't conflict with the storage layout of the
                   * implementation behind the proxy.
                   */
                  contract ERC1967Proxy is Proxy, ERC1967Upgrade {
                      /**
                       * @dev Initializes the upgradeable proxy with an initial implementation specified by `_logic`.
                       *
                       * If `_data` is nonempty, it's used as data in a delegate call to `_logic`. This will typically be an encoded
                       * function call, and allows initializing the storage of the proxy like a Solidity constructor.
                       */
                      constructor(address _logic, bytes memory _data) payable {
                          _upgradeToAndCall(_logic, _data, false);
                      }
                      /**
                       * @dev Returns the current implementation address.
                       */
                      function _implementation() internal view virtual override returns (address impl) {
                          return ERC1967Upgrade._getImplementation();
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.6.0) (proxy/Proxy.sol)
                  pragma solidity ^0.8.0;
                  /**
                   * @dev This abstract contract provides a fallback function that delegates all calls to another contract using the EVM
                   * instruction `delegatecall`. We refer to the second contract as the _implementation_ behind the proxy, and it has to
                   * be specified by overriding the virtual {_implementation} function.
                   *
                   * Additionally, delegation to the implementation can be triggered manually through the {_fallback} function, or to a
                   * different contract through the {_delegate} function.
                   *
                   * The success and return data of the delegated call will be returned back to the caller of the proxy.
                   */
                  abstract contract Proxy {
                      /**
                       * @dev Delegates the current call to `implementation`.
                       *
                       * This function does not return to its internal call site, it will return directly to the external caller.
                       */
                      function _delegate(address implementation) internal virtual {
                          assembly {
                              // Copy msg.data. We take full control of memory in this inline assembly
                              // block because it will not return to Solidity code. We overwrite the
                              // Solidity scratch pad at memory position 0.
                              calldatacopy(0, 0, calldatasize())
                              // Call the implementation.
                              // out and outsize are 0 because we don't know the size yet.
                              let result := delegatecall(gas(), implementation, 0, calldatasize(), 0, 0)
                              // Copy the returned data.
                              returndatacopy(0, 0, returndatasize())
                              switch result
                              // delegatecall returns 0 on error.
                              case 0 {
                                  revert(0, returndatasize())
                              }
                              default {
                                  return(0, returndatasize())
                              }
                          }
                      }
                      /**
                       * @dev This is a virtual function that should be overridden so it returns the address to which the fallback function
                       * and {_fallback} should delegate.
                       */
                      function _implementation() internal view virtual returns (address);
                      /**
                       * @dev Delegates the current call to the address returned by `_implementation()`.
                       *
                       * This function does not return to its internal call site, it will return directly to the external caller.
                       */
                      function _fallback() internal virtual {
                          _beforeFallback();
                          _delegate(_implementation());
                      }
                      /**
                       * @dev Fallback function that delegates calls to the address returned by `_implementation()`. Will run if no other
                       * function in the contract matches the call data.
                       */
                      fallback() external payable virtual {
                          _fallback();
                      }
                      /**
                       * @dev Fallback function that delegates calls to the address returned by `_implementation()`. Will run if call data
                       * is empty.
                       */
                      receive() external payable virtual {
                          _fallback();
                      }
                      /**
                       * @dev Hook that is called before falling back to the implementation. Can happen as part of a manual `_fallback`
                       * call, or as part of the Solidity `fallback` or `receive` functions.
                       *
                       * If overridden should call `super._beforeFallback()`.
                       */
                      function _beforeFallback() internal virtual {}
                  }
                  // 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 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) (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 (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) (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
                          }
                      }
                  }
                  

                  File 6 of 9: TransparentUpgradeableProxy
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.9.0) (proxy/transparent/TransparentUpgradeableProxy.sol)
                  pragma solidity ^0.8.0;
                  import "../ERC1967/ERC1967Proxy.sol";
                  /**
                   * @dev Interface for {TransparentUpgradeableProxy}. In order to implement transparency, {TransparentUpgradeableProxy}
                   * does not implement this interface directly, and some of its functions are implemented by an internal dispatch
                   * mechanism. The compiler is unaware that these functions are implemented by {TransparentUpgradeableProxy} and will not
                   * include them in the ABI so this interface must be used to interact with it.
                   */
                  interface ITransparentUpgradeableProxy is IERC1967 {
                      function admin() external view returns (address);
                      function implementation() external view returns (address);
                      function changeAdmin(address) external;
                      function upgradeTo(address) external;
                      function upgradeToAndCall(address, bytes memory) external payable;
                  }
                  /**
                   * @dev This contract implements a proxy that is upgradeable by an admin.
                   *
                   * To avoid https://medium.com/nomic-labs-blog/malicious-backdoors-in-ethereum-proxies-62629adf3357[proxy selector
                   * clashing], which can potentially be used in an attack, this contract uses the
                   * https://blog.openzeppelin.com/the-transparent-proxy-pattern/[transparent proxy pattern]. This pattern implies two
                   * things that go hand in hand:
                   *
                   * 1. If any account other than the admin calls the proxy, the call will be forwarded to the implementation, even if
                   * that call matches one of the admin functions exposed by the proxy itself.
                   * 2. If the admin calls the proxy, it can access the admin functions, but its calls will never be forwarded to the
                   * implementation. If the admin tries to call a function on the implementation it will fail with an error that says
                   * "admin cannot fallback to proxy target".
                   *
                   * These properties mean that the admin account can only be used for admin actions like upgrading the proxy or changing
                   * the admin, so it's best if it's a dedicated account that is not used for anything else. This will avoid headaches due
                   * to sudden errors when trying to call a function from the proxy implementation.
                   *
                   * Our recommendation is for the dedicated account to be an instance of the {ProxyAdmin} contract. If set up this way,
                   * you should think of the `ProxyAdmin` instance as the real administrative interface of your proxy.
                   *
                   * NOTE: The real interface of this proxy is that defined in `ITransparentUpgradeableProxy`. This contract does not
                   * inherit from that interface, and instead the admin functions are implicitly implemented using a custom dispatch
                   * mechanism in `_fallback`. Consequently, the compiler will not produce an ABI for this contract. This is necessary to
                   * fully implement transparency without decoding reverts caused by selector clashes between the proxy and the
                   * implementation.
                   *
                   * WARNING: It is not recommended to extend this contract to add additional external functions. If you do so, the compiler
                   * will not check that there are no selector conflicts, due to the note above. A selector clash between any new function
                   * and the functions declared in {ITransparentUpgradeableProxy} will be resolved in favor of the new one. This could
                   * render the admin operations inaccessible, which could prevent upgradeability. Transparency may also be compromised.
                   */
                  contract TransparentUpgradeableProxy is ERC1967Proxy {
                      /**
                       * @dev Initializes an upgradeable proxy managed by `_admin`, backed by the implementation at `_logic`, and
                       * optionally initialized with `_data` as explained in {ERC1967Proxy-constructor}.
                       */
                      constructor(address _logic, address admin_, bytes memory _data) payable ERC1967Proxy(_logic, _data) {
                          _changeAdmin(admin_);
                      }
                      /**
                       * @dev Modifier used internally that will delegate the call to the implementation unless the sender is the admin.
                       *
                       * CAUTION: This modifier is deprecated, as it could cause issues if the modified function has arguments, and the
                       * implementation provides a function with the same selector.
                       */
                      modifier ifAdmin() {
                          if (msg.sender == _getAdmin()) {
                              _;
                          } else {
                              _fallback();
                          }
                      }
                      /**
                       * @dev If caller is the admin process the call internally, otherwise transparently fallback to the proxy behavior
                       */
                      function _fallback() internal virtual override {
                          if (msg.sender == _getAdmin()) {
                              bytes memory ret;
                              bytes4 selector = msg.sig;
                              if (selector == ITransparentUpgradeableProxy.upgradeTo.selector) {
                                  ret = _dispatchUpgradeTo();
                              } else if (selector == ITransparentUpgradeableProxy.upgradeToAndCall.selector) {
                                  ret = _dispatchUpgradeToAndCall();
                              } else if (selector == ITransparentUpgradeableProxy.changeAdmin.selector) {
                                  ret = _dispatchChangeAdmin();
                              } else if (selector == ITransparentUpgradeableProxy.admin.selector) {
                                  ret = _dispatchAdmin();
                              } else if (selector == ITransparentUpgradeableProxy.implementation.selector) {
                                  ret = _dispatchImplementation();
                              } else {
                                  revert("TransparentUpgradeableProxy: admin cannot fallback to proxy target");
                              }
                              assembly {
                                  return(add(ret, 0x20), mload(ret))
                              }
                          } else {
                              super._fallback();
                          }
                      }
                      /**
                       * @dev Returns the current admin.
                       *
                       * TIP: To get this value clients can read directly from the storage slot shown below (specified by EIP1967) using the
                       * https://eth.wiki/json-rpc/API#eth_getstorageat[`eth_getStorageAt`] RPC call.
                       * `0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103`
                       */
                      function _dispatchAdmin() private returns (bytes memory) {
                          _requireZeroValue();
                          address admin = _getAdmin();
                          return abi.encode(admin);
                      }
                      /**
                       * @dev Returns the current implementation.
                       *
                       * TIP: To get this value clients can read directly from the storage slot shown below (specified by EIP1967) using the
                       * https://eth.wiki/json-rpc/API#eth_getstorageat[`eth_getStorageAt`] RPC call.
                       * `0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc`
                       */
                      function _dispatchImplementation() private returns (bytes memory) {
                          _requireZeroValue();
                          address implementation = _implementation();
                          return abi.encode(implementation);
                      }
                      /**
                       * @dev Changes the admin of the proxy.
                       *
                       * Emits an {AdminChanged} event.
                       */
                      function _dispatchChangeAdmin() private returns (bytes memory) {
                          _requireZeroValue();
                          address newAdmin = abi.decode(msg.data[4:], (address));
                          _changeAdmin(newAdmin);
                          return "";
                      }
                      /**
                       * @dev Upgrade the implementation of the proxy.
                       */
                      function _dispatchUpgradeTo() private returns (bytes memory) {
                          _requireZeroValue();
                          address newImplementation = abi.decode(msg.data[4:], (address));
                          _upgradeToAndCall(newImplementation, bytes(""), false);
                          return "";
                      }
                      /**
                       * @dev Upgrade the implementation of the proxy, and then call a function from the new implementation as specified
                       * by `data`, which should be an encoded function call. This is useful to initialize new storage variables in the
                       * proxied contract.
                       */
                      function _dispatchUpgradeToAndCall() private returns (bytes memory) {
                          (address newImplementation, bytes memory data) = abi.decode(msg.data[4:], (address, bytes));
                          _upgradeToAndCall(newImplementation, data, true);
                          return "";
                      }
                      /**
                       * @dev Returns the current admin.
                       *
                       * CAUTION: This function is deprecated. Use {ERC1967Upgrade-_getAdmin} instead.
                       */
                      function _admin() internal view virtual returns (address) {
                          return _getAdmin();
                      }
                      /**
                       * @dev To keep this contract fully transparent, all `ifAdmin` functions must be payable. This helper is here to
                       * emulate some proxy functions being non-payable while still allowing value to pass through.
                       */
                      function _requireZeroValue() private {
                          require(msg.value == 0);
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.7.0) (proxy/ERC1967/ERC1967Proxy.sol)
                  pragma solidity ^0.8.0;
                  import "../Proxy.sol";
                  import "./ERC1967Upgrade.sol";
                  /**
                   * @dev This contract implements an upgradeable proxy. It is upgradeable because calls are delegated to an
                   * implementation address that can be changed. This address is stored in storage in the location specified by
                   * https://eips.ethereum.org/EIPS/eip-1967[EIP1967], so that it doesn't conflict with the storage layout of the
                   * implementation behind the proxy.
                   */
                  contract ERC1967Proxy is Proxy, ERC1967Upgrade {
                      /**
                       * @dev Initializes the upgradeable proxy with an initial implementation specified by `_logic`.
                       *
                       * If `_data` is nonempty, it's used as data in a delegate call to `_logic`. This will typically be an encoded
                       * function call, and allows initializing the storage of the proxy like a Solidity constructor.
                       */
                      constructor(address _logic, bytes memory _data) payable {
                          _upgradeToAndCall(_logic, _data, false);
                      }
                      /**
                       * @dev Returns the current implementation address.
                       */
                      function _implementation() internal view virtual override returns (address impl) {
                          return ERC1967Upgrade._getImplementation();
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.6.0) (proxy/Proxy.sol)
                  pragma solidity ^0.8.0;
                  /**
                   * @dev This abstract contract provides a fallback function that delegates all calls to another contract using the EVM
                   * instruction `delegatecall`. We refer to the second contract as the _implementation_ behind the proxy, and it has to
                   * be specified by overriding the virtual {_implementation} function.
                   *
                   * Additionally, delegation to the implementation can be triggered manually through the {_fallback} function, or to a
                   * different contract through the {_delegate} function.
                   *
                   * The success and return data of the delegated call will be returned back to the caller of the proxy.
                   */
                  abstract contract Proxy {
                      /**
                       * @dev Delegates the current call to `implementation`.
                       *
                       * This function does not return to its internal call site, it will return directly to the external caller.
                       */
                      function _delegate(address implementation) internal virtual {
                          assembly {
                              // Copy msg.data. We take full control of memory in this inline assembly
                              // block because it will not return to Solidity code. We overwrite the
                              // Solidity scratch pad at memory position 0.
                              calldatacopy(0, 0, calldatasize())
                              // Call the implementation.
                              // out and outsize are 0 because we don't know the size yet.
                              let result := delegatecall(gas(), implementation, 0, calldatasize(), 0, 0)
                              // Copy the returned data.
                              returndatacopy(0, 0, returndatasize())
                              switch result
                              // delegatecall returns 0 on error.
                              case 0 {
                                  revert(0, returndatasize())
                              }
                              default {
                                  return(0, returndatasize())
                              }
                          }
                      }
                      /**
                       * @dev This is a virtual function that should be overridden so it returns the address to which the fallback function
                       * and {_fallback} should delegate.
                       */
                      function _implementation() internal view virtual returns (address);
                      /**
                       * @dev Delegates the current call to the address returned by `_implementation()`.
                       *
                       * This function does not return to its internal call site, it will return directly to the external caller.
                       */
                      function _fallback() internal virtual {
                          _beforeFallback();
                          _delegate(_implementation());
                      }
                      /**
                       * @dev Fallback function that delegates calls to the address returned by `_implementation()`. Will run if no other
                       * function in the contract matches the call data.
                       */
                      fallback() external payable virtual {
                          _fallback();
                      }
                      /**
                       * @dev Fallback function that delegates calls to the address returned by `_implementation()`. Will run if call data
                       * is empty.
                       */
                      receive() external payable virtual {
                          _fallback();
                      }
                      /**
                       * @dev Hook that is called before falling back to the implementation. Can happen as part of a manual `_fallback`
                       * call, or as part of the Solidity `fallback` or `receive` functions.
                       *
                       * If overridden should call `super._beforeFallback()`.
                       */
                      function _beforeFallback() internal virtual {}
                  }
                  // 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 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) (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 (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) (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
                          }
                      }
                  }
                  

                  File 7 of 9: L1GatewayRouter
                  // SPDX-License-Identifier: MIT
                  pragma solidity =0.8.16;
                  import {OwnableUpgradeable} from "@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol";
                  import {IERC20Upgradeable} from "@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol";
                  import {SafeERC20Upgradeable} from "@openzeppelin/contracts-upgradeable/token/ERC20/utils/SafeERC20Upgradeable.sol";
                  import {IL1ETHGateway} from "./IL1ETHGateway.sol";
                  import {IL1ERC20Gateway} from "./IL1ERC20Gateway.sol";
                  import {IL1GatewayRouter} from "./IL1GatewayRouter.sol";
                  /// @title L1GatewayRouter
                  /// @notice The `L1GatewayRouter` is the main entry for depositing Ether and ERC20 tokens.
                  /// All deposited tokens are routed to corresponding gateways.
                  /// @dev One can also use this contract to query L1/L2 token address mapping.
                  /// In the future, ERC-721 and ERC-1155 tokens will be added to the router too.
                  contract L1GatewayRouter is OwnableUpgradeable, IL1GatewayRouter {
                      using SafeERC20Upgradeable for IERC20Upgradeable;
                      /*************
                       * Variables *
                       *************/
                      /// @notice The address of L1ETHGateway.
                      address public ethGateway;
                      /// @notice The addess of default ERC20 gateway, normally the L1StandardERC20Gateway contract.
                      address public defaultERC20Gateway;
                      /// @notice Mapping from ERC20 token address to corresponding L1ERC20Gateway.
                      // solhint-disable-next-line var-name-mixedcase
                      mapping(address => address) public ERC20Gateway;
                      /// @notice The address of gateway in current execution context.
                      address public gatewayInContext;
                      /**********************
                       * Function Modifiers *
                       **********************/
                      modifier onlyNotInContext() {
                          require(gatewayInContext == address(0), "Only not in context");
                          _;
                      }
                      modifier onlyInContext() {
                          require(_msgSender() == gatewayInContext, "Only in deposit context");
                          _;
                      }
                      /***************
                       * Constructor *
                       ***************/
                      constructor() {
                          _disableInitializers();
                      }
                      /// @notice Initialize the storage of L1GatewayRouter.
                      /// @param _ethGateway The address of L1ETHGateway contract.
                      /// @param _defaultERC20Gateway The address of default ERC20 Gateway contract.
                      function initialize(address _ethGateway, address _defaultERC20Gateway) external initializer {
                          OwnableUpgradeable.__Ownable_init();
                          // it can be zero during initialization
                          if (_defaultERC20Gateway != address(0)) {
                              defaultERC20Gateway = _defaultERC20Gateway;
                              emit SetDefaultERC20Gateway(address(0), _defaultERC20Gateway);
                          }
                          // it can be zero during initialization
                          if (_ethGateway != address(0)) {
                              ethGateway = _ethGateway;
                              emit SetETHGateway(address(0), _ethGateway);
                          }
                      }
                      /*************************
                       * Public View Functions *
                       *************************/
                      /// @inheritdoc IL1ERC20Gateway
                      function getL2ERC20Address(address _l1Address) external view override returns (address) {
                          address _gateway = getERC20Gateway(_l1Address);
                          if (_gateway == address(0)) {
                              return address(0);
                          }
                          return IL1ERC20Gateway(_gateway).getL2ERC20Address(_l1Address);
                      }
                      /// @inheritdoc IL1GatewayRouter
                      function getERC20Gateway(address _token) public view returns (address) {
                          address _gateway = ERC20Gateway[_token];
                          if (_gateway == address(0)) {
                              _gateway = defaultERC20Gateway;
                          }
                          return _gateway;
                      }
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @inheritdoc IL1GatewayRouter
                      /// @dev All the gateways should have reentrancy guard to prevent potential attack though this function.
                      function requestERC20(
                          address _sender,
                          address _token,
                          uint256 _amount
                      ) external onlyInContext returns (uint256) {
                          address _caller = _msgSender();
                          uint256 _balance = IERC20Upgradeable(_token).balanceOf(_caller);
                          IERC20Upgradeable(_token).safeTransferFrom(_sender, _caller, _amount);
                          _amount = IERC20Upgradeable(_token).balanceOf(_caller) - _balance;
                          return _amount;
                      }
                      /*************************************************
                       * Public Mutating Functions from L1ERC20Gateway *
                       *************************************************/
                      /// @inheritdoc IL1ERC20Gateway
                      function depositERC20(
                          address _token,
                          uint256 _amount,
                          uint256 _gasLimit
                      ) external payable override {
                          depositERC20AndCall(_token, _msgSender(), _amount, new bytes(0), _gasLimit);
                      }
                      /// @inheritdoc IL1ERC20Gateway
                      function depositERC20(
                          address _token,
                          address _to,
                          uint256 _amount,
                          uint256 _gasLimit
                      ) external payable override {
                          depositERC20AndCall(_token, _to, _amount, new bytes(0), _gasLimit);
                      }
                      /// @inheritdoc IL1ERC20Gateway
                      function depositERC20AndCall(
                          address _token,
                          address _to,
                          uint256 _amount,
                          bytes memory _data,
                          uint256 _gasLimit
                      ) public payable override onlyNotInContext {
                          address _gateway = getERC20Gateway(_token);
                          require(_gateway != address(0), "no gateway available");
                          // enter deposit context
                          gatewayInContext = _gateway;
                          // encode msg.sender with _data
                          bytes memory _routerData = abi.encode(_msgSender(), _data);
                          IL1ERC20Gateway(_gateway).depositERC20AndCall{value: msg.value}(_token, _to, _amount, _routerData, _gasLimit);
                          // leave deposit context
                          gatewayInContext = address(0);
                      }
                      /// @inheritdoc IL1ERC20Gateway
                      function finalizeWithdrawERC20(
                          address,
                          address,
                          address,
                          address,
                          uint256,
                          bytes calldata
                      ) external payable virtual override {
                          revert("should never be called");
                      }
                      /***********************************************
                       * Public Mutating Functions from L1ETHGateway *
                       ***********************************************/
                      /// @inheritdoc IL1ETHGateway
                      function depositETH(uint256 _amount, uint256 _gasLimit) external payable override {
                          depositETHAndCall(_msgSender(), _amount, new bytes(0), _gasLimit);
                      }
                      /// @inheritdoc IL1ETHGateway
                      function depositETH(
                          address _to,
                          uint256 _amount,
                          uint256 _gasLimit
                      ) external payable override {
                          depositETHAndCall(_to, _amount, new bytes(0), _gasLimit);
                      }
                      /// @inheritdoc IL1ETHGateway
                      function depositETHAndCall(
                          address _to,
                          uint256 _amount,
                          bytes memory _data,
                          uint256 _gasLimit
                      ) public payable override onlyNotInContext {
                          address _gateway = ethGateway;
                          require(_gateway != address(0), "eth gateway available");
                          // encode msg.sender with _data
                          bytes memory _routerData = abi.encode(_msgSender(), _data);
                          IL1ETHGateway(_gateway).depositETHAndCall{value: msg.value}(_to, _amount, _routerData, _gasLimit);
                      }
                      /// @inheritdoc IL1ETHGateway
                      function finalizeWithdrawETH(
                          address,
                          address,
                          uint256,
                          bytes calldata
                      ) external payable virtual override {
                          revert("should never be called");
                      }
                      /************************
                       * Restricted Functions *
                       ************************/
                      /// @inheritdoc IL1GatewayRouter
                      function setETHGateway(address _newEthGateway) external onlyOwner {
                          address _oldETHGateway = ethGateway;
                          ethGateway = _newEthGateway;
                          emit SetETHGateway(_oldETHGateway, _newEthGateway);
                      }
                      /// @inheritdoc IL1GatewayRouter
                      function setDefaultERC20Gateway(address _newDefaultERC20Gateway) external onlyOwner {
                          address _oldDefaultERC20Gateway = defaultERC20Gateway;
                          defaultERC20Gateway = _newDefaultERC20Gateway;
                          emit SetDefaultERC20Gateway(_oldDefaultERC20Gateway, _newDefaultERC20Gateway);
                      }
                      /// @inheritdoc IL1GatewayRouter
                      function setERC20Gateway(address[] memory _tokens, address[] memory _gateways) external onlyOwner {
                          require(_tokens.length == _gateways.length, "length mismatch");
                          for (uint256 i = 0; i < _tokens.length; i++) {
                              address _oldGateway = ERC20Gateway[_tokens[i]];
                              ERC20Gateway[_tokens[i]] = _gateways[i];
                              emit SetERC20Gateway(_tokens[i], _oldGateway, _gateways[i]);
                          }
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.9.0) (access/Ownable.sol)
                  pragma solidity ^0.8.0;
                  import "../utils/ContextUpgradeable.sol";
                  import "../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) (token/ERC20/IERC20.sol)
                  pragma solidity ^0.8.0;
                  /**
                   * @dev Interface of the ERC20 standard as defined in the EIP.
                   */
                  interface IERC20Upgradeable {
                      /**
                       * @dev Emitted when `value` tokens are moved from one account (`from`) to
                       * another (`to`).
                       *
                       * Note that `value` may be zero.
                       */
                      event Transfer(address indexed from, address indexed to, uint256 value);
                      /**
                       * @dev Emitted when the allowance of a `spender` for an `owner` is set by
                       * a call to {approve}. `value` is the new allowance.
                       */
                      event Approval(address indexed owner, address indexed spender, uint256 value);
                      /**
                       * @dev Returns the amount of tokens in existence.
                       */
                      function totalSupply() external view returns (uint256);
                      /**
                       * @dev Returns the amount of tokens owned by `account`.
                       */
                      function balanceOf(address account) external view returns (uint256);
                      /**
                       * @dev Moves `amount` tokens from the caller's account to `to`.
                       *
                       * Returns a boolean value indicating whether the operation succeeded.
                       *
                       * Emits a {Transfer} event.
                       */
                      function transfer(address to, uint256 amount) external returns (bool);
                      /**
                       * @dev Returns the remaining number of tokens that `spender` will be
                       * allowed to spend on behalf of `owner` through {transferFrom}. This is
                       * zero by default.
                       *
                       * This value changes when {approve} or {transferFrom} are called.
                       */
                      function allowance(address owner, address spender) external view returns (uint256);
                      /**
                       * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
                       *
                       * Returns a boolean value indicating whether the operation succeeded.
                       *
                       * IMPORTANT: Beware that changing an allowance with this method brings the risk
                       * that someone may use both the old and the new allowance by unfortunate
                       * transaction ordering. One possible solution to mitigate this race
                       * condition is to first reduce the spender's allowance to 0 and set the
                       * desired value afterwards:
                       * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
                       *
                       * Emits an {Approval} event.
                       */
                      function approve(address spender, uint256 amount) external returns (bool);
                      /**
                       * @dev Moves `amount` tokens from `from` to `to` using the
                       * allowance mechanism. `amount` is then deducted from the caller's
                       * allowance.
                       *
                       * Returns a boolean value indicating whether the operation succeeded.
                       *
                       * Emits a {Transfer} event.
                       */
                      function transferFrom(address from, address to, uint256 amount) external returns (bool);
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.9.3) (token/ERC20/utils/SafeERC20.sol)
                  pragma solidity ^0.8.0;
                  import "../IERC20Upgradeable.sol";
                  import "../extensions/IERC20PermitUpgradeable.sol";
                  import "../../../utils/AddressUpgradeable.sol";
                  /**
                   * @title SafeERC20
                   * @dev Wrappers around ERC20 operations that throw on failure (when the token
                   * contract returns false). Tokens that return no value (and instead revert or
                   * throw on failure) are also supported, non-reverting calls are assumed to be
                   * successful.
                   * To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract,
                   * which allows you to call the safe operations as `token.safeTransfer(...)`, etc.
                   */
                  library SafeERC20Upgradeable {
                      using AddressUpgradeable for address;
                      /**
                       * @dev Transfer `value` amount of `token` from the calling contract to `to`. If `token` returns no value,
                       * non-reverting calls are assumed to be successful.
                       */
                      function safeTransfer(IERC20Upgradeable token, address to, uint256 value) internal {
                          _callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value));
                      }
                      /**
                       * @dev Transfer `value` amount of `token` from `from` to `to`, spending the approval given by `from` to the
                       * calling contract. If `token` returns no value, non-reverting calls are assumed to be successful.
                       */
                      function safeTransferFrom(IERC20Upgradeable token, address from, address to, uint256 value) internal {
                          _callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value));
                      }
                      /**
                       * @dev Deprecated. This function has issues similar to the ones found in
                       * {IERC20-approve}, and its usage is discouraged.
                       *
                       * Whenever possible, use {safeIncreaseAllowance} and
                       * {safeDecreaseAllowance} instead.
                       */
                      function safeApprove(IERC20Upgradeable token, address spender, uint256 value) internal {
                          // safeApprove should only be called when setting an initial allowance,
                          // or when resetting it to zero. To increase and decrease it, use
                          // 'safeIncreaseAllowance' and 'safeDecreaseAllowance'
                          require(
                              (value == 0) || (token.allowance(address(this), spender) == 0),
                              "SafeERC20: approve from non-zero to non-zero allowance"
                          );
                          _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value));
                      }
                      /**
                       * @dev Increase the calling contract's allowance toward `spender` by `value`. If `token` returns no value,
                       * non-reverting calls are assumed to be successful.
                       */
                      function safeIncreaseAllowance(IERC20Upgradeable token, address spender, uint256 value) internal {
                          uint256 oldAllowance = token.allowance(address(this), spender);
                          _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, oldAllowance + value));
                      }
                      /**
                       * @dev Decrease the calling contract's allowance toward `spender` by `value`. If `token` returns no value,
                       * non-reverting calls are assumed to be successful.
                       */
                      function safeDecreaseAllowance(IERC20Upgradeable token, address spender, uint256 value) internal {
                          unchecked {
                              uint256 oldAllowance = token.allowance(address(this), spender);
                              require(oldAllowance >= value, "SafeERC20: decreased allowance below zero");
                              _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, oldAllowance - value));
                          }
                      }
                      /**
                       * @dev Set the calling contract's allowance toward `spender` to `value`. If `token` returns no value,
                       * non-reverting calls are assumed to be successful. Meant to be used with tokens that require the approval
                       * to be set to zero before setting it to a non-zero value, such as USDT.
                       */
                      function forceApprove(IERC20Upgradeable token, address spender, uint256 value) internal {
                          bytes memory approvalCall = abi.encodeWithSelector(token.approve.selector, spender, value);
                          if (!_callOptionalReturnBool(token, approvalCall)) {
                              _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, 0));
                              _callOptionalReturn(token, approvalCall);
                          }
                      }
                      /**
                       * @dev Use a ERC-2612 signature to set the `owner` approval toward `spender` on `token`.
                       * Revert on invalid signature.
                       */
                      function safePermit(
                          IERC20PermitUpgradeable token,
                          address owner,
                          address spender,
                          uint256 value,
                          uint256 deadline,
                          uint8 v,
                          bytes32 r,
                          bytes32 s
                      ) internal {
                          uint256 nonceBefore = token.nonces(owner);
                          token.permit(owner, spender, value, deadline, v, r, s);
                          uint256 nonceAfter = token.nonces(owner);
                          require(nonceAfter == nonceBefore + 1, "SafeERC20: permit did not succeed");
                      }
                      /**
                       * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement
                       * on the return value: the return value is optional (but if data is returned, it must not be false).
                       * @param token The token targeted by the call.
                       * @param data The call data (encoded using abi.encode or one of its variants).
                       */
                      function _callOptionalReturn(IERC20Upgradeable token, bytes memory data) private {
                          // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since
                          // we're implementing it ourselves. We use {Address-functionCall} to perform this call, which verifies that
                          // the target address contains contract code and also asserts for success in the low-level call.
                          bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed");
                          require(returndata.length == 0 || abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
                      }
                      /**
                       * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement
                       * on the return value: the return value is optional (but if data is returned, it must not be false).
                       * @param token The token targeted by the call.
                       * @param data The call data (encoded using abi.encode or one of its variants).
                       *
                       * This is a variant of {_callOptionalReturn} that silents catches all reverts and returns a bool instead.
                       */
                      function _callOptionalReturnBool(IERC20Upgradeable token, bytes memory data) private returns (bool) {
                          // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since
                          // we're implementing it ourselves. We cannot use {Address-functionCall} here since this should return false
                          // and not revert is the subcall reverts.
                          (bool success, bytes memory returndata) = address(token).call(data);
                          return
                              success && (returndata.length == 0 || abi.decode(returndata, (bool))) && AddressUpgradeable.isContract(address(token));
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.16;
                  interface IL1ETHGateway {
                      /**********
                       * Events *
                       **********/
                      /// @notice Emitted when ETH is withdrawn from L2 to L1 and transfer to recipient.
                      /// @param from The address of sender in L2.
                      /// @param to The address of recipient in L1.
                      /// @param amount The amount of ETH withdrawn from L2 to L1.
                      /// @param data The optional calldata passed to recipient in L1.
                      event FinalizeWithdrawETH(address indexed from, address indexed to, uint256 amount, bytes data);
                      /// @notice Emitted when someone deposit ETH from L1 to L2.
                      /// @param from The address of sender in L1.
                      /// @param to The address of recipient in L2.
                      /// @param amount The amount of ETH will be deposited from L1 to L2.
                      /// @param data The optional calldata passed to recipient in L2.
                      event DepositETH(address indexed from, address indexed to, uint256 amount, bytes data);
                      /// @notice Emitted when some ETH is refunded.
                      /// @param recipient The address of receiver in L1.
                      /// @param amount The amount of ETH refunded to receiver.
                      event RefundETH(address indexed recipient, uint256 amount);
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @notice Deposit ETH to caller's account in L2.
                      /// @param amount The amount of ETH to be deposited.
                      /// @param gasLimit Gas limit required to complete the deposit on L2.
                      function depositETH(uint256 amount, uint256 gasLimit) external payable;
                      /// @notice Deposit ETH to some recipient's account in L2.
                      /// @param to The address of recipient's account on L2.
                      /// @param amount The amount of ETH to be deposited.
                      /// @param gasLimit Gas limit required to complete the deposit on L2.
                      function depositETH(
                          address to,
                          uint256 amount,
                          uint256 gasLimit
                      ) external payable;
                      /// @notice Deposit ETH to some recipient's account in L2 and call the target contract.
                      /// @param to The address of recipient's account on L2.
                      /// @param amount The amount of ETH to be deposited.
                      /// @param data Optional data to forward to recipient's account.
                      /// @param gasLimit Gas limit required to complete the deposit on L2.
                      function depositETHAndCall(
                          address to,
                          uint256 amount,
                          bytes calldata data,
                          uint256 gasLimit
                      ) external payable;
                      /// @notice Complete ETH withdraw from L2 to L1 and send fund to recipient's account in L1.
                      /// @dev This function should only be called by L1ScrollMessenger.
                      ///      This function should also only be called by L1ETHGateway in L2.
                      /// @param from The address of account who withdraw ETH in L2.
                      /// @param to The address of recipient in L1 to receive ETH.
                      /// @param amount The amount of ETH to withdraw.
                      /// @param data Optional data to forward to recipient's account.
                      function finalizeWithdrawETH(
                          address from,
                          address to,
                          uint256 amount,
                          bytes calldata data
                      ) external payable;
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.16;
                  interface IL1ERC20Gateway {
                      /**********
                       * Events *
                       **********/
                      /// @notice Emitted when ERC20 token is withdrawn from L2 to L1 and transfer to recipient.
                      /// @param l1Token The address of the token in L1.
                      /// @param l2Token The address of the token in L2.
                      /// @param from The address of sender in L2.
                      /// @param to The address of recipient in L1.
                      /// @param amount The amount of token withdrawn from L2 to L1.
                      /// @param data The optional calldata passed to recipient in L1.
                      event FinalizeWithdrawERC20(
                          address indexed l1Token,
                          address indexed l2Token,
                          address indexed from,
                          address to,
                          uint256 amount,
                          bytes data
                      );
                      /// @notice Emitted when someone deposit ERC20 token from L1 to L2.
                      /// @param l1Token The address of the token in L1.
                      /// @param l2Token The address of the token in L2.
                      /// @param from The address of sender in L1.
                      /// @param to The address of recipient in L2.
                      /// @param amount The amount of token will be deposited from L1 to L2.
                      /// @param data The optional calldata passed to recipient in L2.
                      event DepositERC20(
                          address indexed l1Token,
                          address indexed l2Token,
                          address indexed from,
                          address to,
                          uint256 amount,
                          bytes data
                      );
                      /// @notice Emitted when some ERC20 token is refunded.
                      /// @param token The address of the token in L1.
                      /// @param recipient The address of receiver in L1.
                      /// @param amount The amount of token refunded to receiver.
                      event RefundERC20(address indexed token, address indexed recipient, uint256 amount);
                      /*************************
                       * Public View Functions *
                       *************************/
                      /// @notice Return the corresponding l2 token address given l1 token address.
                      /// @param _l1Token The address of l1 token.
                      function getL2ERC20Address(address _l1Token) external view returns (address);
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @notice Deposit some token to a caller's account on L2.
                      /// @dev Make this function payable to send relayer fee in Ether.
                      /// @param _token The address of token in L1.
                      /// @param _amount The amount of token to transfer.
                      /// @param _gasLimit Gas limit required to complete the deposit on L2.
                      function depositERC20(
                          address _token,
                          uint256 _amount,
                          uint256 _gasLimit
                      ) external payable;
                      /// @notice Deposit some token to a recipient's account on L2.
                      /// @dev Make this function payable to send relayer fee in Ether.
                      /// @param _token The address of token in L1.
                      /// @param _to The address of recipient's account on L2.
                      /// @param _amount The amount of token to transfer.
                      /// @param _gasLimit Gas limit required to complete the deposit on L2.
                      function depositERC20(
                          address _token,
                          address _to,
                          uint256 _amount,
                          uint256 _gasLimit
                      ) external payable;
                      /// @notice Deposit some token to a recipient's account on L2 and call.
                      /// @dev Make this function payable to send relayer fee in Ether.
                      /// @param _token The address of token in L1.
                      /// @param _to The address of recipient's account on L2.
                      /// @param _amount The amount of token to transfer.
                      /// @param _data Optional data to forward to recipient's account.
                      /// @param _gasLimit Gas limit required to complete the deposit on L2.
                      function depositERC20AndCall(
                          address _token,
                          address _to,
                          uint256 _amount,
                          bytes memory _data,
                          uint256 _gasLimit
                      ) external payable;
                      /// @notice Complete ERC20 withdraw from L2 to L1 and send fund to recipient's account in L1.
                      /// @dev Make this function payable to handle WETH deposit/withdraw.
                      ///      The function should only be called by L1ScrollMessenger.
                      ///      The function should also only be called by L2ERC20Gateway in L2.
                      /// @param _l1Token The address of corresponding L1 token.
                      /// @param _l2Token The address of corresponding L2 token.
                      /// @param _from The address of account who withdraw the token in L2.
                      /// @param _to The address of recipient in L1 to receive the token.
                      /// @param _amount The amount of the token to withdraw.
                      /// @param _data Optional data to forward to recipient's account.
                      function finalizeWithdrawERC20(
                          address _l1Token,
                          address _l2Token,
                          address _from,
                          address _to,
                          uint256 _amount,
                          bytes calldata _data
                      ) external payable;
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.16;
                  import {IL1ETHGateway} from "./IL1ETHGateway.sol";
                  import {IL1ERC20Gateway} from "./IL1ERC20Gateway.sol";
                  interface IL1GatewayRouter is IL1ETHGateway, IL1ERC20Gateway {
                      /**********
                       * Events *
                       **********/
                      /// @notice Emitted when the address of ETH Gateway is updated.
                      /// @param oldETHGateway The address of the old ETH Gateway.
                      /// @param newEthGateway The address of the new ETH Gateway.
                      event SetETHGateway(address indexed oldETHGateway, address indexed newEthGateway);
                      /// @notice Emitted when the address of default ERC20 Gateway is updated.
                      /// @param oldDefaultERC20Gateway The address of the old default ERC20 Gateway.
                      /// @param newDefaultERC20Gateway The address of the new default ERC20 Gateway.
                      event SetDefaultERC20Gateway(address indexed oldDefaultERC20Gateway, address indexed newDefaultERC20Gateway);
                      /// @notice Emitted when the `gateway` for `token` is updated.
                      /// @param token The address of token updated.
                      /// @param oldGateway The corresponding address of the old gateway.
                      /// @param newGateway The corresponding address of the new gateway.
                      event SetERC20Gateway(address indexed token, address indexed oldGateway, address indexed newGateway);
                      /*************************
                       * Public View Functions *
                       *************************/
                      /// @notice Return the corresponding gateway address for given token address.
                      /// @param _token The address of token to query.
                      function getERC20Gateway(address _token) external view returns (address);
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @notice Request ERC20 token transfer from users to gateways.
                      /// @param sender The address of sender to request fund.
                      /// @param token The address of token to request.
                      /// @param amount The amount of token to request.
                      function requestERC20(
                          address sender,
                          address token,
                          uint256 amount
                      ) external returns (uint256);
                      /************************
                       * Restricted Functions *
                       ************************/
                      /// @notice Update the address of ETH gateway contract.
                      /// @dev This function should only be called by contract owner.
                      /// @param _ethGateway The address to update.
                      function setETHGateway(address _ethGateway) external;
                      /// @notice Update the address of default ERC20 gateway contract.
                      /// @dev This function should only be called by contract owner.
                      /// @param _defaultERC20Gateway The address to update.
                      function setDefaultERC20Gateway(address _defaultERC20Gateway) external;
                      /// @notice Update the mapping from token address to gateway address.
                      /// @dev This function should only be called by contract owner.
                      /// @param _tokens The list of addresses of tokens to update.
                      /// @param _gateways The list of addresses of gateways to update.
                      function setERC20Gateway(address[] memory _tokens, address[] memory _gateways) external;
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts v4.4.1 (utils/Context.sol)
                  pragma solidity ^0.8.0;
                  import "../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;
                      }
                      /**
                       * @dev This empty reserved space is put in place to allow future versions to add new
                       * variables without shifting down storage in the inheritance chain.
                       * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
                       */
                      uint256[50] private __gap;
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.9.0) (proxy/utils/Initializable.sol)
                  pragma solidity ^0.8.2;
                  import "../../utils/AddressUpgradeable.sol";
                  /**
                   * @dev This is a base contract to aid in writing upgradeable contracts, or any kind of contract that will be deployed
                   * behind a proxy. Since proxied contracts do not make use of a constructor, it's common to move constructor logic to an
                   * external initializer function, usually called `initialize`. It then becomes necessary to protect this initializer
                   * function so it can only be called once. The {initializer} modifier provided by this contract will have this effect.
                   *
                   * The initialization functions use a version number. Once a version number is used, it is consumed and cannot be
                   * reused. This mechanism prevents re-execution of each "step" but allows the creation of new initialization steps in
                   * case an upgrade adds a module that needs to be initialized.
                   *
                   * For example:
                   *
                   * [.hljs-theme-light.nopadding]
                   * ```solidity
                   * contract MyToken is ERC20Upgradeable {
                   *     function initialize() initializer public {
                   *         __ERC20_init("MyToken", "MTK");
                   *     }
                   * }
                   *
                   * contract MyTokenV2 is MyToken, ERC20PermitUpgradeable {
                   *     function initializeV2() reinitializer(2) public {
                   *         __ERC20Permit_init("MyToken");
                   *     }
                   * }
                   * ```
                   *
                   * TIP: To avoid leaving the proxy in an uninitialized state, the initializer function should be called as early as
                   * possible by providing the encoded function call as the `_data` argument to {ERC1967Proxy-constructor}.
                   *
                   * CAUTION: When used with inheritance, manual care must be taken to not invoke a parent initializer twice, or to ensure
                   * that all initializers are idempotent. This is not verified automatically as constructors are by Solidity.
                   *
                   * [CAUTION]
                   * ====
                   * Avoid leaving a contract uninitialized.
                   *
                   * An uninitialized contract can be taken over by an attacker. This applies to both a proxy and its implementation
                   * contract, which may impact the proxy. To prevent the implementation contract from being used, you should invoke
                   * the {_disableInitializers} function in the constructor to automatically lock it when it is deployed:
                   *
                   * [.hljs-theme-light.nopadding]
                   * ```
                   * /// @custom:oz-upgrades-unsafe-allow constructor
                   * constructor() {
                   *     _disableInitializers();
                   * }
                   * ```
                   * ====
                   */
                  abstract contract Initializable {
                      /**
                       * @dev Indicates that the contract has been initialized.
                       * @custom:oz-retyped-from bool
                       */
                      uint8 private _initialized;
                      /**
                       * @dev Indicates that the contract is in the process of being initialized.
                       */
                      bool private _initializing;
                      /**
                       * @dev Triggered when the contract has been initialized or reinitialized.
                       */
                      event Initialized(uint8 version);
                      /**
                       * @dev A modifier that defines a protected initializer function that can be invoked at most once. In its scope,
                       * `onlyInitializing` functions can be used to initialize parent contracts.
                       *
                       * Similar to `reinitializer(1)`, except that functions marked with `initializer` can be nested in the context of a
                       * constructor.
                       *
                       * Emits an {Initialized} event.
                       */
                      modifier initializer() {
                          bool isTopLevelCall = !_initializing;
                          require(
                              (isTopLevelCall && _initialized < 1) || (!AddressUpgradeable.isContract(address(this)) && _initialized == 1),
                              "Initializable: contract is already initialized"
                          );
                          _initialized = 1;
                          if (isTopLevelCall) {
                              _initializing = true;
                          }
                          _;
                          if (isTopLevelCall) {
                              _initializing = false;
                              emit Initialized(1);
                          }
                      }
                      /**
                       * @dev A modifier that defines a protected reinitializer function that can be invoked at most once, and only if the
                       * contract hasn't been initialized to a greater version before. In its scope, `onlyInitializing` functions can be
                       * used to initialize parent contracts.
                       *
                       * A reinitializer may be used after the original initialization step. This is essential to configure modules that
                       * are added through upgrades and that require initialization.
                       *
                       * When `version` is 1, this modifier is similar to `initializer`, except that functions marked with `reinitializer`
                       * cannot be nested. If one is invoked in the context of another, execution will revert.
                       *
                       * Note that versions can jump in increments greater than 1; this implies that if multiple reinitializers coexist in
                       * a contract, executing them in the right order is up to the developer or operator.
                       *
                       * WARNING: setting the version to 255 will prevent any future reinitialization.
                       *
                       * Emits an {Initialized} event.
                       */
                      modifier reinitializer(uint8 version) {
                          require(!_initializing && _initialized < version, "Initializable: contract is already initialized");
                          _initialized = version;
                          _initializing = true;
                          _;
                          _initializing = false;
                          emit Initialized(version);
                      }
                      /**
                       * @dev Modifier to protect an initialization function so that it can only be invoked by functions with the
                       * {initializer} and {reinitializer} modifiers, directly or indirectly.
                       */
                      modifier onlyInitializing() {
                          require(_initializing, "Initializable: contract is not initializing");
                          _;
                      }
                      /**
                       * @dev Locks the contract, preventing any future reinitialization. This cannot be part of an initializer call.
                       * Calling this in the constructor of a contract will prevent that contract from being initialized or reinitialized
                       * to any version. It is recommended to use this to lock implementation contracts that are designed to be called
                       * through proxies.
                       *
                       * Emits an {Initialized} event the first time it is successfully executed.
                       */
                      function _disableInitializers() internal virtual {
                          require(!_initializing, "Initializable: contract is initializing");
                          if (_initialized != type(uint8).max) {
                              _initialized = type(uint8).max;
                              emit Initialized(type(uint8).max);
                          }
                      }
                      /**
                       * @dev Returns the highest version that has been initialized. See {reinitializer}.
                       */
                      function _getInitializedVersion() internal view returns (uint8) {
                          return _initialized;
                      }
                      /**
                       * @dev Returns `true` if the contract is currently initializing. See {onlyInitializing}.
                       */
                      function _isInitializing() internal view returns (bool) {
                          return _initializing;
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/extensions/IERC20Permit.sol)
                  pragma solidity ^0.8.0;
                  /**
                   * @dev Interface of the ERC20 Permit extension allowing approvals to be made via signatures, as defined in
                   * https://eips.ethereum.org/EIPS/eip-2612[EIP-2612].
                   *
                   * Adds the {permit} method, which can be used to change an account's ERC20 allowance (see {IERC20-allowance}) by
                   * presenting a message signed by the account. By not relying on {IERC20-approve}, the token holder account doesn't
                   * need to send a transaction, and thus is not required to hold Ether at all.
                   */
                  interface IERC20PermitUpgradeable {
                      /**
                       * @dev Sets `value` as the allowance of `spender` over ``owner``'s tokens,
                       * given ``owner``'s signed approval.
                       *
                       * IMPORTANT: The same issues {IERC20-approve} has related to transaction
                       * ordering also apply here.
                       *
                       * Emits an {Approval} event.
                       *
                       * Requirements:
                       *
                       * - `spender` cannot be the zero address.
                       * - `deadline` must be a timestamp in the future.
                       * - `v`, `r` and `s` must be a valid `secp256k1` signature from `owner`
                       * over the EIP712-formatted function arguments.
                       * - the signature must use ``owner``'s current nonce (see {nonces}).
                       *
                       * For more information on the signature format, see the
                       * https://eips.ethereum.org/EIPS/eip-2612#specification[relevant EIP
                       * section].
                       */
                      function permit(
                          address owner,
                          address spender,
                          uint256 value,
                          uint256 deadline,
                          uint8 v,
                          bytes32 r,
                          bytes32 s
                      ) external;
                      /**
                       * @dev Returns the current nonce for `owner`. This value must be
                       * included whenever a signature is generated for {permit}.
                       *
                       * Every successful call to {permit} increases ``owner``'s nonce by one. This
                       * prevents a signature from being used multiple times.
                       */
                      function nonces(address owner) external view returns (uint256);
                      /**
                       * @dev Returns the domain separator used in the encoding of the signature for {permit}, as defined by {EIP712}.
                       */
                      // solhint-disable-next-line func-name-mixedcase
                      function DOMAIN_SEPARATOR() external view returns (bytes32);
                  }
                  // SPDX-License-Identifier: MIT
                  // 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);
                          }
                      }
                  }
                  

                  File 8 of 9: L1CustomERC20Gateway
                  // SPDX-License-Identifier: MIT
                  pragma solidity =0.8.24;
                  import {IERC20Upgradeable} from "@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol";
                  import {SafeERC20Upgradeable} from "@openzeppelin/contracts-upgradeable/token/ERC20/utils/SafeERC20Upgradeable.sol";
                  import {IL2ERC20Gateway} from "../../L2/gateways/IL2ERC20Gateway.sol";
                  import {IL1ScrollMessenger} from "../IL1ScrollMessenger.sol";
                  import {IL1ERC20Gateway} from "./IL1ERC20Gateway.sol";
                  import {ScrollGatewayBase} from "../../libraries/gateway/ScrollGatewayBase.sol";
                  import {L1ERC20Gateway} from "./L1ERC20Gateway.sol";
                  /// @title L1CustomERC20Gateway
                  /// @notice The `L1CustomERC20Gateway` is used to deposit custom ERC20 compatible tokens on layer 1 and
                  /// finalize withdraw the tokens from layer 2.
                  /// @dev The deposited tokens are held in this gateway. On finalizing withdraw, the corresponding
                  /// tokens will be transfer to the recipient directly.
                  contract L1CustomERC20Gateway is L1ERC20Gateway {
                      using SafeERC20Upgradeable for IERC20Upgradeable;
                      /**********
                       * Events *
                       **********/
                      /// @notice Emitted when token mapping for ERC20 token is updated.
                      /// @param l1Token The address of ERC20 token in layer 1.
                      /// @param oldL2Token The address of the old corresponding ERC20 token in layer 2.
                      /// @param newL2Token The address of the new corresponding ERC20 token in layer 2.
                      event UpdateTokenMapping(address indexed l1Token, address indexed oldL2Token, address indexed newL2Token);
                      /*************
                       * Variables *
                       *************/
                      /// @notice Mapping from l1 token address to l2 token address for ERC20 token.
                      mapping(address => address) public tokenMapping;
                      /***************
                       * Constructor *
                       ***************/
                      /// @notice Constructor for `L1CustomERC20Gateway` implementation contract.
                      ///
                      /// @param _counterpart The address of `L2USDCGateway` contract in L2.
                      /// @param _router The address of `L1GatewayRouter` contract in L1.
                      /// @param _messenger The address of `L1ScrollMessenger` contract L1.
                      constructor(
                          address _counterpart,
                          address _router,
                          address _messenger
                      ) ScrollGatewayBase(_counterpart, _router, _messenger) {
                          if (_router == address(0)) revert ErrorZeroAddress();
                          _disableInitializers();
                      }
                      /// @notice Initialize the storage of L1CustomERC20Gateway.
                      ///
                      /// @dev The parameters `_counterpart`, `_router` and `_messenger` are no longer used.
                      ///
                      /// @param _counterpart The address of L2CustomERC20Gateway in L2.
                      /// @param _router The address of L1GatewayRouter in L1.
                      /// @param _messenger The address of L1ScrollMessenger in L1.
                      function initialize(
                          address _counterpart,
                          address _router,
                          address _messenger
                      ) external initializer {
                          ScrollGatewayBase._initialize(_counterpart, _router, _messenger);
                      }
                      /*************************
                       * Public View Functions *
                       *************************/
                      /// @inheritdoc IL1ERC20Gateway
                      function getL2ERC20Address(address _l1Token) public view override returns (address) {
                          return tokenMapping[_l1Token];
                      }
                      /************************
                       * Restricted Functions *
                       ************************/
                      /// @notice Update layer 1 to layer 2 token mapping.
                      /// @param _l1Token The address of ERC20 token on layer 1.
                      /// @param _l2Token The address of corresponding ERC20 token on layer 2.
                      function updateTokenMapping(address _l1Token, address _l2Token) external onlyOwner {
                          require(_l2Token != address(0), "token address cannot be 0");
                          address _oldL2Token = tokenMapping[_l1Token];
                          tokenMapping[_l1Token] = _l2Token;
                          emit UpdateTokenMapping(_l1Token, _oldL2Token, _l2Token);
                      }
                      /**********************
                       * Internal Functions *
                       **********************/
                      /// @inheritdoc L1ERC20Gateway
                      function _beforeFinalizeWithdrawERC20(
                          address _l1Token,
                          address _l2Token,
                          address,
                          address,
                          uint256,
                          bytes calldata
                      ) internal virtual override {
                          require(msg.value == 0, "nonzero msg.value");
                          require(_l2Token != address(0), "token address cannot be 0");
                          require(_l2Token == tokenMapping[_l1Token], "l2 token mismatch");
                      }
                      /// @inheritdoc L1ERC20Gateway
                      function _beforeDropMessage(
                          address,
                          address,
                          uint256
                      ) internal virtual override {
                          require(msg.value == 0, "nonzero msg.value");
                      }
                      /// @inheritdoc L1ERC20Gateway
                      function _deposit(
                          address _token,
                          address _to,
                          uint256 _amount,
                          bytes memory _data,
                          uint256 _gasLimit
                      ) internal virtual override nonReentrant {
                          address _l2Token = tokenMapping[_token];
                          require(_l2Token != address(0), "no corresponding l2 token");
                          // 1. Transfer token into this contract.
                          address _from;
                          (_from, _amount, _data) = _transferERC20In(_token, _amount, _data);
                          // 2. Generate message passed to L2CustomERC20Gateway.
                          bytes memory _message = abi.encodeCall(
                              IL2ERC20Gateway.finalizeDepositERC20,
                              (_token, _l2Token, _from, _to, _amount, _data)
                          );
                          // 3. Send message to L1ScrollMessenger.
                          IL1ScrollMessenger(messenger).sendMessage{value: msg.value}(counterpart, 0, _message, _gasLimit, _from);
                          emit DepositERC20(_token, _l2Token, _from, _to, _amount, _data);
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/IERC20.sol)
                  pragma solidity ^0.8.0;
                  /**
                   * @dev Interface of the ERC20 standard as defined in the EIP.
                   */
                  interface IERC20Upgradeable {
                      /**
                       * @dev Emitted when `value` tokens are moved from one account (`from`) to
                       * another (`to`).
                       *
                       * Note that `value` may be zero.
                       */
                      event Transfer(address indexed from, address indexed to, uint256 value);
                      /**
                       * @dev Emitted when the allowance of a `spender` for an `owner` is set by
                       * a call to {approve}. `value` is the new allowance.
                       */
                      event Approval(address indexed owner, address indexed spender, uint256 value);
                      /**
                       * @dev Returns the amount of tokens in existence.
                       */
                      function totalSupply() external view returns (uint256);
                      /**
                       * @dev Returns the amount of tokens owned by `account`.
                       */
                      function balanceOf(address account) external view returns (uint256);
                      /**
                       * @dev Moves `amount` tokens from the caller's account to `to`.
                       *
                       * Returns a boolean value indicating whether the operation succeeded.
                       *
                       * Emits a {Transfer} event.
                       */
                      function transfer(address to, uint256 amount) external returns (bool);
                      /**
                       * @dev Returns the remaining number of tokens that `spender` will be
                       * allowed to spend on behalf of `owner` through {transferFrom}. This is
                       * zero by default.
                       *
                       * This value changes when {approve} or {transferFrom} are called.
                       */
                      function allowance(address owner, address spender) external view returns (uint256);
                      /**
                       * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
                       *
                       * Returns a boolean value indicating whether the operation succeeded.
                       *
                       * IMPORTANT: Beware that changing an allowance with this method brings the risk
                       * that someone may use both the old and the new allowance by unfortunate
                       * transaction ordering. One possible solution to mitigate this race
                       * condition is to first reduce the spender's allowance to 0 and set the
                       * desired value afterwards:
                       * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
                       *
                       * Emits an {Approval} event.
                       */
                      function approve(address spender, uint256 amount) external returns (bool);
                      /**
                       * @dev Moves `amount` tokens from `from` to `to` using the
                       * allowance mechanism. `amount` is then deducted from the caller's
                       * allowance.
                       *
                       * Returns a boolean value indicating whether the operation succeeded.
                       *
                       * Emits a {Transfer} event.
                       */
                      function transferFrom(address from, address to, uint256 amount) external returns (bool);
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.9.3) (token/ERC20/utils/SafeERC20.sol)
                  pragma solidity ^0.8.0;
                  import "../IERC20Upgradeable.sol";
                  import "../extensions/IERC20PermitUpgradeable.sol";
                  import "../../../utils/AddressUpgradeable.sol";
                  /**
                   * @title SafeERC20
                   * @dev Wrappers around ERC20 operations that throw on failure (when the token
                   * contract returns false). Tokens that return no value (and instead revert or
                   * throw on failure) are also supported, non-reverting calls are assumed to be
                   * successful.
                   * To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract,
                   * which allows you to call the safe operations as `token.safeTransfer(...)`, etc.
                   */
                  library SafeERC20Upgradeable {
                      using AddressUpgradeable for address;
                      /**
                       * @dev Transfer `value` amount of `token` from the calling contract to `to`. If `token` returns no value,
                       * non-reverting calls are assumed to be successful.
                       */
                      function safeTransfer(IERC20Upgradeable token, address to, uint256 value) internal {
                          _callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value));
                      }
                      /**
                       * @dev Transfer `value` amount of `token` from `from` to `to`, spending the approval given by `from` to the
                       * calling contract. If `token` returns no value, non-reverting calls are assumed to be successful.
                       */
                      function safeTransferFrom(IERC20Upgradeable token, address from, address to, uint256 value) internal {
                          _callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value));
                      }
                      /**
                       * @dev Deprecated. This function has issues similar to the ones found in
                       * {IERC20-approve}, and its usage is discouraged.
                       *
                       * Whenever possible, use {safeIncreaseAllowance} and
                       * {safeDecreaseAllowance} instead.
                       */
                      function safeApprove(IERC20Upgradeable token, address spender, uint256 value) internal {
                          // safeApprove should only be called when setting an initial allowance,
                          // or when resetting it to zero. To increase and decrease it, use
                          // 'safeIncreaseAllowance' and 'safeDecreaseAllowance'
                          require(
                              (value == 0) || (token.allowance(address(this), spender) == 0),
                              "SafeERC20: approve from non-zero to non-zero allowance"
                          );
                          _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value));
                      }
                      /**
                       * @dev Increase the calling contract's allowance toward `spender` by `value`. If `token` returns no value,
                       * non-reverting calls are assumed to be successful.
                       */
                      function safeIncreaseAllowance(IERC20Upgradeable token, address spender, uint256 value) internal {
                          uint256 oldAllowance = token.allowance(address(this), spender);
                          _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, oldAllowance + value));
                      }
                      /**
                       * @dev Decrease the calling contract's allowance toward `spender` by `value`. If `token` returns no value,
                       * non-reverting calls are assumed to be successful.
                       */
                      function safeDecreaseAllowance(IERC20Upgradeable token, address spender, uint256 value) internal {
                          unchecked {
                              uint256 oldAllowance = token.allowance(address(this), spender);
                              require(oldAllowance >= value, "SafeERC20: decreased allowance below zero");
                              _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, oldAllowance - value));
                          }
                      }
                      /**
                       * @dev Set the calling contract's allowance toward `spender` to `value`. If `token` returns no value,
                       * non-reverting calls are assumed to be successful. Meant to be used with tokens that require the approval
                       * to be set to zero before setting it to a non-zero value, such as USDT.
                       */
                      function forceApprove(IERC20Upgradeable token, address spender, uint256 value) internal {
                          bytes memory approvalCall = abi.encodeWithSelector(token.approve.selector, spender, value);
                          if (!_callOptionalReturnBool(token, approvalCall)) {
                              _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, 0));
                              _callOptionalReturn(token, approvalCall);
                          }
                      }
                      /**
                       * @dev Use a ERC-2612 signature to set the `owner` approval toward `spender` on `token`.
                       * Revert on invalid signature.
                       */
                      function safePermit(
                          IERC20PermitUpgradeable token,
                          address owner,
                          address spender,
                          uint256 value,
                          uint256 deadline,
                          uint8 v,
                          bytes32 r,
                          bytes32 s
                      ) internal {
                          uint256 nonceBefore = token.nonces(owner);
                          token.permit(owner, spender, value, deadline, v, r, s);
                          uint256 nonceAfter = token.nonces(owner);
                          require(nonceAfter == nonceBefore + 1, "SafeERC20: permit did not succeed");
                      }
                      /**
                       * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement
                       * on the return value: the return value is optional (but if data is returned, it must not be false).
                       * @param token The token targeted by the call.
                       * @param data The call data (encoded using abi.encode or one of its variants).
                       */
                      function _callOptionalReturn(IERC20Upgradeable token, bytes memory data) private {
                          // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since
                          // we're implementing it ourselves. We use {Address-functionCall} to perform this call, which verifies that
                          // the target address contains contract code and also asserts for success in the low-level call.
                          bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed");
                          require(returndata.length == 0 || abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
                      }
                      /**
                       * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement
                       * on the return value: the return value is optional (but if data is returned, it must not be false).
                       * @param token The token targeted by the call.
                       * @param data The call data (encoded using abi.encode or one of its variants).
                       *
                       * This is a variant of {_callOptionalReturn} that silents catches all reverts and returns a bool instead.
                       */
                      function _callOptionalReturnBool(IERC20Upgradeable token, bytes memory data) private returns (bool) {
                          // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since
                          // we're implementing it ourselves. We cannot use {Address-functionCall} here since this should return false
                          // and not revert is the subcall reverts.
                          (bool success, bytes memory returndata) = address(token).call(data);
                          return
                              success && (returndata.length == 0 || abi.decode(returndata, (bool))) && AddressUpgradeable.isContract(address(token));
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.24;
                  interface IL2ERC20Gateway {
                      /**********
                       * Events *
                       **********/
                      /// @notice Emitted when ERC20 token is deposited from L1 to L2 and transfer to recipient.
                      /// @param l1Token The address of the token in L1.
                      /// @param l2Token The address of the token in L2.
                      /// @param from The address of sender in L1.
                      /// @param to The address of recipient in L2.
                      /// @param amount The amount of token withdrawn from L1 to L2.
                      /// @param data The optional calldata passed to recipient in L2.
                      event FinalizeDepositERC20(
                          address indexed l1Token,
                          address indexed l2Token,
                          address indexed from,
                          address to,
                          uint256 amount,
                          bytes data
                      );
                      /// @notice Emitted when someone withdraw ERC20 token from L2 to L1.
                      /// @param l1Token The address of the token in L1.
                      /// @param l2Token The address of the token in L2.
                      /// @param from The address of sender in L2.
                      /// @param to The address of recipient in L1.
                      /// @param amount The amount of token will be deposited from L2 to L1.
                      /// @param data The optional calldata passed to recipient in L1.
                      event WithdrawERC20(
                          address indexed l1Token,
                          address indexed l2Token,
                          address indexed from,
                          address to,
                          uint256 amount,
                          bytes data
                      );
                      /*************************
                       * Public View Functions *
                       *************************/
                      /// @notice Return the corresponding l1 token address given l2 token address.
                      /// @param l2Token The address of l2 token.
                      function getL1ERC20Address(address l2Token) external view returns (address);
                      /// @notice Return the corresponding l2 token address given l1 token address.
                      /// @param l1Token The address of l1 token.
                      function getL2ERC20Address(address l1Token) external view returns (address);
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @notice Withdraw of some token to a caller's account on L1.
                      /// @dev Make this function payable to send relayer fee in Ether.
                      /// @param token The address of token in L2.
                      /// @param amount The amount of token to transfer.
                      /// @param gasLimit Unused, but included for potential forward compatibility considerations.
                      function withdrawERC20(
                          address token,
                          uint256 amount,
                          uint256 gasLimit
                      ) external payable;
                      /// @notice Withdraw of some token to a recipient's account on L1.
                      /// @dev Make this function payable to send relayer fee in Ether.
                      /// @param token The address of token in L2.
                      /// @param to The address of recipient's account on L1.
                      /// @param amount The amount of token to transfer.
                      /// @param gasLimit Unused, but included for potential forward compatibility considerations.
                      function withdrawERC20(
                          address token,
                          address to,
                          uint256 amount,
                          uint256 gasLimit
                      ) external payable;
                      /// @notice Withdraw of some token to a recipient's account on L1 and call.
                      /// @dev Make this function payable to send relayer fee in Ether.
                      /// @param token The address of token in L2.
                      /// @param to The address of recipient's account on L1.
                      /// @param amount The amount of token to transfer.
                      /// @param data Optional data to forward to recipient's account.
                      /// @param gasLimit Unused, but included for potential forward compatibility considerations.
                      function withdrawERC20AndCall(
                          address token,
                          address to,
                          uint256 amount,
                          bytes calldata data,
                          uint256 gasLimit
                      ) external payable;
                      /// @notice Complete a deposit from L1 to L2 and send fund to recipient's account in L2.
                      /// @dev Make this function payable to handle WETH deposit/withdraw.
                      ///      The function should only be called by L2ScrollMessenger.
                      ///      The function should also only be called by L1ERC20Gateway in L1.
                      /// @param l1Token The address of corresponding L1 token.
                      /// @param l2Token The address of corresponding L2 token.
                      /// @param from The address of account who deposits the token in L1.
                      /// @param to The address of recipient in L2 to receive the token.
                      /// @param amount The amount of the token to deposit.
                      /// @param data Optional data to forward to recipient's account.
                      function finalizeDepositERC20(
                          address l1Token,
                          address l2Token,
                          address from,
                          address to,
                          uint256 amount,
                          bytes calldata data
                      ) external payable;
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.24;
                  import {IScrollMessenger} from "../libraries/IScrollMessenger.sol";
                  interface IL1ScrollMessenger is IScrollMessenger {
                      /**********
                       * Events *
                       **********/
                      /// @notice Emitted when the maximum number of times each message can be replayed is updated.
                      /// @param oldMaxReplayTimes The old maximum number of times each message can be replayed.
                      /// @param newMaxReplayTimes The new maximum number of times each message can be replayed.
                      event UpdateMaxReplayTimes(uint256 oldMaxReplayTimes, uint256 newMaxReplayTimes);
                      /***********
                       * Structs *
                       ***********/
                      struct L2MessageProof {
                          // The index of the batch where the message belongs to.
                          uint256 batchIndex;
                          // Concatenation of merkle proof for withdraw merkle trie.
                          bytes merkleProof;
                      }
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @notice Relay a L2 => L1 message with message proof.
                      /// @param from The address of the sender of the message.
                      /// @param to The address of the recipient of the message.
                      /// @param value The msg.value passed to the message call.
                      /// @param nonce The nonce of the message to avoid replay attack.
                      /// @param message The content of the message.
                      /// @param proof The proof used to verify the correctness of the transaction.
                      function relayMessageWithProof(
                          address from,
                          address to,
                          uint256 value,
                          uint256 nonce,
                          bytes memory message,
                          L2MessageProof memory proof
                      ) external;
                      /// @notice Replay an existing message.
                      /// @param from The address of the sender of the message.
                      /// @param to The address of the recipient of the message.
                      /// @param value The msg.value passed to the message call.
                      /// @param messageNonce The nonce for the message to replay.
                      /// @param message The content of the message.
                      /// @param newGasLimit New gas limit to be used for this message.
                      /// @param refundAddress The address of account who will receive the refunded fee.
                      function replayMessage(
                          address from,
                          address to,
                          uint256 value,
                          uint256 messageNonce,
                          bytes memory message,
                          uint32 newGasLimit,
                          address refundAddress
                      ) external payable;
                      /// @notice Drop a skipped message.
                      /// @param from The address of the sender of the message.
                      /// @param to The address of the recipient of the message.
                      /// @param value The msg.value passed to the message call.
                      /// @param messageNonce The nonce for the message to drop.
                      /// @param message The content of the message.
                      function dropMessage(
                          address from,
                          address to,
                          uint256 value,
                          uint256 messageNonce,
                          bytes memory message
                      ) external;
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.24;
                  interface IL1ERC20Gateway {
                      /**********
                       * Events *
                       **********/
                      /// @notice Emitted when ERC20 token is withdrawn from L2 to L1 and transfer to recipient.
                      /// @param l1Token The address of the token in L1.
                      /// @param l2Token The address of the token in L2.
                      /// @param from The address of sender in L2.
                      /// @param to The address of recipient in L1.
                      /// @param amount The amount of token withdrawn from L2 to L1.
                      /// @param data The optional calldata passed to recipient in L1.
                      event FinalizeWithdrawERC20(
                          address indexed l1Token,
                          address indexed l2Token,
                          address indexed from,
                          address to,
                          uint256 amount,
                          bytes data
                      );
                      /// @notice Emitted when someone deposit ERC20 token from L1 to L2.
                      /// @param l1Token The address of the token in L1.
                      /// @param l2Token The address of the token in L2.
                      /// @param from The address of sender in L1.
                      /// @param to The address of recipient in L2.
                      /// @param amount The amount of token will be deposited from L1 to L2.
                      /// @param data The optional calldata passed to recipient in L2.
                      event DepositERC20(
                          address indexed l1Token,
                          address indexed l2Token,
                          address indexed from,
                          address to,
                          uint256 amount,
                          bytes data
                      );
                      /// @notice Emitted when some ERC20 token is refunded.
                      /// @param token The address of the token in L1.
                      /// @param recipient The address of receiver in L1.
                      /// @param amount The amount of token refunded to receiver.
                      event RefundERC20(address indexed token, address indexed recipient, uint256 amount);
                      /*************************
                       * Public View Functions *
                       *************************/
                      /// @notice Return the corresponding l2 token address given l1 token address.
                      /// @param _l1Token The address of l1 token.
                      function getL2ERC20Address(address _l1Token) external view returns (address);
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @notice Deposit some token to a caller's account on L2.
                      /// @dev Make this function payable to send relayer fee in Ether.
                      /// @param _token The address of token in L1.
                      /// @param _amount The amount of token to transfer.
                      /// @param _gasLimit Gas limit required to complete the deposit on L2.
                      function depositERC20(
                          address _token,
                          uint256 _amount,
                          uint256 _gasLimit
                      ) external payable;
                      /// @notice Deposit some token to a recipient's account on L2.
                      /// @dev Make this function payable to send relayer fee in Ether.
                      /// @param _token The address of token in L1.
                      /// @param _to The address of recipient's account on L2.
                      /// @param _amount The amount of token to transfer.
                      /// @param _gasLimit Gas limit required to complete the deposit on L2.
                      function depositERC20(
                          address _token,
                          address _to,
                          uint256 _amount,
                          uint256 _gasLimit
                      ) external payable;
                      /// @notice Deposit some token to a recipient's account on L2 and call.
                      /// @dev Make this function payable to send relayer fee in Ether.
                      /// @param _token The address of token in L1.
                      /// @param _to The address of recipient's account on L2.
                      /// @param _amount The amount of token to transfer.
                      /// @param _data Optional data to forward to recipient's account.
                      /// @param _gasLimit Gas limit required to complete the deposit on L2.
                      function depositERC20AndCall(
                          address _token,
                          address _to,
                          uint256 _amount,
                          bytes memory _data,
                          uint256 _gasLimit
                      ) external payable;
                      /// @notice Complete ERC20 withdraw from L2 to L1 and send fund to recipient's account in L1.
                      /// @dev Make this function payable to handle WETH deposit/withdraw.
                      ///      The function should only be called by L1ScrollMessenger.
                      ///      The function should also only be called by L2ERC20Gateway in L2.
                      /// @param _l1Token The address of corresponding L1 token.
                      /// @param _l2Token The address of corresponding L2 token.
                      /// @param _from The address of account who withdraw the token in L2.
                      /// @param _to The address of recipient in L1 to receive the token.
                      /// @param _amount The amount of the token to withdraw.
                      /// @param _data Optional data to forward to recipient's account.
                      function finalizeWithdrawERC20(
                          address _l1Token,
                          address _l2Token,
                          address _from,
                          address _to,
                          uint256 _amount,
                          bytes calldata _data
                      ) external payable;
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.24;
                  import {OwnableUpgradeable} from "@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol";
                  import {ReentrancyGuardUpgradeable} from "@openzeppelin/contracts-upgradeable/security/ReentrancyGuardUpgradeable.sol";
                  import {IScrollGateway} from "./IScrollGateway.sol";
                  import {IScrollMessenger} from "../IScrollMessenger.sol";
                  import {IScrollGatewayCallback} from "../callbacks/IScrollGatewayCallback.sol";
                  import {ScrollConstants} from "../constants/ScrollConstants.sol";
                  import {ITokenRateLimiter} from "../../rate-limiter/ITokenRateLimiter.sol";
                  /// @title ScrollGatewayBase
                  /// @notice The `ScrollGatewayBase` is a base contract for gateway contracts used in both in L1 and L2.
                  abstract contract ScrollGatewayBase is ReentrancyGuardUpgradeable, OwnableUpgradeable, IScrollGateway {
                      /*************
                       * Constants *
                       *************/
                      /// @inheritdoc IScrollGateway
                      address public immutable override counterpart;
                      /// @inheritdoc IScrollGateway
                      address public immutable override router;
                      /// @inheritdoc IScrollGateway
                      address public immutable override messenger;
                      /*************
                       * Variables *
                       *************/
                      /// @dev The storage slot used as counterpart gateway contract, which is deprecated now.
                      address private __counterpart;
                      /// @dev The storage slot used as gateway router contract, which is deprecated now.
                      address private __router;
                      /// @dev The storage slot used as scroll messenger contract, which is deprecated now.
                      address private __messenger;
                      /// @dev The storage slot used as token rate limiter contract, which is deprecated now.
                      address private __rateLimiter;
                      /// @dev The storage slots for future usage.
                      uint256[46] private __gap;
                      /**********************
                       * Function Modifiers *
                       **********************/
                      modifier onlyCallByCounterpart() {
                          // check caller is messenger
                          if (_msgSender() != messenger) {
                              revert ErrorCallerIsNotMessenger();
                          }
                          // check cross domain caller is counterpart gateway
                          if (counterpart != IScrollMessenger(messenger).xDomainMessageSender()) {
                              revert ErrorCallerIsNotCounterpartGateway();
                          }
                          _;
                      }
                      modifier onlyInDropContext() {
                          // check caller is messenger
                          if (_msgSender() != messenger) {
                              revert ErrorCallerIsNotMessenger();
                          }
                          // check we are dropping message in ScrollMessenger.
                          if (ScrollConstants.DROP_XDOMAIN_MESSAGE_SENDER != IScrollMessenger(messenger).xDomainMessageSender()) {
                              revert ErrorNotInDropMessageContext();
                          }
                          _;
                      }
                      /***************
                       * Constructor *
                       ***************/
                      constructor(
                          address _counterpart,
                          address _router,
                          address _messenger
                      ) {
                          if (_counterpart == address(0) || _messenger == address(0)) {
                              revert ErrorZeroAddress();
                          }
                          counterpart = _counterpart;
                          router = _router;
                          messenger = _messenger;
                      }
                      function _initialize(
                          address,
                          address,
                          address
                      ) internal {
                          ReentrancyGuardUpgradeable.__ReentrancyGuard_init();
                          OwnableUpgradeable.__Ownable_init();
                      }
                      /**********************
                       * Internal Functions *
                       **********************/
                      /// @dev Internal function to forward calldata to target contract.
                      /// @param _to The address of contract to call.
                      /// @param _data The calldata passed to the contract.
                      function _doCallback(address _to, bytes memory _data) internal {
                          if (_data.length > 0 && _to.code.length > 0) {
                              IScrollGatewayCallback(_to).onScrollGatewayCallback(_data);
                          }
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.24;
                  import {IERC20Upgradeable} from "@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol";
                  import {SafeERC20Upgradeable} from "@openzeppelin/contracts-upgradeable/token/ERC20/utils/SafeERC20Upgradeable.sol";
                  import {IL1ERC20Gateway} from "./IL1ERC20Gateway.sol";
                  import {IL1GatewayRouter} from "./IL1GatewayRouter.sol";
                  import {IL2ERC20Gateway} from "../../L2/gateways/IL2ERC20Gateway.sol";
                  import {ScrollGatewayBase} from "../../libraries/gateway/ScrollGatewayBase.sol";
                  import {IMessageDropCallback} from "../../libraries/callbacks/IMessageDropCallback.sol";
                  /// @title L1ERC20Gateway
                  /// @notice The `L1ERC20Gateway` as a base contract for ERC20 gateways in L1.
                  /// It has implementation of common used functions for ERC20 gateways.
                  abstract contract L1ERC20Gateway is IL1ERC20Gateway, IMessageDropCallback, ScrollGatewayBase {
                      using SafeERC20Upgradeable for IERC20Upgradeable;
                      /*************
                       * Variables *
                       *************/
                      /// @dev The storage slots for future usage.
                      uint256[50] private __gap;
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @inheritdoc IL1ERC20Gateway
                      function depositERC20(
                          address _token,
                          uint256 _amount,
                          uint256 _gasLimit
                      ) external payable override {
                          _deposit(_token, _msgSender(), _amount, new bytes(0), _gasLimit);
                      }
                      /// @inheritdoc IL1ERC20Gateway
                      function depositERC20(
                          address _token,
                          address _to,
                          uint256 _amount,
                          uint256 _gasLimit
                      ) external payable override {
                          _deposit(_token, _to, _amount, new bytes(0), _gasLimit);
                      }
                      /// @inheritdoc IL1ERC20Gateway
                      function depositERC20AndCall(
                          address _token,
                          address _to,
                          uint256 _amount,
                          bytes memory _data,
                          uint256 _gasLimit
                      ) external payable override {
                          _deposit(_token, _to, _amount, _data, _gasLimit);
                      }
                      /// @inheritdoc IL1ERC20Gateway
                      function finalizeWithdrawERC20(
                          address _l1Token,
                          address _l2Token,
                          address _from,
                          address _to,
                          uint256 _amount,
                          bytes calldata _data
                      ) external payable virtual override onlyCallByCounterpart nonReentrant {
                          _beforeFinalizeWithdrawERC20(_l1Token, _l2Token, _from, _to, _amount, _data);
                          // @note can possible trigger reentrant call to this contract or messenger,
                          // but it seems not a big problem.
                          IERC20Upgradeable(_l1Token).safeTransfer(_to, _amount);
                          _doCallback(_to, _data);
                          emit FinalizeWithdrawERC20(_l1Token, _l2Token, _from, _to, _amount, _data);
                      }
                      /// @inheritdoc IMessageDropCallback
                      function onDropMessage(bytes calldata _message) external payable virtual onlyInDropContext nonReentrant {
                          // _message should start with 0x8431f5c1  =>  finalizeDepositERC20(address,address,address,address,uint256,bytes)
                          require(bytes4(_message[0:4]) == IL2ERC20Gateway.finalizeDepositERC20.selector, "invalid selector");
                          // decode (token, receiver, amount)
                          (address _token, , address _receiver, , uint256 _amount, ) = abi.decode(
                              _message[4:],
                              (address, address, address, address, uint256, bytes)
                          );
                          // do dome check for each custom gateway
                          _beforeDropMessage(_token, _receiver, _amount);
                          IERC20Upgradeable(_token).safeTransfer(_receiver, _amount);
                          emit RefundERC20(_token, _receiver, _amount);
                      }
                      /**********************
                       * Internal Functions *
                       **********************/
                      /// @dev Internal function hook to perform checks and actions before finalizing the withdrawal.
                      /// @param _l1Token The address of corresponding L1 token in L1.
                      /// @param _l2Token The address of corresponding L2 token in L2.
                      /// @param _from The address of account who withdraw the token in L2.
                      /// @param _to The address of recipient in L1 to receive the token.
                      /// @param _amount The amount of the token to withdraw.
                      /// @param _data Optional data to forward to recipient's account.
                      function _beforeFinalizeWithdrawERC20(
                          address _l1Token,
                          address _l2Token,
                          address _from,
                          address _to,
                          uint256 _amount,
                          bytes calldata _data
                      ) internal virtual;
                      /// @dev Internal function hook to perform checks and actions before dropping the message.
                      /// @param _token The L1 token address.
                      /// @param _receiver The recipient address on L1.
                      /// @param _amount The amount of token to refund.
                      function _beforeDropMessage(
                          address _token,
                          address _receiver,
                          uint256 _amount
                      ) internal virtual;
                      /// @dev Internal function to transfer ERC20 token to this contract.
                      /// @param _token The address of token to transfer.
                      /// @param _amount The amount of token to transfer.
                      /// @param _data The data passed by caller.
                      function _transferERC20In(
                          address _token,
                          uint256 _amount,
                          bytes memory _data
                      )
                          internal
                          returns (
                              address,
                              uint256,
                              bytes memory
                          )
                      {
                          address _sender = _msgSender();
                          address _from = _sender;
                          if (router == _sender) {
                              // Extract real sender if this call is from L1GatewayRouter.
                              (_from, _data) = abi.decode(_data, (address, bytes));
                              _amount = IL1GatewayRouter(_sender).requestERC20(_from, _token, _amount);
                          } else {
                              // common practice to handle fee on transfer token.
                              uint256 _before = IERC20Upgradeable(_token).balanceOf(address(this));
                              IERC20Upgradeable(_token).safeTransferFrom(_from, address(this), _amount);
                              uint256 _after = IERC20Upgradeable(_token).balanceOf(address(this));
                              // no unchecked here, since some weird token may return arbitrary balance.
                              _amount = _after - _before;
                          }
                          // ignore weird fee on transfer token
                          require(_amount > 0, "deposit zero amount");
                          return (_from, _amount, _data);
                      }
                      /// @dev Internal function to do all the deposit operations.
                      ///
                      /// @param _token The token to deposit.
                      /// @param _to The recipient address to recieve the token in L2.
                      /// @param _amount The amount of token to deposit.
                      /// @param _data Optional data to forward to recipient's account.
                      /// @param _gasLimit Gas limit required to complete the deposit on L2.
                      function _deposit(
                          address _token,
                          address _to,
                          uint256 _amount,
                          bytes memory _data,
                          uint256 _gasLimit
                      ) internal virtual;
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/extensions/IERC20Permit.sol)
                  pragma solidity ^0.8.0;
                  /**
                   * @dev Interface of the ERC20 Permit extension allowing approvals to be made via signatures, as defined in
                   * https://eips.ethereum.org/EIPS/eip-2612[EIP-2612].
                   *
                   * Adds the {permit} method, which can be used to change an account's ERC20 allowance (see {IERC20-allowance}) by
                   * presenting a message signed by the account. By not relying on {IERC20-approve}, the token holder account doesn't
                   * need to send a transaction, and thus is not required to hold Ether at all.
                   */
                  interface IERC20PermitUpgradeable {
                      /**
                       * @dev Sets `value` as the allowance of `spender` over ``owner``'s tokens,
                       * given ``owner``'s signed approval.
                       *
                       * IMPORTANT: The same issues {IERC20-approve} has related to transaction
                       * ordering also apply here.
                       *
                       * Emits an {Approval} event.
                       *
                       * Requirements:
                       *
                       * - `spender` cannot be the zero address.
                       * - `deadline` must be a timestamp in the future.
                       * - `v`, `r` and `s` must be a valid `secp256k1` signature from `owner`
                       * over the EIP712-formatted function arguments.
                       * - the signature must use ``owner``'s current nonce (see {nonces}).
                       *
                       * For more information on the signature format, see the
                       * https://eips.ethereum.org/EIPS/eip-2612#specification[relevant EIP
                       * section].
                       */
                      function permit(
                          address owner,
                          address spender,
                          uint256 value,
                          uint256 deadline,
                          uint8 v,
                          bytes32 r,
                          bytes32 s
                      ) external;
                      /**
                       * @dev Returns the current nonce for `owner`. This value must be
                       * included whenever a signature is generated for {permit}.
                       *
                       * Every successful call to {permit} increases ``owner``'s nonce by one. This
                       * prevents a signature from being used multiple times.
                       */
                      function nonces(address owner) external view returns (uint256);
                      /**
                       * @dev Returns the domain separator used in the encoding of the signature for {permit}, as defined by {EIP712}.
                       */
                      // solhint-disable-next-line func-name-mixedcase
                      function DOMAIN_SEPARATOR() external view returns (bytes32);
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.9.0) (utils/Address.sol)
                  pragma solidity ^0.8.1;
                  /**
                   * @dev Collection of functions related to the address type
                   */
                  library AddressUpgradeable {
                      /**
                       * @dev Returns true if `account` is a contract.
                       *
                       * [IMPORTANT]
                       * ====
                       * It is unsafe to assume that an address for which this function returns
                       * false is an externally-owned account (EOA) and not a contract.
                       *
                       * Among others, `isContract` will return false for the following
                       * types of addresses:
                       *
                       *  - an externally-owned account
                       *  - a contract in construction
                       *  - an address where a contract will be created
                       *  - an address where a contract lived, but was destroyed
                       *
                       * Furthermore, `isContract` will also return true if the target contract within
                       * the same transaction is already scheduled for destruction by `SELFDESTRUCT`,
                       * which only has an effect at the end of a transaction.
                       * ====
                       *
                       * [IMPORTANT]
                       * ====
                       * You shouldn't rely on `isContract` to protect against flash loan attacks!
                       *
                       * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets
                       * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract
                       * constructor.
                       * ====
                       */
                      function isContract(address account) internal view returns (bool) {
                          // This method relies on extcodesize/address.code.length, which returns 0
                          // for contracts in construction, since the code is only stored at the end
                          // of the constructor execution.
                          return account.code.length > 0;
                      }
                      /**
                       * @dev Replacement for Solidity's `transfer`: sends `amount` wei to
                       * `recipient`, forwarding all available gas and reverting on errors.
                       *
                       * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
                       * of certain opcodes, possibly making contracts go over the 2300 gas limit
                       * imposed by `transfer`, making them unable to receive funds via
                       * `transfer`. {sendValue} removes this limitation.
                       *
                       * https://consensys.net/diligence/blog/2019/09/stop-using-soliditys-transfer-now/[Learn more].
                       *
                       * IMPORTANT: because control is transferred to `recipient`, care must be
                       * taken to not create reentrancy vulnerabilities. Consider using
                       * {ReentrancyGuard} or the
                       * https://solidity.readthedocs.io/en/v0.8.0/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
                       */
                      function sendValue(address payable recipient, uint256 amount) internal {
                          require(address(this).balance >= amount, "Address: insufficient balance");
                          (bool success, ) = recipient.call{value: amount}("");
                          require(success, "Address: unable to send value, recipient may have reverted");
                      }
                      /**
                       * @dev Performs a Solidity function call using a low level `call`. A
                       * plain `call` is an unsafe replacement for a function call: use this
                       * function instead.
                       *
                       * If `target` reverts with a revert reason, it is bubbled up by this
                       * function (like regular Solidity function calls).
                       *
                       * Returns the raw returned data. To convert to the expected return value,
                       * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
                       *
                       * Requirements:
                       *
                       * - `target` must be a contract.
                       * - calling `target` with `data` must not revert.
                       *
                       * _Available since v3.1._
                       */
                      function functionCall(address target, bytes memory data) internal returns (bytes memory) {
                          return functionCallWithValue(target, data, 0, "Address: low-level call failed");
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
                       * `errorMessage` as a fallback revert reason when `target` reverts.
                       *
                       * _Available since v3.1._
                       */
                      function functionCall(
                          address target,
                          bytes memory data,
                          string memory errorMessage
                      ) internal returns (bytes memory) {
                          return functionCallWithValue(target, data, 0, errorMessage);
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                       * but also transferring `value` wei to `target`.
                       *
                       * Requirements:
                       *
                       * - the calling contract must have an ETH balance of at least `value`.
                       * - the called Solidity function must be `payable`.
                       *
                       * _Available since v3.1._
                       */
                      function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
                          return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
                      }
                      /**
                       * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
                       * with `errorMessage` as a fallback revert reason when `target` reverts.
                       *
                       * _Available since v3.1._
                       */
                      function functionCallWithValue(
                          address target,
                          bytes memory data,
                          uint256 value,
                          string memory errorMessage
                      ) internal returns (bytes memory) {
                          require(address(this).balance >= value, "Address: insufficient balance for call");
                          (bool success, bytes memory returndata) = target.call{value: value}(data);
                          return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                       * but performing a static call.
                       *
                       * _Available since v3.3._
                       */
                      function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
                          return functionStaticCall(target, data, "Address: low-level static call failed");
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                       * but performing a static call.
                       *
                       * _Available since v3.3._
                       */
                      function functionStaticCall(
                          address target,
                          bytes memory data,
                          string memory errorMessage
                      ) internal view returns (bytes memory) {
                          (bool success, bytes memory returndata) = target.staticcall(data);
                          return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                       * but performing a delegate call.
                       *
                       * _Available since v3.4._
                       */
                      function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
                          return functionDelegateCall(target, data, "Address: low-level delegate call failed");
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                       * but performing a delegate call.
                       *
                       * _Available since v3.4._
                       */
                      function functionDelegateCall(
                          address target,
                          bytes memory data,
                          string memory errorMessage
                      ) internal returns (bytes memory) {
                          (bool success, bytes memory returndata) = target.delegatecall(data);
                          return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                      }
                      /**
                       * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling
                       * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract.
                       *
                       * _Available since v4.8._
                       */
                      function verifyCallResultFromTarget(
                          address target,
                          bool success,
                          bytes memory returndata,
                          string memory errorMessage
                      ) internal view returns (bytes memory) {
                          if (success) {
                              if (returndata.length == 0) {
                                  // only check isContract if the call was successful and the return data is empty
                                  // otherwise we already know that it was a contract
                                  require(isContract(target), "Address: call to non-contract");
                              }
                              return returndata;
                          } else {
                              _revert(returndata, errorMessage);
                          }
                      }
                      /**
                       * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the
                       * revert reason or using the provided one.
                       *
                       * _Available since v4.3._
                       */
                      function verifyCallResult(
                          bool success,
                          bytes memory returndata,
                          string memory errorMessage
                      ) internal pure returns (bytes memory) {
                          if (success) {
                              return returndata;
                          } else {
                              _revert(returndata, errorMessage);
                          }
                      }
                      function _revert(bytes memory returndata, string memory errorMessage) private pure {
                          // Look for revert reason and bubble it up if present
                          if (returndata.length > 0) {
                              // The easiest way to bubble the revert reason is using memory via assembly
                              /// @solidity memory-safe-assembly
                              assembly {
                                  let returndata_size := mload(returndata)
                                  revert(add(32, returndata), returndata_size)
                              }
                          } else {
                              revert(errorMessage);
                          }
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.24;
                  interface IScrollMessenger {
                      /**********
                       * Events *
                       **********/
                      /// @notice Emitted when a cross domain message is sent.
                      /// @param sender The address of the sender who initiates the message.
                      /// @param target The address of target contract to call.
                      /// @param value The amount of value passed to the target contract.
                      /// @param messageNonce The nonce of the message.
                      /// @param gasLimit The optional gas limit passed to L1 or L2.
                      /// @param message The calldata passed to the target contract.
                      event SentMessage(
                          address indexed sender,
                          address indexed target,
                          uint256 value,
                          uint256 messageNonce,
                          uint256 gasLimit,
                          bytes message
                      );
                      /// @notice Emitted when a cross domain message is relayed successfully.
                      /// @param messageHash The hash of the message.
                      event RelayedMessage(bytes32 indexed messageHash);
                      /// @notice Emitted when a cross domain message is failed to relay.
                      /// @param messageHash The hash of the message.
                      event FailedRelayedMessage(bytes32 indexed messageHash);
                      /**********
                       * Errors *
                       **********/
                      /// @dev Thrown when the given address is `address(0)`.
                      error ErrorZeroAddress();
                      /*************************
                       * Public View Functions *
                       *************************/
                      /// @notice Return the sender of a cross domain message.
                      function xDomainMessageSender() external view returns (address);
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @notice Send cross chain message from L1 to L2 or L2 to L1.
                      /// @param target The address of account who receive the message.
                      /// @param value The amount of ether passed when call target contract.
                      /// @param message The content of the message.
                      /// @param gasLimit Gas limit required to complete the message relay on corresponding chain.
                      function sendMessage(
                          address target,
                          uint256 value,
                          bytes calldata message,
                          uint256 gasLimit
                      ) external payable;
                      /// @notice Send cross chain message from L1 to L2 or L2 to L1.
                      /// @param target The address of account who receive the message.
                      /// @param value The amount of ether passed when call target contract.
                      /// @param message The content of the message.
                      /// @param gasLimit Gas limit required to complete the message relay on corresponding chain.
                      /// @param refundAddress The address of account who will receive the refunded fee.
                      function sendMessage(
                          address target,
                          uint256 value,
                          bytes calldata message,
                          uint256 gasLimit,
                          address refundAddress
                      ) external payable;
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.9.0) (access/Ownable.sol)
                  pragma solidity ^0.8.0;
                  import "../utils/ContextUpgradeable.sol";
                  import "../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) (security/ReentrancyGuard.sol)
                  pragma solidity ^0.8.0;
                  import "../proxy/utils/Initializable.sol";
                  /**
                   * @dev Contract module that helps prevent reentrant calls to a function.
                   *
                   * Inheriting from `ReentrancyGuard` will make the {nonReentrant} modifier
                   * available, which can be applied to functions to make sure there are no nested
                   * (reentrant) calls to them.
                   *
                   * Note that because there is a single `nonReentrant` guard, functions marked as
                   * `nonReentrant` may not call one another. This can be worked around by making
                   * those functions `private`, and then adding `external` `nonReentrant` entry
                   * points to them.
                   *
                   * TIP: If you would like to learn more about reentrancy and alternative ways
                   * to protect against it, check out our blog post
                   * https://blog.openzeppelin.com/reentrancy-after-istanbul/[Reentrancy After Istanbul].
                   */
                  abstract contract ReentrancyGuardUpgradeable is Initializable {
                      // Booleans are more expensive than uint256 or any type that takes up a full
                      // word because each write operation emits an extra SLOAD to first read the
                      // slot's contents, replace the bits taken up by the boolean, and then write
                      // back. This is the compiler's defense against contract upgrades and
                      // pointer aliasing, and it cannot be disabled.
                      // The values being non-zero value makes deployment a bit more expensive,
                      // but in exchange the refund on every call to nonReentrant will be lower in
                      // amount. Since refunds are capped to a percentage of the total
                      // transaction's gas, it is best to keep them low in cases like this one, to
                      // increase the likelihood of the full refund coming into effect.
                      uint256 private constant _NOT_ENTERED = 1;
                      uint256 private constant _ENTERED = 2;
                      uint256 private _status;
                      function __ReentrancyGuard_init() internal onlyInitializing {
                          __ReentrancyGuard_init_unchained();
                      }
                      function __ReentrancyGuard_init_unchained() internal onlyInitializing {
                          _status = _NOT_ENTERED;
                      }
                      /**
                       * @dev Prevents a contract from calling itself, directly or indirectly.
                       * Calling a `nonReentrant` function from another `nonReentrant`
                       * function is not supported. It is possible to prevent this from happening
                       * by making the `nonReentrant` function external, and making it call a
                       * `private` function that does the actual work.
                       */
                      modifier nonReentrant() {
                          _nonReentrantBefore();
                          _;
                          _nonReentrantAfter();
                      }
                      function _nonReentrantBefore() private {
                          // On the first call to nonReentrant, _status will be _NOT_ENTERED
                          require(_status != _ENTERED, "ReentrancyGuard: reentrant call");
                          // Any calls to nonReentrant after this point will fail
                          _status = _ENTERED;
                      }
                      function _nonReentrantAfter() private {
                          // By storing the original value once again, a refund is triggered (see
                          // https://eips.ethereum.org/EIPS/eip-2200)
                          _status = _NOT_ENTERED;
                      }
                      /**
                       * @dev Returns true if the reentrancy guard is currently set to "entered", which indicates there is a
                       * `nonReentrant` function in the call stack.
                       */
                      function _reentrancyGuardEntered() internal view returns (bool) {
                          return _status == _ENTERED;
                      }
                      /**
                       * @dev This empty reserved space is put in place to allow future versions to add new
                       * variables without shifting down storage in the inheritance chain.
                       * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
                       */
                      uint256[49] private __gap;
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.24;
                  interface IScrollGateway {
                      /**********
                       * Errors *
                       **********/
                      /// @dev Thrown when the given address is `address(0)`.
                      error ErrorZeroAddress();
                      /// @dev Thrown when the caller is not corresponding `L1ScrollMessenger` or `L2ScrollMessenger`.
                      error ErrorCallerIsNotMessenger();
                      /// @dev Thrown when the cross chain sender is not the counterpart gateway contract.
                      error ErrorCallerIsNotCounterpartGateway();
                      /// @dev Thrown when ScrollMessenger is not dropping message.
                      error ErrorNotInDropMessageContext();
                      /*************************
                       * Public View Functions *
                       *************************/
                      /// @notice The address of corresponding L1/L2 Gateway contract.
                      function counterpart() external view returns (address);
                      /// @notice The address of L1GatewayRouter/L2GatewayRouter contract.
                      function router() external view returns (address);
                      /// @notice The address of corresponding L1ScrollMessenger/L2ScrollMessenger contract.
                      function messenger() external view returns (address);
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.24;
                  interface IScrollGatewayCallback {
                      function onScrollGatewayCallback(bytes memory data) external;
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.24;
                  library ScrollConstants {
                      /// @notice The address of default cross chain message sender.
                      address internal constant DEFAULT_XDOMAIN_MESSAGE_SENDER = address(1);
                      /// @notice The address for dropping message.
                      /// @dev The first 20 bytes of keccak("drop")
                      address internal constant DROP_XDOMAIN_MESSAGE_SENDER = 0x6f297C61B5C92eF107fFD30CD56AFFE5A273e841;
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.24;
                  interface ITokenRateLimiter {
                      /**********
                       * Events *
                       **********/
                      /// @notice Emitted when the total limit is updated.
                      /// @param oldTotalLimit The previous value of total limit before updating.
                      /// @param newTotalLimit The current value of total limit after updating.
                      event UpdateTotalLimit(address indexed token, uint256 oldTotalLimit, uint256 newTotalLimit);
                      /**********
                       * Errors *
                       **********/
                      /// @dev Thrown when the `periodDuration` is initialized to zero.
                      error PeriodIsZero();
                      /// @dev Thrown when the `totalAmount` is initialized to zero.
                      /// @param token The address of the token.
                      error TotalLimitIsZero(address token);
                      /// @dev Thrown when an amount breaches the total limit in the period.
                      /// @param token The address of the token.
                      error ExceedTotalLimit(address token);
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @notice Request some token usage for `sender`.
                      /// @param token The address of the token.
                      /// @param amount The amount of token to use.
                      function addUsedAmount(address token, uint256 amount) external;
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.24;
                  import {IL1ETHGateway} from "./IL1ETHGateway.sol";
                  import {IL1ERC20Gateway} from "./IL1ERC20Gateway.sol";
                  interface IL1GatewayRouter is IL1ETHGateway, IL1ERC20Gateway {
                      /**********
                       * Events *
                       **********/
                      /// @notice Emitted when the address of ETH Gateway is updated.
                      /// @param oldETHGateway The address of the old ETH Gateway.
                      /// @param newEthGateway The address of the new ETH Gateway.
                      event SetETHGateway(address indexed oldETHGateway, address indexed newEthGateway);
                      /// @notice Emitted when the address of default ERC20 Gateway is updated.
                      /// @param oldDefaultERC20Gateway The address of the old default ERC20 Gateway.
                      /// @param newDefaultERC20Gateway The address of the new default ERC20 Gateway.
                      event SetDefaultERC20Gateway(address indexed oldDefaultERC20Gateway, address indexed newDefaultERC20Gateway);
                      /// @notice Emitted when the `gateway` for `token` is updated.
                      /// @param token The address of token updated.
                      /// @param oldGateway The corresponding address of the old gateway.
                      /// @param newGateway The corresponding address of the new gateway.
                      event SetERC20Gateway(address indexed token, address indexed oldGateway, address indexed newGateway);
                      /*************************
                       * Public View Functions *
                       *************************/
                      /// @notice Return the corresponding gateway address for given token address.
                      /// @param _token The address of token to query.
                      function getERC20Gateway(address _token) external view returns (address);
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @notice Request ERC20 token transfer from users to gateways.
                      /// @param sender The address of sender to request fund.
                      /// @param token The address of token to request.
                      /// @param amount The amount of token to request.
                      function requestERC20(
                          address sender,
                          address token,
                          uint256 amount
                      ) external returns (uint256);
                      /************************
                       * Restricted Functions *
                       ************************/
                      /// @notice Update the address of ETH gateway contract.
                      /// @dev This function should only be called by contract owner.
                      /// @param _ethGateway The address to update.
                      function setETHGateway(address _ethGateway) external;
                      /// @notice Update the address of default ERC20 gateway contract.
                      /// @dev This function should only be called by contract owner.
                      /// @param _defaultERC20Gateway The address to update.
                      function setDefaultERC20Gateway(address _defaultERC20Gateway) external;
                      /// @notice Update the mapping from token address to gateway address.
                      /// @dev This function should only be called by contract owner.
                      /// @param _tokens The list of addresses of tokens to update.
                      /// @param _gateways The list of addresses of gateways to update.
                      function setERC20Gateway(address[] calldata _tokens, address[] calldata _gateways) external;
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.24;
                  interface IMessageDropCallback {
                      function onDropMessage(bytes memory message) external payable;
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts v4.4.1 (utils/Context.sol)
                  pragma solidity ^0.8.0;
                  import "../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;
                      }
                      /**
                       * @dev This empty reserved space is put in place to allow future versions to add new
                       * variables without shifting down storage in the inheritance chain.
                       * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
                       */
                      uint256[50] private __gap;
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.9.0) (proxy/utils/Initializable.sol)
                  pragma solidity ^0.8.2;
                  import "../../utils/AddressUpgradeable.sol";
                  /**
                   * @dev This is a base contract to aid in writing upgradeable contracts, or any kind of contract that will be deployed
                   * behind a proxy. Since proxied contracts do not make use of a constructor, it's common to move constructor logic to an
                   * external initializer function, usually called `initialize`. It then becomes necessary to protect this initializer
                   * function so it can only be called once. The {initializer} modifier provided by this contract will have this effect.
                   *
                   * The initialization functions use a version number. Once a version number is used, it is consumed and cannot be
                   * reused. This mechanism prevents re-execution of each "step" but allows the creation of new initialization steps in
                   * case an upgrade adds a module that needs to be initialized.
                   *
                   * For example:
                   *
                   * [.hljs-theme-light.nopadding]
                   * ```solidity
                   * contract MyToken is ERC20Upgradeable {
                   *     function initialize() initializer public {
                   *         __ERC20_init("MyToken", "MTK");
                   *     }
                   * }
                   *
                   * contract MyTokenV2 is MyToken, ERC20PermitUpgradeable {
                   *     function initializeV2() reinitializer(2) public {
                   *         __ERC20Permit_init("MyToken");
                   *     }
                   * }
                   * ```
                   *
                   * TIP: To avoid leaving the proxy in an uninitialized state, the initializer function should be called as early as
                   * possible by providing the encoded function call as the `_data` argument to {ERC1967Proxy-constructor}.
                   *
                   * CAUTION: When used with inheritance, manual care must be taken to not invoke a parent initializer twice, or to ensure
                   * that all initializers are idempotent. This is not verified automatically as constructors are by Solidity.
                   *
                   * [CAUTION]
                   * ====
                   * Avoid leaving a contract uninitialized.
                   *
                   * An uninitialized contract can be taken over by an attacker. This applies to both a proxy and its implementation
                   * contract, which may impact the proxy. To prevent the implementation contract from being used, you should invoke
                   * the {_disableInitializers} function in the constructor to automatically lock it when it is deployed:
                   *
                   * [.hljs-theme-light.nopadding]
                   * ```
                   * /// @custom:oz-upgrades-unsafe-allow constructor
                   * constructor() {
                   *     _disableInitializers();
                   * }
                   * ```
                   * ====
                   */
                  abstract contract Initializable {
                      /**
                       * @dev Indicates that the contract has been initialized.
                       * @custom:oz-retyped-from bool
                       */
                      uint8 private _initialized;
                      /**
                       * @dev Indicates that the contract is in the process of being initialized.
                       */
                      bool private _initializing;
                      /**
                       * @dev Triggered when the contract has been initialized or reinitialized.
                       */
                      event Initialized(uint8 version);
                      /**
                       * @dev A modifier that defines a protected initializer function that can be invoked at most once. In its scope,
                       * `onlyInitializing` functions can be used to initialize parent contracts.
                       *
                       * Similar to `reinitializer(1)`, except that functions marked with `initializer` can be nested in the context of a
                       * constructor.
                       *
                       * Emits an {Initialized} event.
                       */
                      modifier initializer() {
                          bool isTopLevelCall = !_initializing;
                          require(
                              (isTopLevelCall && _initialized < 1) || (!AddressUpgradeable.isContract(address(this)) && _initialized == 1),
                              "Initializable: contract is already initialized"
                          );
                          _initialized = 1;
                          if (isTopLevelCall) {
                              _initializing = true;
                          }
                          _;
                          if (isTopLevelCall) {
                              _initializing = false;
                              emit Initialized(1);
                          }
                      }
                      /**
                       * @dev A modifier that defines a protected reinitializer function that can be invoked at most once, and only if the
                       * contract hasn't been initialized to a greater version before. In its scope, `onlyInitializing` functions can be
                       * used to initialize parent contracts.
                       *
                       * A reinitializer may be used after the original initialization step. This is essential to configure modules that
                       * are added through upgrades and that require initialization.
                       *
                       * When `version` is 1, this modifier is similar to `initializer`, except that functions marked with `reinitializer`
                       * cannot be nested. If one is invoked in the context of another, execution will revert.
                       *
                       * Note that versions can jump in increments greater than 1; this implies that if multiple reinitializers coexist in
                       * a contract, executing them in the right order is up to the developer or operator.
                       *
                       * WARNING: setting the version to 255 will prevent any future reinitialization.
                       *
                       * Emits an {Initialized} event.
                       */
                      modifier reinitializer(uint8 version) {
                          require(!_initializing && _initialized < version, "Initializable: contract is already initialized");
                          _initialized = version;
                          _initializing = true;
                          _;
                          _initializing = false;
                          emit Initialized(version);
                      }
                      /**
                       * @dev Modifier to protect an initialization function so that it can only be invoked by functions with the
                       * {initializer} and {reinitializer} modifiers, directly or indirectly.
                       */
                      modifier onlyInitializing() {
                          require(_initializing, "Initializable: contract is not initializing");
                          _;
                      }
                      /**
                       * @dev Locks the contract, preventing any future reinitialization. This cannot be part of an initializer call.
                       * Calling this in the constructor of a contract will prevent that contract from being initialized or reinitialized
                       * to any version. It is recommended to use this to lock implementation contracts that are designed to be called
                       * through proxies.
                       *
                       * Emits an {Initialized} event the first time it is successfully executed.
                       */
                      function _disableInitializers() internal virtual {
                          require(!_initializing, "Initializable: contract is initializing");
                          if (_initialized != type(uint8).max) {
                              _initialized = type(uint8).max;
                              emit Initialized(type(uint8).max);
                          }
                      }
                      /**
                       * @dev Returns the highest version that has been initialized. See {reinitializer}.
                       */
                      function _getInitializedVersion() internal view returns (uint8) {
                          return _initialized;
                      }
                      /**
                       * @dev Returns `true` if the contract is currently initializing. See {onlyInitializing}.
                       */
                      function _isInitializing() internal view returns (bool) {
                          return _initializing;
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.24;
                  interface IL1ETHGateway {
                      /**********
                       * Events *
                       **********/
                      /// @notice Emitted when ETH is withdrawn from L2 to L1 and transfer to recipient.
                      /// @param from The address of sender in L2.
                      /// @param to The address of recipient in L1.
                      /// @param amount The amount of ETH withdrawn from L2 to L1.
                      /// @param data The optional calldata passed to recipient in L1.
                      event FinalizeWithdrawETH(address indexed from, address indexed to, uint256 amount, bytes data);
                      /// @notice Emitted when someone deposit ETH from L1 to L2.
                      /// @param from The address of sender in L1.
                      /// @param to The address of recipient in L2.
                      /// @param amount The amount of ETH will be deposited from L1 to L2.
                      /// @param data The optional calldata passed to recipient in L2.
                      event DepositETH(address indexed from, address indexed to, uint256 amount, bytes data);
                      /// @notice Emitted when some ETH is refunded.
                      /// @param recipient The address of receiver in L1.
                      /// @param amount The amount of ETH refunded to receiver.
                      event RefundETH(address indexed recipient, uint256 amount);
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @notice Deposit ETH to caller's account in L2.
                      /// @param amount The amount of ETH to be deposited.
                      /// @param gasLimit Gas limit required to complete the deposit on L2.
                      function depositETH(uint256 amount, uint256 gasLimit) external payable;
                      /// @notice Deposit ETH to some recipient's account in L2.
                      /// @param to The address of recipient's account on L2.
                      /// @param amount The amount of ETH to be deposited.
                      /// @param gasLimit Gas limit required to complete the deposit on L2.
                      function depositETH(
                          address to,
                          uint256 amount,
                          uint256 gasLimit
                      ) external payable;
                      /// @notice Deposit ETH to some recipient's account in L2 and call the target contract.
                      /// @param to The address of recipient's account on L2.
                      /// @param amount The amount of ETH to be deposited.
                      /// @param data Optional data to forward to recipient's account.
                      /// @param gasLimit Gas limit required to complete the deposit on L2.
                      function depositETHAndCall(
                          address to,
                          uint256 amount,
                          bytes calldata data,
                          uint256 gasLimit
                      ) external payable;
                      /// @notice Complete ETH withdraw from L2 to L1 and send fund to recipient's account in L1.
                      /// @dev This function should only be called by L1ScrollMessenger.
                      ///      This function should also only be called by L1ETHGateway in L2.
                      /// @param from The address of account who withdraw ETH in L2.
                      /// @param to The address of recipient in L1 to receive ETH.
                      /// @param amount The amount of ETH to withdraw.
                      /// @param data Optional data to forward to recipient's account.
                      function finalizeWithdrawETH(
                          address from,
                          address to,
                          uint256 amount,
                          bytes calldata data
                      ) external payable;
                  }
                  

                  File 9 of 9: L1ScrollMessenger
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.9.0) (access/Ownable.sol)
                  pragma solidity ^0.8.0;
                  import "../utils/ContextUpgradeable.sol";
                  import "../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/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
                  // OpenZeppelin Contracts (last updated v4.7.0) (security/Pausable.sol)
                  pragma solidity ^0.8.0;
                  import "../utils/ContextUpgradeable.sol";
                  import "../proxy/utils/Initializable.sol";
                  /**
                   * @dev Contract module which allows children to implement an emergency stop
                   * mechanism that can be triggered by an authorized account.
                   *
                   * This module is used through inheritance. It will make available the
                   * modifiers `whenNotPaused` and `whenPaused`, which can be applied to
                   * the functions of your contract. Note that they will not be pausable by
                   * simply including this module, only once the modifiers are put in place.
                   */
                  abstract contract PausableUpgradeable is Initializable, ContextUpgradeable {
                      /**
                       * @dev Emitted when the pause is triggered by `account`.
                       */
                      event Paused(address account);
                      /**
                       * @dev Emitted when the pause is lifted by `account`.
                       */
                      event Unpaused(address account);
                      bool private _paused;
                      /**
                       * @dev Initializes the contract in unpaused state.
                       */
                      function __Pausable_init() internal onlyInitializing {
                          __Pausable_init_unchained();
                      }
                      function __Pausable_init_unchained() internal onlyInitializing {
                          _paused = false;
                      }
                      /**
                       * @dev Modifier to make a function callable only when the contract is not paused.
                       *
                       * Requirements:
                       *
                       * - The contract must not be paused.
                       */
                      modifier whenNotPaused() {
                          _requireNotPaused();
                          _;
                      }
                      /**
                       * @dev Modifier to make a function callable only when the contract is paused.
                       *
                       * Requirements:
                       *
                       * - The contract must be paused.
                       */
                      modifier whenPaused() {
                          _requirePaused();
                          _;
                      }
                      /**
                       * @dev Returns true if the contract is paused, and false otherwise.
                       */
                      function paused() public view virtual returns (bool) {
                          return _paused;
                      }
                      /**
                       * @dev Throws if the contract is paused.
                       */
                      function _requireNotPaused() internal view virtual {
                          require(!paused(), "Pausable: paused");
                      }
                      /**
                       * @dev Throws if the contract is not paused.
                       */
                      function _requirePaused() internal view virtual {
                          require(paused(), "Pausable: not paused");
                      }
                      /**
                       * @dev Triggers stopped state.
                       *
                       * Requirements:
                       *
                       * - The contract must not be paused.
                       */
                      function _pause() internal virtual whenNotPaused {
                          _paused = true;
                          emit Paused(_msgSender());
                      }
                      /**
                       * @dev Returns to normal state.
                       *
                       * Requirements:
                       *
                       * - The contract must be paused.
                       */
                      function _unpause() internal virtual whenPaused {
                          _paused = false;
                          emit Unpaused(_msgSender());
                      }
                      /**
                       * @dev This empty reserved space is put in place to allow future versions to add new
                       * variables without shifting down storage in the inheritance chain.
                       * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
                       */
                      uint256[49] private __gap;
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.9.0) (security/ReentrancyGuard.sol)
                  pragma solidity ^0.8.0;
                  import "../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) (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 v4.4.1 (utils/Context.sol)
                  pragma solidity ^0.8.0;
                  import "../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;
                      }
                      /**
                       * @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
                  pragma solidity ^0.8.16;
                  import {IScrollMessenger} from "../libraries/IScrollMessenger.sol";
                  interface IL1ScrollMessenger is IScrollMessenger {
                      /**********
                       * Events *
                       **********/
                      /// @notice Emitted when the maximum number of times each message can be replayed is updated.
                      /// @param oldMaxReplayTimes The old maximum number of times each message can be replayed.
                      /// @param newMaxReplayTimes The new maximum number of times each message can be replayed.
                      event UpdateMaxReplayTimes(uint256 oldMaxReplayTimes, uint256 newMaxReplayTimes);
                      /***********
                       * Structs *
                       ***********/
                      struct L2MessageProof {
                          // The index of the batch where the message belongs to.
                          uint256 batchIndex;
                          // Concatenation of merkle proof for withdraw merkle trie.
                          bytes merkleProof;
                      }
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @notice Relay a L2 => L1 message with message proof.
                      /// @param from The address of the sender of the message.
                      /// @param to The address of the recipient of the message.
                      /// @param value The msg.value passed to the message call.
                      /// @param nonce The nonce of the message to avoid replay attack.
                      /// @param message The content of the message.
                      /// @param proof The proof used to verify the correctness of the transaction.
                      function relayMessageWithProof(
                          address from,
                          address to,
                          uint256 value,
                          uint256 nonce,
                          bytes memory message,
                          L2MessageProof memory proof
                      ) external;
                      /// @notice Replay an existing message.
                      /// @param from The address of the sender of the message.
                      /// @param to The address of the recipient of the message.
                      /// @param value The msg.value passed to the message call.
                      /// @param messageNonce The nonce for the message to replay.
                      /// @param message The content of the message.
                      /// @param newGasLimit New gas limit to be used for this message.
                      /// @param refundAddress The address of account who will receive the refunded fee.
                      function replayMessage(
                          address from,
                          address to,
                          uint256 value,
                          uint256 messageNonce,
                          bytes memory message,
                          uint32 newGasLimit,
                          address refundAddress
                      ) external payable;
                      /// @notice Drop a skipped message.
                      /// @param from The address of the sender of the message.
                      /// @param to The address of the recipient of the message.
                      /// @param value The msg.value passed to the message call.
                      /// @param messageNonce The nonce for the message to drop.
                      /// @param message The content of the message.
                      function dropMessage(
                          address from,
                          address to,
                          uint256 value,
                          uint256 messageNonce,
                          bytes memory message
                      ) external;
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity =0.8.16;
                  import {IScrollChain} from "./rollup/IScrollChain.sol";
                  import {IL1MessageQueue} from "./rollup/IL1MessageQueue.sol";
                  import {IL1ScrollMessenger} from "./IL1ScrollMessenger.sol";
                  import {ScrollConstants} from "../libraries/constants/ScrollConstants.sol";
                  import {IScrollMessenger} from "../libraries/IScrollMessenger.sol";
                  import {ScrollMessengerBase} from "../libraries/ScrollMessengerBase.sol";
                  import {WithdrawTrieVerifier} from "../libraries/verifier/WithdrawTrieVerifier.sol";
                  import {IMessageDropCallback} from "../libraries/callbacks/IMessageDropCallback.sol";
                  // solhint-disable avoid-low-level-calls
                  // solhint-disable not-rely-on-time
                  // solhint-disable reason-string
                  /// @title L1ScrollMessenger
                  /// @notice The `L1ScrollMessenger` contract can:
                  ///
                  /// 1. send messages from layer 1 to layer 2;
                  /// 2. relay messages from layer 2 layer 1;
                  /// 3. replay failed message by replacing the gas limit;
                  /// 4. drop expired message due to sequencer problems.
                  ///
                  /// @dev All deposited Ether (including `WETH` deposited throng `L1WETHGateway`) will locked in
                  /// this contract.
                  contract L1ScrollMessenger is ScrollMessengerBase, IL1ScrollMessenger {
                      /*************
                       * Constants *
                       *************/
                      /// @notice The address of Rollup contract.
                      address public immutable rollup;
                      /// @notice The address of L1MessageQueue contract.
                      address public immutable messageQueue;
                      /***********
                       * Structs *
                       ***********/
                      struct ReplayState {
                          // The number of replayed times.
                          uint128 times;
                          // The queue index of lastest replayed one. If it is zero, it means the message has not been replayed.
                          uint128 lastIndex;
                      }
                      /*************
                       * Variables *
                       *************/
                      /// @notice Mapping from L1 message hash to the timestamp when the message is sent.
                      mapping(bytes32 => uint256) public messageSendTimestamp;
                      /// @notice Mapping from L2 message hash to a boolean value indicating if the message has been successfully executed.
                      mapping(bytes32 => bool) public isL2MessageExecuted;
                      /// @notice Mapping from L1 message hash to drop status.
                      mapping(bytes32 => bool) public isL1MessageDropped;
                      /// @dev The storage slot used as Rollup contract, which is deprecated now.
                      address private __rollup;
                      /// @dev The storage slot used as L1MessageQueue contract, which is deprecated now.
                      address private __messageQueue;
                      /// @notice The maximum number of times each L1 message can be replayed.
                      uint256 public maxReplayTimes;
                      /// @notice Mapping from L1 message hash to replay state.
                      mapping(bytes32 => ReplayState) public replayStates;
                      /// @notice Mapping from queue index to previous replay queue index.
                      ///
                      /// @dev If a message `x` was replayed 3 times with index `q1`, `q2` and `q3`, the
                      /// value of `prevReplayIndex` and `replayStates` will be `replayStates[hash(x)].lastIndex = q3`,
                      /// `replayStates[hash(x)].times = 3`, `prevReplayIndex[q3] = q2`, `prevReplayIndex[q2] = q1`,
                      /// `prevReplayIndex[q1] = x` and `prevReplayIndex[x]=nil`.
                      ///
                      /// @dev The index `x` that `prevReplayIndex[x]=nil` is used as the termination of the list.
                      /// Usually we use `0` to represent `nil`, but we cannot distinguish it with the first message
                      /// with index zero. So a nonzero offset `1` is added to the value of `prevReplayIndex[x]` to
                      /// avoid such situation.
                      mapping(uint256 => uint256) public prevReplayIndex;
                      /***************
                       * Constructor *
                       ***************/
                      constructor(
                          address _counterpart,
                          address _rollup,
                          address _messageQueue
                      ) ScrollMessengerBase(_counterpart) {
                          if (_rollup == address(0) || _messageQueue == address(0)) {
                              revert ErrorZeroAddress();
                          }
                          _disableInitializers();
                          rollup = _rollup;
                          messageQueue = _messageQueue;
                      }
                      /// @notice Initialize the storage of L1ScrollMessenger.
                      ///
                      /// @dev The parameters `_counterpart`, `_rollup` and `_messageQueue` are no longer used.
                      ///
                      /// @param _counterpart The address of L2ScrollMessenger contract in L2.
                      /// @param _feeVault The address of fee vault, which will be used to collect relayer fee.
                      /// @param _rollup The address of ScrollChain contract.
                      /// @param _messageQueue The address of L1MessageQueue contract.
                      function initialize(
                          address _counterpart,
                          address _feeVault,
                          address _rollup,
                          address _messageQueue
                      ) public initializer {
                          ScrollMessengerBase.__ScrollMessengerBase_init(_counterpart, _feeVault);
                          __rollup = _rollup;
                          __messageQueue = _messageQueue;
                          maxReplayTimes = 3;
                          emit UpdateMaxReplayTimes(0, 3);
                      }
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @inheritdoc IScrollMessenger
                      function sendMessage(
                          address _to,
                          uint256 _value,
                          bytes memory _message,
                          uint256 _gasLimit
                      ) external payable override whenNotPaused {
                          _sendMessage(_to, _value, _message, _gasLimit, _msgSender());
                      }
                      /// @inheritdoc IScrollMessenger
                      function sendMessage(
                          address _to,
                          uint256 _value,
                          bytes calldata _message,
                          uint256 _gasLimit,
                          address _refundAddress
                      ) external payable override whenNotPaused {
                          _sendMessage(_to, _value, _message, _gasLimit, _refundAddress);
                      }
                      /// @inheritdoc IL1ScrollMessenger
                      function relayMessageWithProof(
                          address _from,
                          address _to,
                          uint256 _value,
                          uint256 _nonce,
                          bytes memory _message,
                          L2MessageProof memory _proof
                      ) external override whenNotPaused notInExecution {
                          bytes32 _xDomainCalldataHash = keccak256(_encodeXDomainCalldata(_from, _to, _value, _nonce, _message));
                          require(!isL2MessageExecuted[_xDomainCalldataHash], "Message was already successfully executed");
                          {
                              require(IScrollChain(rollup).isBatchFinalized(_proof.batchIndex), "Batch is not finalized");
                              bytes32 _messageRoot = IScrollChain(rollup).withdrawRoots(_proof.batchIndex);
                              require(
                                  WithdrawTrieVerifier.verifyMerkleProof(_messageRoot, _xDomainCalldataHash, _nonce, _proof.merkleProof),
                                  "Invalid proof"
                              );
                          }
                          // @note check more `_to` address to avoid attack in the future when we add more gateways.
                          require(_to != messageQueue, "Forbid to call message queue");
                          _validateTargetAddress(_to);
                          // @note This usually will never happen, just in case.
                          require(_from != xDomainMessageSender, "Invalid message sender");
                          xDomainMessageSender = _from;
                          (bool success, ) = _to.call{value: _value}(_message);
                          // reset value to refund gas.
                          xDomainMessageSender = ScrollConstants.DEFAULT_XDOMAIN_MESSAGE_SENDER;
                          if (success) {
                              isL2MessageExecuted[_xDomainCalldataHash] = true;
                              emit RelayedMessage(_xDomainCalldataHash);
                          } else {
                              emit FailedRelayedMessage(_xDomainCalldataHash);
                          }
                      }
                      /// @inheritdoc IL1ScrollMessenger
                      function replayMessage(
                          address _from,
                          address _to,
                          uint256 _value,
                          uint256 _messageNonce,
                          bytes memory _message,
                          uint32 _newGasLimit,
                          address _refundAddress
                      ) external payable override whenNotPaused notInExecution {
                          // We will use a different `queueIndex` for the replaced message. However, the original `queueIndex` or `nonce`
                          // is encoded in the `_message`. We will check the `xDomainCalldata` on layer 2 to avoid duplicated execution.
                          // So, only one message will succeed on layer 2. If one of the message is executed successfully, the other one
                          // will revert with "Message was already successfully executed".
                          bytes memory _xDomainCalldata = _encodeXDomainCalldata(_from, _to, _value, _messageNonce, _message);
                          bytes32 _xDomainCalldataHash = keccak256(_xDomainCalldata);
                          require(messageSendTimestamp[_xDomainCalldataHash] > 0, "Provided message has not been enqueued");
                          // cannot replay dropped message
                          require(!isL1MessageDropped[_xDomainCalldataHash], "Message already dropped");
                          // compute and deduct the messaging fee to fee vault.
                          uint256 _fee = IL1MessageQueue(messageQueue).estimateCrossDomainMessageFee(_newGasLimit);
                          // charge relayer fee
                          require(msg.value >= _fee, "Insufficient msg.value for fee");
                          if (_fee > 0) {
                              (bool _success, ) = feeVault.call{value: _fee}("");
                              require(_success, "Failed to deduct the fee");
                          }
                          // enqueue the new transaction
                          uint256 _nextQueueIndex = IL1MessageQueue(messageQueue).nextCrossDomainMessageIndex();
                          IL1MessageQueue(messageQueue).appendCrossDomainMessage(counterpart, _newGasLimit, _xDomainCalldata);
                          ReplayState memory _replayState = replayStates[_xDomainCalldataHash];
                          // update the replayed message chain.
                          unchecked {
                              if (_replayState.lastIndex == 0) {
                                  // the message has not been replayed before.
                                  prevReplayIndex[_nextQueueIndex] = _messageNonce + 1;
                              } else {
                                  prevReplayIndex[_nextQueueIndex] = _replayState.lastIndex + 1;
                              }
                          }
                          _replayState.lastIndex = uint128(_nextQueueIndex);
                          // update replay times
                          require(_replayState.times < maxReplayTimes, "Exceed maximum replay times");
                          unchecked {
                              _replayState.times += 1;
                          }
                          replayStates[_xDomainCalldataHash] = _replayState;
                          // refund fee to `_refundAddress`
                          unchecked {
                              uint256 _refund = msg.value - _fee;
                              if (_refund > 0) {
                                  (bool _success, ) = _refundAddress.call{value: _refund}("");
                                  require(_success, "Failed to refund the fee");
                              }
                          }
                      }
                      /// @inheritdoc IL1ScrollMessenger
                      function dropMessage(
                          address _from,
                          address _to,
                          uint256 _value,
                          uint256 _messageNonce,
                          bytes memory _message
                      ) external override whenNotPaused notInExecution {
                          // The criteria for dropping a message:
                          // 1. The message is a L1 message.
                          // 2. The message has not been dropped before.
                          // 3. the message and all of its replacement are finalized in L1.
                          // 4. the message and all of its replacement are skipped.
                          //
                          // Possible denial of service attack:
                          // + replayMessage is called every time someone want to drop the message.
                          // + replayMessage is called so many times for a skipped message, thus results a long list.
                          //
                          // We limit the number of `replayMessage` calls of each message, which may solve the above problem.
                          // check message exists
                          bytes memory _xDomainCalldata = _encodeXDomainCalldata(_from, _to, _value, _messageNonce, _message);
                          bytes32 _xDomainCalldataHash = keccak256(_xDomainCalldata);
                          require(messageSendTimestamp[_xDomainCalldataHash] > 0, "Provided message has not been enqueued");
                          // check message not dropped
                          require(!isL1MessageDropped[_xDomainCalldataHash], "Message already dropped");
                          // check message is finalized
                          uint256 _lastIndex = replayStates[_xDomainCalldataHash].lastIndex;
                          if (_lastIndex == 0) _lastIndex = _messageNonce;
                          // check message is skipped and drop it.
                          // @note If the list is very long, the message may never be dropped.
                          while (true) {
                              IL1MessageQueue(messageQueue).dropCrossDomainMessage(_lastIndex);
                              _lastIndex = prevReplayIndex[_lastIndex];
                              if (_lastIndex == 0) break;
                              unchecked {
                                  _lastIndex = _lastIndex - 1;
                              }
                          }
                          isL1MessageDropped[_xDomainCalldataHash] = true;
                          // set execution context
                          xDomainMessageSender = ScrollConstants.DROP_XDOMAIN_MESSAGE_SENDER;
                          IMessageDropCallback(_from).onDropMessage{value: _value}(_message);
                          // clear execution context
                          xDomainMessageSender = ScrollConstants.DEFAULT_XDOMAIN_MESSAGE_SENDER;
                      }
                      /************************
                       * Restricted Functions *
                       ************************/
                      /// @notice Update max replay times.
                      /// @dev This function can only called by contract owner.
                      /// @param _newMaxReplayTimes The new max replay times.
                      function updateMaxReplayTimes(uint256 _newMaxReplayTimes) external onlyOwner {
                          uint256 _oldMaxReplayTimes = maxReplayTimes;
                          maxReplayTimes = _newMaxReplayTimes;
                          emit UpdateMaxReplayTimes(_oldMaxReplayTimes, _newMaxReplayTimes);
                      }
                      /**********************
                       * Internal Functions *
                       **********************/
                      function _sendMessage(
                          address _to,
                          uint256 _value,
                          bytes memory _message,
                          uint256 _gasLimit,
                          address _refundAddress
                      ) internal nonReentrant {
                          // compute the actual cross domain message calldata.
                          uint256 _messageNonce = IL1MessageQueue(messageQueue).nextCrossDomainMessageIndex();
                          bytes memory _xDomainCalldata = _encodeXDomainCalldata(_msgSender(), _to, _value, _messageNonce, _message);
                          // compute and deduct the messaging fee to fee vault.
                          uint256 _fee = IL1MessageQueue(messageQueue).estimateCrossDomainMessageFee(_gasLimit);
                          require(msg.value >= _fee + _value, "Insufficient msg.value");
                          if (_fee > 0) {
                              (bool _success, ) = feeVault.call{value: _fee}("");
                              require(_success, "Failed to deduct the fee");
                          }
                          // append message to L1MessageQueue
                          IL1MessageQueue(messageQueue).appendCrossDomainMessage(counterpart, _gasLimit, _xDomainCalldata);
                          // record the message hash for future use.
                          bytes32 _xDomainCalldataHash = keccak256(_xDomainCalldata);
                          // normally this won't happen, since each message has different nonce, but just in case.
                          require(messageSendTimestamp[_xDomainCalldataHash] == 0, "Duplicated message");
                          messageSendTimestamp[_xDomainCalldataHash] = block.timestamp;
                          emit SentMessage(_msgSender(), _to, _value, _messageNonce, _gasLimit, _message);
                          // refund fee to `_refundAddress`
                          unchecked {
                              uint256 _refund = msg.value - _fee - _value;
                              if (_refund > 0) {
                                  (bool _success, ) = _refundAddress.call{value: _refund}("");
                                  require(_success, "Failed to refund the fee");
                              }
                          }
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.16;
                  interface IL1MessageQueue {
                      /**********
                       * Events *
                       **********/
                      /// @notice Emitted when a new L1 => L2 transaction is appended to the queue.
                      /// @param sender The address of account who initiates the transaction.
                      /// @param target The address of account who will receive the transaction.
                      /// @param value The value passed with the transaction.
                      /// @param queueIndex The index of this transaction in the queue.
                      /// @param gasLimit Gas limit required to complete the message relay on L2.
                      /// @param data The calldata of the transaction.
                      event QueueTransaction(
                          address indexed sender,
                          address indexed target,
                          uint256 value,
                          uint64 queueIndex,
                          uint256 gasLimit,
                          bytes data
                      );
                      /// @notice Emitted when some L1 => L2 transactions are included in L1.
                      /// @param startIndex The start index of messages popped.
                      /// @param count The number of messages popped.
                      /// @param skippedBitmap A bitmap indicates whether a message is skipped.
                      event DequeueTransaction(uint256 startIndex, uint256 count, uint256 skippedBitmap);
                      /// @notice Emitted when a message is dropped from L1.
                      /// @param index The index of message dropped.
                      event DropTransaction(uint256 index);
                      /// @notice Emitted when owner updates gas oracle contract.
                      /// @param _oldGasOracle The address of old gas oracle contract.
                      /// @param _newGasOracle The address of new gas oracle contract.
                      event UpdateGasOracle(address indexed _oldGasOracle, address indexed _newGasOracle);
                      /// @notice Emitted when owner updates max gas limit.
                      /// @param _oldMaxGasLimit The old max gas limit.
                      /// @param _newMaxGasLimit The new max gas limit.
                      event UpdateMaxGasLimit(uint256 _oldMaxGasLimit, uint256 _newMaxGasLimit);
                      /**********
                       * Errors *
                       **********/
                      /// @dev Thrown when the given address is `address(0)`.
                      error ErrorZeroAddress();
                      /*************************
                       * Public View Functions *
                       *************************/
                      /// @notice The start index of all pending inclusion messages.
                      function pendingQueueIndex() external view returns (uint256);
                      /// @notice Return the index of next appended message.
                      /// @dev Also the total number of appended messages.
                      function nextCrossDomainMessageIndex() external view returns (uint256);
                      /// @notice Return the message of in `queueIndex`.
                      /// @param queueIndex The index to query.
                      function getCrossDomainMessage(uint256 queueIndex) external view returns (bytes32);
                      /// @notice Return the amount of ETH should pay for cross domain message.
                      /// @param gasLimit Gas limit required to complete the message relay on L2.
                      function estimateCrossDomainMessageFee(uint256 gasLimit) external view returns (uint256);
                      /// @notice Return the amount of intrinsic gas fee should pay for cross domain message.
                      /// @param _calldata The calldata of L1-initiated transaction.
                      function calculateIntrinsicGasFee(bytes calldata _calldata) external view returns (uint256);
                      /// @notice Return the hash of a L1 message.
                      /// @param sender The address of sender.
                      /// @param queueIndex The queue index of this message.
                      /// @param value The amount of Ether transfer to target.
                      /// @param target The address of target.
                      /// @param gasLimit The gas limit provided.
                      /// @param data The calldata passed to target address.
                      function computeTransactionHash(
                          address sender,
                          uint256 queueIndex,
                          uint256 value,
                          address target,
                          uint256 gasLimit,
                          bytes calldata data
                      ) external view returns (bytes32);
                      /// @notice Return whether the message is skipped.
                      /// @param queueIndex The queue index of the message to check.
                      function isMessageSkipped(uint256 queueIndex) external view returns (bool);
                      /// @notice Return whether the message is dropped.
                      /// @param queueIndex The queue index of the message to check.
                      function isMessageDropped(uint256 queueIndex) external view returns (bool);
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @notice Append a L1 to L2 message into this contract.
                      /// @param target The address of target contract to call in L2.
                      /// @param gasLimit The maximum gas should be used for relay this message in L2.
                      /// @param data The calldata passed to target contract.
                      function appendCrossDomainMessage(
                          address target,
                          uint256 gasLimit,
                          bytes calldata data
                      ) external;
                      /// @notice Append an enforced transaction to this contract.
                      /// @dev The address of sender should be an EOA.
                      /// @param sender The address of sender who will initiate this transaction in L2.
                      /// @param target The address of target contract to call in L2.
                      /// @param value The value passed
                      /// @param gasLimit The maximum gas should be used for this transaction in L2.
                      /// @param data The calldata passed to target contract.
                      function appendEnforcedTransaction(
                          address sender,
                          address target,
                          uint256 value,
                          uint256 gasLimit,
                          bytes calldata data
                      ) external;
                      /// @notice Pop finalized messages from queue.
                      ///
                      /// @dev We can pop at most 256 messages each time. And if the message is not skipped,
                      ///      the corresponding entry will be cleared.
                      ///
                      /// @param startIndex The start index to pop.
                      /// @param count The number of messages to pop.
                      /// @param skippedBitmap A bitmap indicates whether a message is skipped.
                      function popCrossDomainMessage(
                          uint256 startIndex,
                          uint256 count,
                          uint256 skippedBitmap
                      ) external;
                      /// @notice Drop a skipped message from the queue.
                      function dropCrossDomainMessage(uint256 index) external;
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.16;
                  interface IScrollChain {
                      /**********
                       * Events *
                       **********/
                      /// @notice Emitted when a new batch is committed.
                      /// @param batchIndex The index of the batch.
                      /// @param batchHash The hash of the batch.
                      event CommitBatch(uint256 indexed batchIndex, bytes32 indexed batchHash);
                      /// @notice revert a pending batch.
                      /// @param batchIndex The index of the batch.
                      /// @param batchHash The hash of the batch
                      event RevertBatch(uint256 indexed batchIndex, bytes32 indexed batchHash);
                      /// @notice Emitted when a batch is finalized.
                      /// @param batchIndex The index of the batch.
                      /// @param batchHash The hash of the batch
                      /// @param stateRoot The state root on layer 2 after this batch.
                      /// @param withdrawRoot The merkle root on layer2 after this batch.
                      event FinalizeBatch(uint256 indexed batchIndex, bytes32 indexed batchHash, bytes32 stateRoot, bytes32 withdrawRoot);
                      /**********
                       * Errors *
                       **********/
                      /// @dev Thrown when the given address is `address(0)`.
                      error ErrorZeroAddress();
                      /*************************
                       * Public View Functions *
                       *************************/
                      /// @notice The latest finalized batch index.
                      function lastFinalizedBatchIndex() external view returns (uint256);
                      /// @notice Return the batch hash of a committed batch.
                      /// @param batchIndex The index of the batch.
                      function committedBatches(uint256 batchIndex) external view returns (bytes32);
                      /// @notice Return the state root of a committed batch.
                      /// @param batchIndex The index of the batch.
                      function finalizedStateRoots(uint256 batchIndex) external view returns (bytes32);
                      /// @notice Return the message root of a committed batch.
                      /// @param batchIndex The index of the batch.
                      function withdrawRoots(uint256 batchIndex) external view returns (bytes32);
                      /// @notice Return whether the batch is finalized by batch index.
                      /// @param batchIndex The index of the batch.
                      function isBatchFinalized(uint256 batchIndex) external view returns (bool);
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @notice Commit a batch of transactions on layer 1.
                      ///
                      /// @param version The version of current batch.
                      /// @param parentBatchHeader The header of parent batch, see the comments of `BatchHeaderV0Codec`.
                      /// @param chunks The list of encoded chunks, see the comments of `ChunkCodec`.
                      /// @param skippedL1MessageBitmap The bitmap indicates whether each L1 message is skipped or not.
                      function commitBatch(
                          uint8 version,
                          bytes calldata parentBatchHeader,
                          bytes[] memory chunks,
                          bytes calldata skippedL1MessageBitmap
                      ) external;
                      /// @notice Revert a pending batch.
                      /// @dev one can only revert unfinalized batches.
                      /// @param batchHeader The header of current batch, see the encoding in comments of `commitBatch`.
                      /// @param count The number of subsequent batches to revert, including current batch.
                      function revertBatch(bytes calldata batchHeader, uint256 count) external;
                      /// @notice Finalize a committed batch on layer 1.
                      /// @param batchHeader The header of current batch, see the encoding in comments of `commitBatch.
                      /// @param prevStateRoot The state root of parent batch.
                      /// @param postStateRoot The state root of current batch.
                      /// @param withdrawRoot The withdraw trie root of current batch.
                      /// @param aggrProof The aggregation proof for current batch.
                      function finalizeBatchWithProof(
                          bytes calldata batchHeader,
                          bytes32 prevStateRoot,
                          bytes32 postStateRoot,
                          bytes32 withdrawRoot,
                          bytes calldata aggrProof
                      ) external;
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.16;
                  interface IMessageDropCallback {
                      function onDropMessage(bytes memory message) external payable;
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.16;
                  library ScrollConstants {
                      /// @notice The address of default cross chain message sender.
                      address internal constant DEFAULT_XDOMAIN_MESSAGE_SENDER = address(1);
                      /// @notice The address for dropping message.
                      /// @dev The first 20 bytes of keccak("drop")
                      address internal constant DROP_XDOMAIN_MESSAGE_SENDER = 0x6f297C61B5C92eF107fFD30CD56AFFE5A273e841;
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.16;
                  interface IScrollMessenger {
                      /**********
                       * Events *
                       **********/
                      /// @notice Emitted when a cross domain message is sent.
                      /// @param sender The address of the sender who initiates the message.
                      /// @param target The address of target contract to call.
                      /// @param value The amount of value passed to the target contract.
                      /// @param messageNonce The nonce of the message.
                      /// @param gasLimit The optional gas limit passed to L1 or L2.
                      /// @param message The calldata passed to the target contract.
                      event SentMessage(
                          address indexed sender,
                          address indexed target,
                          uint256 value,
                          uint256 messageNonce,
                          uint256 gasLimit,
                          bytes message
                      );
                      /// @notice Emitted when a cross domain message is relayed successfully.
                      /// @param messageHash The hash of the message.
                      event RelayedMessage(bytes32 indexed messageHash);
                      /// @notice Emitted when a cross domain message is failed to relay.
                      /// @param messageHash The hash of the message.
                      event FailedRelayedMessage(bytes32 indexed messageHash);
                      /**********
                       * Errors *
                       **********/
                      /// @dev Thrown when the given address is `address(0)`.
                      error ErrorZeroAddress();
                      /*************************
                       * Public View Functions *
                       *************************/
                      /// @notice Return the sender of a cross domain message.
                      function xDomainMessageSender() external view returns (address);
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @notice Send cross chain message from L1 to L2 or L2 to L1.
                      /// @param target The address of account who receive the message.
                      /// @param value The amount of ether passed when call target contract.
                      /// @param message The content of the message.
                      /// @param gasLimit Gas limit required to complete the message relay on corresponding chain.
                      function sendMessage(
                          address target,
                          uint256 value,
                          bytes calldata message,
                          uint256 gasLimit
                      ) external payable;
                      /// @notice Send cross chain message from L1 to L2 or L2 to L1.
                      /// @param target The address of account who receive the message.
                      /// @param value The amount of ether passed when call target contract.
                      /// @param message The content of the message.
                      /// @param gasLimit Gas limit required to complete the message relay on corresponding chain.
                      /// @param refundAddress The address of account who will receive the refunded fee.
                      function sendMessage(
                          address target,
                          uint256 value,
                          bytes calldata message,
                          uint256 gasLimit,
                          address refundAddress
                      ) external payable;
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.16;
                  import {OwnableUpgradeable} from "@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol";
                  import {PausableUpgradeable} from "@openzeppelin/contracts-upgradeable/security/PausableUpgradeable.sol";
                  import {ReentrancyGuardUpgradeable} from "@openzeppelin/contracts-upgradeable/security/ReentrancyGuardUpgradeable.sol";
                  import {ScrollConstants} from "./constants/ScrollConstants.sol";
                  import {IScrollMessenger} from "./IScrollMessenger.sol";
                  // solhint-disable var-name-mixedcase
                  abstract contract ScrollMessengerBase is
                      OwnableUpgradeable,
                      PausableUpgradeable,
                      ReentrancyGuardUpgradeable,
                      IScrollMessenger
                  {
                      /**********
                       * Events *
                       **********/
                      /// @notice Emitted when owner updates fee vault contract.
                      /// @param _oldFeeVault The address of old fee vault contract.
                      /// @param _newFeeVault The address of new fee vault contract.
                      event UpdateFeeVault(address _oldFeeVault, address _newFeeVault);
                      /*************
                       * Constants *
                       *************/
                      /// @notice The address of counterpart ScrollMessenger contract in L1/L2.
                      address public immutable counterpart;
                      /*************
                       * Variables *
                       *************/
                      /// @notice See {IScrollMessenger-xDomainMessageSender}
                      address public override xDomainMessageSender;
                      /// @dev The storage slot used as counterpart ScrollMessenger contract, which is deprecated now.
                      address private __counterpart;
                      /// @notice The address of fee vault, collecting cross domain messaging fee.
                      address public feeVault;
                      /// @dev The storage slot used as ETH rate limiter contract, which is deprecated now.
                      address private __rateLimiter;
                      /// @dev The storage slots for future usage.
                      uint256[46] private __gap;
                      /**********************
                       * Function Modifiers *
                       **********************/
                      modifier notInExecution() {
                          require(
                              xDomainMessageSender == ScrollConstants.DEFAULT_XDOMAIN_MESSAGE_SENDER,
                              "Message is already in execution"
                          );
                          _;
                      }
                      /***************
                       * Constructor *
                       ***************/
                      constructor(address _counterpart) {
                          if (_counterpart == address(0)) {
                              revert ErrorZeroAddress();
                          }
                          counterpart = _counterpart;
                      }
                      function __ScrollMessengerBase_init(address, address _feeVault) internal onlyInitializing {
                          OwnableUpgradeable.__Ownable_init();
                          PausableUpgradeable.__Pausable_init();
                          ReentrancyGuardUpgradeable.__ReentrancyGuard_init();
                          // initialize to a nonzero value
                          xDomainMessageSender = ScrollConstants.DEFAULT_XDOMAIN_MESSAGE_SENDER;
                          if (_feeVault != address(0)) {
                              feeVault = _feeVault;
                          }
                      }
                      // make sure only owner can send ether to messenger to avoid possible user fund loss.
                      receive() external payable onlyOwner {}
                      /************************
                       * Restricted Functions *
                       ************************/
                      /// @notice Update fee vault contract.
                      /// @dev This function can only called by contract owner.
                      /// @param _newFeeVault The address of new fee vault contract.
                      function updateFeeVault(address _newFeeVault) external onlyOwner {
                          address _oldFeeVault = feeVault;
                          feeVault = _newFeeVault;
                          emit UpdateFeeVault(_oldFeeVault, _newFeeVault);
                      }
                      /// @notice Pause the contract
                      /// @dev This function can only called by contract owner.
                      /// @param _status The pause status to update.
                      function setPause(bool _status) external onlyOwner {
                          if (_status) {
                              _pause();
                          } else {
                              _unpause();
                          }
                      }
                      /**********************
                       * Internal Functions *
                       **********************/
                      /// @dev Internal function to generate the correct cross domain calldata for a message.
                      /// @param _sender Message sender address.
                      /// @param _target Target contract address.
                      /// @param _value The amount of ETH pass to the target.
                      /// @param _messageNonce Nonce for the provided message.
                      /// @param _message Message to send to the target.
                      /// @return ABI encoded cross domain calldata.
                      function _encodeXDomainCalldata(
                          address _sender,
                          address _target,
                          uint256 _value,
                          uint256 _messageNonce,
                          bytes memory _message
                      ) internal pure returns (bytes memory) {
                          return
                              abi.encodeWithSignature(
                                  "relayMessage(address,address,uint256,uint256,bytes)",
                                  _sender,
                                  _target,
                                  _value,
                                  _messageNonce,
                                  _message
                              );
                      }
                      /// @dev Internal function to check whether the `_target` address is allowed to avoid attack.
                      /// @param _target The address of target address to check.
                      function _validateTargetAddress(address _target) internal view {
                          // @note check more `_target` address to avoid attack in the future when we add more external contracts.
                          require(_target != address(this), "Forbid to call self");
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.16;
                  // solhint-disable no-inline-assembly
                  library WithdrawTrieVerifier {
                      /// @dev Verify the merkle proof given root, leaf node and proof.
                      ///
                      /// Vulnerability:
                      ///   The initially provided message hash can be hashed with the first hash of the proof,
                      ///   thereby giving an intermediate node of the trie. This can then be used with a shortened
                      ///   proof to pass the verification, which may lead to replayability.
                      ///
                      ///   However, it is designed to verify the withdraw trie in `L2MessageQueue`. The `_hash` given
                      ///   in the parameter is always a leaf node. So we assume the length of proof is correct and
                      ///   cannot be shortened.
                      /// @param _root The expected root node hash of the withdraw trie.
                      /// @param _hash The leaf node hash of the withdraw trie.
                      /// @param _nonce The index of the leaf node from left to right, starting from 0.
                      /// @param _proof The concatenated merkle proof verified the leaf node.
                      function verifyMerkleProof(
                          bytes32 _root,
                          bytes32 _hash,
                          uint256 _nonce,
                          bytes memory _proof
                      ) internal pure returns (bool) {
                          require(_proof.length % 32 == 0, "Invalid proof");
                          uint256 _length = _proof.length / 32;
                          for (uint256 i = 0; i < _length; i++) {
                              bytes32 item;
                              assembly {
                                  item := mload(add(add(_proof, 0x20), mul(i, 0x20)))
                              }
                              if (_nonce % 2 == 0) {
                                  _hash = _efficientHash(_hash, item);
                              } else {
                                  _hash = _efficientHash(item, _hash);
                              }
                              _nonce /= 2;
                          }
                          return _hash == _root;
                      }
                      function _efficientHash(bytes32 a, bytes32 b) private pure returns (bytes32 value) {
                          // solhint-disable-next-line no-inline-assembly
                          assembly {
                              mstore(0x00, a)
                              mstore(0x20, b)
                              value := keccak256(0x00, 0x40)
                          }
                      }
                  }