ETH Price: $2,530.02 (+1.39%)

Transaction Decoder

Block:
21147078 at Nov-09-2024 02:22:23 AM +UTC
Transaction Fee:
0.001611520268693 ETH $4.08
Gas Used:
204,250 Gas / 7.889940116 Gwei

Emitted Events:

287 SafeProxy.0x3d0ce9bfc3ed7d6862dbb28b2dea94561fe714a1b4d019aa8af39730d1ad7c3d( 0x3d0ce9bfc3ed7d6862dbb28b2dea94561fe714a1b4d019aa8af39730d1ad7c3d, 0x000000000000000000000000dc71366effa760804dcfc3edf87fa2a6f1623304, 000000000000000000000000000000000000000000000000000000544206e140 )
288 TransparentUpgradeableProxy.0x69cfcb8e6d4192b8aba9902243912587f37e550d75c1fa801491fce26717f37e( 0x69cfcb8e6d4192b8aba9902243912587f37e550d75c1fa801491fce26717f37e, 0x000000000000000000000000ed82366effa760804dcfc3edf87fa2a6f1624415, 0x0000000000000000000000005300000000000000000000000000000000000007, 0000000000000000000000000000000000000000000000000000000000000000, 00000000000000000000000000000000000000000000000000000000000006ad, 000000000000000000000000000000000000000000000000000000000005859d, 0000000000000000000000000000000000000000000000000000000000000080, 0000000000000000000000000000000000000000000000000000000000000184, 8ef1332e0000000000000000000000001c1ffb5828c3a48b54e8910f1c75256a, 498ade6800000000000000000000000053000000000000000000000000000000, 00000006000000000000000000000000000000000000000000000000002386f2, 6fc1000000000000000000000000000000000000000000000000000000000000, 000006ad00000000000000000000000000000000000000000000000000000000, 000000a000000000000000000000000000000000000000000000000000000000, 000000a4232e87480000000000000000000000003bd1848aa08f2fb725172d9f, 6aa738adf22bbd2b0000000000000000000000003bd1848aa08f2fb725172d9f, 6aa738adf22bbd2b000000000000000000000000000000000000000000000000, 002386f26fc10000000000000000000000000000000000000000000000000000, 0000000000000080000000000000000000000000000000000000000000000000, 0000000000000000000000000000000000000000000000000000000000000000, 0000000000000000000000000000000000000000000000000000000000000000 )
289 TransparentUpgradeableProxy.0x104371f3b442861a2a7b82a070afbbaab748bb13757bf47769e170e37809ec1e( 0x104371f3b442861a2a7b82a070afbbaab748bb13757bf47769e170e37809ec1e, 0x0000000000000000000000001c1ffb5828c3a48b54e8910f1c75256a498ade68, 0x0000000000000000000000005300000000000000000000000000000000000006, 000000000000000000000000000000000000000000000000002386f26fc10000, 00000000000000000000000000000000000000000000000000000000000006ad, 000000000000000000000000000000000000000000000000000000000005859d, 0000000000000000000000000000000000000000000000000000000000000080, 00000000000000000000000000000000000000000000000000000000000000a4, 232e87480000000000000000000000003bd1848aa08f2fb725172d9f6aa738ad, f22bbd2b0000000000000000000000003bd1848aa08f2fb725172d9f6aa738ad, f22bbd2b000000000000000000000000000000000000000000000000002386f2, 6fc1000000000000000000000000000000000000000000000000000000000000, 0000008000000000000000000000000000000000000000000000000000000000, 0000000000000000000000000000000000000000000000000000000000000000 )
290 TransparentUpgradeableProxy.0xa900620ce06f0a525c07f9b89600c2297c6da3322a0cd2f034fbded0c1148eda( 0xa900620ce06f0a525c07f9b89600c2297c6da3322a0cd2f034fbded0c1148eda, 0x0000000000000000000000003bd1848aa08f2fb725172d9f6aa738adf22bbd2b, 0x0000000000000000000000003bd1848aa08f2fb725172d9f6aa738adf22bbd2b, 000000000000000000000000000000000000000000000000002386f26fc10000, 0000000000000000000000000000000000000000000000000000000000000060, 00000000000000000000000000000000000000000000000000000000000006ad, 0000000000000000000000000000000000000000000000000000000000000000 )

Account State Difference:

  Address   Before After State Difference Code
0x3931Ade8...DcE6cC1EF
0x3bd1848a...DF22Bbd2B
0.020413834697687856 Eth
Nonce: 114
0.008801952543994856 Eth
Nonce: 115
0.011611882153693
(Titan Builder)
12.699760335009447257 Eth12.699958457509447257 Eth0.0001981225
0xB822319a...5BFb04377 0.100467363607 Eth0.100467725492 Eth0.000000361885
0xDc71366E...6f1623304 920.532767459649702182 Eth920.542767459649702182 Eth0.01

Execution Trace

ETH 0.010000361885 TransparentUpgradeableProxy.ce0b63ce( )
  • ETH 0.010000361885 L1GatewayRouter.depositETH( _to=0x3bd1848aA08F2FB725172d9f6aA738aDF22Bbd2B, _amount=10000000000000000, _gasLimit=361885 )
    • ETH 0.010000361885 TransparentUpgradeableProxy.aac476f8( )
      • ETH 0.010000361885 L1ETHGateway.depositETHAndCall( _to=0x3bd1848aA08F2FB725172d9f6aA738aDF22Bbd2B, _amount=10000000000000000, _data=0x0000000000000000000000003BD1848AA08F2FB725172D9F6AA738ADF22BBD2B00000000000000000000000000000000000000000000000000000000000000400000000000000000000000000000000000000000000000000000000000000000, _gasLimit=361885 )
        • TransparentUpgradeableProxy.STATICCALL( )
          • L1CrossDomainMessenger.DELEGATECALL( )
            • TransparentUpgradeableProxy.STATICCALL( )
            • ETH 0.010000361885 TransparentUpgradeableProxy.5f7b1577( )
              • ETH 0.010000361885 L1CrossDomainMessenger.sendMessage( _to=0x5300000000000000000000000000000000000006, _value=10000000000000000, _message=0x232E87480000000000000000000000003BD1848AA08F2FB725172D9F6AA738ADF22BBD2B0000000000000000000000003BD1848AA08F2FB725172D9F6AA738ADF22BBD2B000000000000000000000000000000000000000000000000002386F26FC1000000000000000000000000000000000000000000000000000000000000000000800000000000000000000000000000000000000000000000000000000000000000, _gasLimit=361885, _refundAddress=0x3bd1848aA08F2FB725172d9f6aA738aDF22Bbd2B )
                • TransparentUpgradeableProxy.STATICCALL( )
                • TransparentUpgradeableProxy.3e4cbbe6( )
                • ETH 0.000000361885 SafeProxy.CALL( )
                • TransparentUpgradeableProxy.9b159782( )
                  File 1 of 8: TransparentUpgradeableProxy
                  // 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) (interfaces/IERC1967.sol)
                  pragma solidity ^0.8.0;
                  /**
                   * @dev ERC-1967: Proxy Storage Slots. This interface contains the events defined in the ERC.
                   *
                   * _Available since v4.8.3._
                   */
                  interface IERC1967 {
                      /**
                       * @dev Emitted when the implementation is upgraded.
                       */
                      event Upgraded(address indexed implementation);
                      /**
                       * @dev Emitted when the admin account has changed.
                       */
                      event AdminChanged(address previousAdmin, address newAdmin);
                      /**
                       * @dev Emitted when the beacon is changed.
                       */
                      event BeaconUpgraded(address indexed beacon);
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts v4.4.1 (proxy/beacon/IBeacon.sol)
                  pragma solidity ^0.8.0;
                  /**
                   * @dev This is the interface that {BeaconProxy} expects of its beacon.
                   */
                  interface IBeacon {
                      /**
                       * @dev Must return an address that can be used as a delegate call target.
                       *
                       * {BeaconProxy} will check that this address is a contract.
                       */
                      function implementation() external view returns (address);
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.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.9.0) (proxy/ERC1967/ERC1967Upgrade.sol)
                  pragma solidity ^0.8.2;
                  import "../beacon/IBeacon.sol";
                  import "../../interfaces/IERC1967.sol";
                  import "../../interfaces/draft-IERC1822.sol";
                  import "../../utils/Address.sol";
                  import "../../utils/StorageSlot.sol";
                  /**
                   * @dev This abstract contract provides getters and event emitting update functions for
                   * https://eips.ethereum.org/EIPS/eip-1967[EIP1967] slots.
                   *
                   * _Available since v4.1._
                   */
                  abstract contract ERC1967Upgrade is IERC1967 {
                      // This is the keccak-256 hash of "eip1967.proxy.rollback" subtracted by 1
                      bytes32 private constant _ROLLBACK_SLOT = 0x4910fdfa16fed3260ed0e7147f7cc6da11a60208b5b9406d12a635614ffd9143;
                      /**
                       * @dev Storage slot with the address of the current implementation.
                       * This is the keccak-256 hash of "eip1967.proxy.implementation" subtracted by 1, and is
                       * validated in the constructor.
                       */
                      bytes32 internal constant _IMPLEMENTATION_SLOT = 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc;
                      /**
                       * @dev Returns the current implementation address.
                       */
                      function _getImplementation() internal view returns (address) {
                          return StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value;
                      }
                      /**
                       * @dev Stores a new address in the EIP1967 implementation slot.
                       */
                      function _setImplementation(address newImplementation) private {
                          require(Address.isContract(newImplementation), "ERC1967: new implementation is not a contract");
                          StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value = newImplementation;
                      }
                      /**
                       * @dev Perform implementation upgrade
                       *
                       * Emits an {Upgraded} event.
                       */
                      function _upgradeTo(address newImplementation) internal {
                          _setImplementation(newImplementation);
                          emit Upgraded(newImplementation);
                      }
                      /**
                       * @dev Perform implementation upgrade with additional setup call.
                       *
                       * Emits an {Upgraded} event.
                       */
                      function _upgradeToAndCall(address newImplementation, bytes memory data, bool forceCall) internal {
                          _upgradeTo(newImplementation);
                          if (data.length > 0 || forceCall) {
                              Address.functionDelegateCall(newImplementation, data);
                          }
                      }
                      /**
                       * @dev Perform implementation upgrade with security checks for UUPS proxies, and additional setup call.
                       *
                       * Emits an {Upgraded} event.
                       */
                      function _upgradeToAndCallUUPS(address newImplementation, bytes memory data, bool forceCall) internal {
                          // Upgrades from old implementations will perform a rollback test. This test requires the new
                          // implementation to upgrade back to the old, non-ERC1822 compliant, implementation. Removing
                          // this special case will break upgrade paths from old UUPS implementation to new ones.
                          if (StorageSlot.getBooleanSlot(_ROLLBACK_SLOT).value) {
                              _setImplementation(newImplementation);
                          } else {
                              try IERC1822Proxiable(newImplementation).proxiableUUID() returns (bytes32 slot) {
                                  require(slot == _IMPLEMENTATION_SLOT, "ERC1967Upgrade: unsupported proxiableUUID");
                              } catch {
                                  revert("ERC1967Upgrade: new implementation is not UUPS");
                              }
                              _upgradeToAndCall(newImplementation, data, forceCall);
                          }
                      }
                      /**
                       * @dev Storage slot with the admin of the contract.
                       * This is the keccak-256 hash of "eip1967.proxy.admin" subtracted by 1, and is
                       * validated in the constructor.
                       */
                      bytes32 internal constant _ADMIN_SLOT = 0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103;
                      /**
                       * @dev Returns the current admin.
                       */
                      function _getAdmin() internal view returns (address) {
                          return StorageSlot.getAddressSlot(_ADMIN_SLOT).value;
                      }
                      /**
                       * @dev Stores a new address in the EIP1967 admin slot.
                       */
                      function _setAdmin(address newAdmin) private {
                          require(newAdmin != address(0), "ERC1967: new admin is the zero address");
                          StorageSlot.getAddressSlot(_ADMIN_SLOT).value = newAdmin;
                      }
                      /**
                       * @dev Changes the admin of the proxy.
                       *
                       * Emits an {AdminChanged} event.
                       */
                      function _changeAdmin(address newAdmin) internal {
                          emit AdminChanged(_getAdmin(), newAdmin);
                          _setAdmin(newAdmin);
                      }
                      /**
                       * @dev The storage slot of the UpgradeableBeacon contract which defines the implementation for this proxy.
                       * This is bytes32(uint256(keccak256('eip1967.proxy.beacon')) - 1)) and is validated in the constructor.
                       */
                      bytes32 internal constant _BEACON_SLOT = 0xa3f0ad74e5423aebfd80d3ef4346578335a9a72aeaee59ff6cb3582b35133d50;
                      /**
                       * @dev Returns the current beacon.
                       */
                      function _getBeacon() internal view returns (address) {
                          return StorageSlot.getAddressSlot(_BEACON_SLOT).value;
                      }
                      /**
                       * @dev Stores a new beacon in the EIP1967 beacon slot.
                       */
                      function _setBeacon(address newBeacon) private {
                          require(Address.isContract(newBeacon), "ERC1967: new beacon is not a contract");
                          require(
                              Address.isContract(IBeacon(newBeacon).implementation()),
                              "ERC1967: beacon implementation is not a contract"
                          );
                          StorageSlot.getAddressSlot(_BEACON_SLOT).value = newBeacon;
                      }
                      /**
                       * @dev Perform beacon upgrade with additional setup call. Note: This upgrades the address of the beacon, it does
                       * not upgrade the implementation contained in the beacon (see {UpgradeableBeacon-_setImplementation} for that).
                       *
                       * Emits a {BeaconUpgraded} event.
                       */
                      function _upgradeBeaconToAndCall(address newBeacon, bytes memory data, bool forceCall) internal {
                          _setBeacon(newBeacon);
                          emit BeaconUpgraded(newBeacon);
                          if (data.length > 0 || forceCall) {
                              Address.functionDelegateCall(IBeacon(newBeacon).implementation(), data);
                          }
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.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/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.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 8: SafeProxy
                  // SPDX-License-Identifier: LGPL-3.0-only
                  pragma solidity >=0.7.0 <0.9.0;
                  /**
                   * @title IProxy - Helper interface to access the singleton address of the Proxy on-chain.
                   * @author Richard Meissner - @rmeissner
                   */
                  interface IProxy {
                      function masterCopy() external view returns (address);
                  }
                  /**
                   * @title SafeProxy - 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 SafeProxy {
                      // 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;
                      /**
                       * @notice 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())
                          }
                      }
                  }
                  

                  File 3 of 8: TransparentUpgradeableProxy
                  // 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) (interfaces/IERC1967.sol)
                  pragma solidity ^0.8.0;
                  /**
                   * @dev ERC-1967: Proxy Storage Slots. This interface contains the events defined in the ERC.
                   *
                   * _Available since v4.8.3._
                   */
                  interface IERC1967 {
                      /**
                       * @dev Emitted when the implementation is upgraded.
                       */
                      event Upgraded(address indexed implementation);
                      /**
                       * @dev Emitted when the admin account has changed.
                       */
                      event AdminChanged(address previousAdmin, address newAdmin);
                      /**
                       * @dev Emitted when the beacon is changed.
                       */
                      event BeaconUpgraded(address indexed beacon);
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts v4.4.1 (proxy/beacon/IBeacon.sol)
                  pragma solidity ^0.8.0;
                  /**
                   * @dev This is the interface that {BeaconProxy} expects of its beacon.
                   */
                  interface IBeacon {
                      /**
                       * @dev Must return an address that can be used as a delegate call target.
                       *
                       * {BeaconProxy} will check that this address is a contract.
                       */
                      function implementation() external view returns (address);
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.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.9.0) (proxy/ERC1967/ERC1967Upgrade.sol)
                  pragma solidity ^0.8.2;
                  import "../beacon/IBeacon.sol";
                  import "../../interfaces/IERC1967.sol";
                  import "../../interfaces/draft-IERC1822.sol";
                  import "../../utils/Address.sol";
                  import "../../utils/StorageSlot.sol";
                  /**
                   * @dev This abstract contract provides getters and event emitting update functions for
                   * https://eips.ethereum.org/EIPS/eip-1967[EIP1967] slots.
                   *
                   * _Available since v4.1._
                   */
                  abstract contract ERC1967Upgrade is IERC1967 {
                      // This is the keccak-256 hash of "eip1967.proxy.rollback" subtracted by 1
                      bytes32 private constant _ROLLBACK_SLOT = 0x4910fdfa16fed3260ed0e7147f7cc6da11a60208b5b9406d12a635614ffd9143;
                      /**
                       * @dev Storage slot with the address of the current implementation.
                       * This is the keccak-256 hash of "eip1967.proxy.implementation" subtracted by 1, and is
                       * validated in the constructor.
                       */
                      bytes32 internal constant _IMPLEMENTATION_SLOT = 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc;
                      /**
                       * @dev Returns the current implementation address.
                       */
                      function _getImplementation() internal view returns (address) {
                          return StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value;
                      }
                      /**
                       * @dev Stores a new address in the EIP1967 implementation slot.
                       */
                      function _setImplementation(address newImplementation) private {
                          require(Address.isContract(newImplementation), "ERC1967: new implementation is not a contract");
                          StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value = newImplementation;
                      }
                      /**
                       * @dev Perform implementation upgrade
                       *
                       * Emits an {Upgraded} event.
                       */
                      function _upgradeTo(address newImplementation) internal {
                          _setImplementation(newImplementation);
                          emit Upgraded(newImplementation);
                      }
                      /**
                       * @dev Perform implementation upgrade with additional setup call.
                       *
                       * Emits an {Upgraded} event.
                       */
                      function _upgradeToAndCall(address newImplementation, bytes memory data, bool forceCall) internal {
                          _upgradeTo(newImplementation);
                          if (data.length > 0 || forceCall) {
                              Address.functionDelegateCall(newImplementation, data);
                          }
                      }
                      /**
                       * @dev Perform implementation upgrade with security checks for UUPS proxies, and additional setup call.
                       *
                       * Emits an {Upgraded} event.
                       */
                      function _upgradeToAndCallUUPS(address newImplementation, bytes memory data, bool forceCall) internal {
                          // Upgrades from old implementations will perform a rollback test. This test requires the new
                          // implementation to upgrade back to the old, non-ERC1822 compliant, implementation. Removing
                          // this special case will break upgrade paths from old UUPS implementation to new ones.
                          if (StorageSlot.getBooleanSlot(_ROLLBACK_SLOT).value) {
                              _setImplementation(newImplementation);
                          } else {
                              try IERC1822Proxiable(newImplementation).proxiableUUID() returns (bytes32 slot) {
                                  require(slot == _IMPLEMENTATION_SLOT, "ERC1967Upgrade: unsupported proxiableUUID");
                              } catch {
                                  revert("ERC1967Upgrade: new implementation is not UUPS");
                              }
                              _upgradeToAndCall(newImplementation, data, forceCall);
                          }
                      }
                      /**
                       * @dev Storage slot with the admin of the contract.
                       * This is the keccak-256 hash of "eip1967.proxy.admin" subtracted by 1, and is
                       * validated in the constructor.
                       */
                      bytes32 internal constant _ADMIN_SLOT = 0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103;
                      /**
                       * @dev Returns the current admin.
                       */
                      function _getAdmin() internal view returns (address) {
                          return StorageSlot.getAddressSlot(_ADMIN_SLOT).value;
                      }
                      /**
                       * @dev Stores a new address in the EIP1967 admin slot.
                       */
                      function _setAdmin(address newAdmin) private {
                          require(newAdmin != address(0), "ERC1967: new admin is the zero address");
                          StorageSlot.getAddressSlot(_ADMIN_SLOT).value = newAdmin;
                      }
                      /**
                       * @dev Changes the admin of the proxy.
                       *
                       * Emits an {AdminChanged} event.
                       */
                      function _changeAdmin(address newAdmin) internal {
                          emit AdminChanged(_getAdmin(), newAdmin);
                          _setAdmin(newAdmin);
                      }
                      /**
                       * @dev The storage slot of the UpgradeableBeacon contract which defines the implementation for this proxy.
                       * This is bytes32(uint256(keccak256('eip1967.proxy.beacon')) - 1)) and is validated in the constructor.
                       */
                      bytes32 internal constant _BEACON_SLOT = 0xa3f0ad74e5423aebfd80d3ef4346578335a9a72aeaee59ff6cb3582b35133d50;
                      /**
                       * @dev Returns the current beacon.
                       */
                      function _getBeacon() internal view returns (address) {
                          return StorageSlot.getAddressSlot(_BEACON_SLOT).value;
                      }
                      /**
                       * @dev Stores a new beacon in the EIP1967 beacon slot.
                       */
                      function _setBeacon(address newBeacon) private {
                          require(Address.isContract(newBeacon), "ERC1967: new beacon is not a contract");
                          require(
                              Address.isContract(IBeacon(newBeacon).implementation()),
                              "ERC1967: beacon implementation is not a contract"
                          );
                          StorageSlot.getAddressSlot(_BEACON_SLOT).value = newBeacon;
                      }
                      /**
                       * @dev Perform beacon upgrade with additional setup call. Note: This upgrades the address of the beacon, it does
                       * not upgrade the implementation contained in the beacon (see {UpgradeableBeacon-_setImplementation} for that).
                       *
                       * Emits a {BeaconUpgraded} event.
                       */
                      function _upgradeBeaconToAndCall(address newBeacon, bytes memory data, bool forceCall) internal {
                          _setBeacon(newBeacon);
                          emit BeaconUpgraded(newBeacon);
                          if (data.length > 0 || forceCall) {
                              Address.functionDelegateCall(IBeacon(newBeacon).implementation(), data);
                          }
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.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/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.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 4 of 8: TransparentUpgradeableProxy
                  // 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) (interfaces/IERC1967.sol)
                  pragma solidity ^0.8.0;
                  /**
                   * @dev ERC-1967: Proxy Storage Slots. This interface contains the events defined in the ERC.
                   *
                   * _Available since v4.8.3._
                   */
                  interface IERC1967 {
                      /**
                       * @dev Emitted when the implementation is upgraded.
                       */
                      event Upgraded(address indexed implementation);
                      /**
                       * @dev Emitted when the admin account has changed.
                       */
                      event AdminChanged(address previousAdmin, address newAdmin);
                      /**
                       * @dev Emitted when the beacon is changed.
                       */
                      event BeaconUpgraded(address indexed beacon);
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts v4.4.1 (proxy/beacon/IBeacon.sol)
                  pragma solidity ^0.8.0;
                  /**
                   * @dev This is the interface that {BeaconProxy} expects of its beacon.
                   */
                  interface IBeacon {
                      /**
                       * @dev Must return an address that can be used as a delegate call target.
                       *
                       * {BeaconProxy} will check that this address is a contract.
                       */
                      function implementation() external view returns (address);
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.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.9.0) (proxy/ERC1967/ERC1967Upgrade.sol)
                  pragma solidity ^0.8.2;
                  import "../beacon/IBeacon.sol";
                  import "../../interfaces/IERC1967.sol";
                  import "../../interfaces/draft-IERC1822.sol";
                  import "../../utils/Address.sol";
                  import "../../utils/StorageSlot.sol";
                  /**
                   * @dev This abstract contract provides getters and event emitting update functions for
                   * https://eips.ethereum.org/EIPS/eip-1967[EIP1967] slots.
                   *
                   * _Available since v4.1._
                   */
                  abstract contract ERC1967Upgrade is IERC1967 {
                      // This is the keccak-256 hash of "eip1967.proxy.rollback" subtracted by 1
                      bytes32 private constant _ROLLBACK_SLOT = 0x4910fdfa16fed3260ed0e7147f7cc6da11a60208b5b9406d12a635614ffd9143;
                      /**
                       * @dev Storage slot with the address of the current implementation.
                       * This is the keccak-256 hash of "eip1967.proxy.implementation" subtracted by 1, and is
                       * validated in the constructor.
                       */
                      bytes32 internal constant _IMPLEMENTATION_SLOT = 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc;
                      /**
                       * @dev Returns the current implementation address.
                       */
                      function _getImplementation() internal view returns (address) {
                          return StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value;
                      }
                      /**
                       * @dev Stores a new address in the EIP1967 implementation slot.
                       */
                      function _setImplementation(address newImplementation) private {
                          require(Address.isContract(newImplementation), "ERC1967: new implementation is not a contract");
                          StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value = newImplementation;
                      }
                      /**
                       * @dev Perform implementation upgrade
                       *
                       * Emits an {Upgraded} event.
                       */
                      function _upgradeTo(address newImplementation) internal {
                          _setImplementation(newImplementation);
                          emit Upgraded(newImplementation);
                      }
                      /**
                       * @dev Perform implementation upgrade with additional setup call.
                       *
                       * Emits an {Upgraded} event.
                       */
                      function _upgradeToAndCall(address newImplementation, bytes memory data, bool forceCall) internal {
                          _upgradeTo(newImplementation);
                          if (data.length > 0 || forceCall) {
                              Address.functionDelegateCall(newImplementation, data);
                          }
                      }
                      /**
                       * @dev Perform implementation upgrade with security checks for UUPS proxies, and additional setup call.
                       *
                       * Emits an {Upgraded} event.
                       */
                      function _upgradeToAndCallUUPS(address newImplementation, bytes memory data, bool forceCall) internal {
                          // Upgrades from old implementations will perform a rollback test. This test requires the new
                          // implementation to upgrade back to the old, non-ERC1822 compliant, implementation. Removing
                          // this special case will break upgrade paths from old UUPS implementation to new ones.
                          if (StorageSlot.getBooleanSlot(_ROLLBACK_SLOT).value) {
                              _setImplementation(newImplementation);
                          } else {
                              try IERC1822Proxiable(newImplementation).proxiableUUID() returns (bytes32 slot) {
                                  require(slot == _IMPLEMENTATION_SLOT, "ERC1967Upgrade: unsupported proxiableUUID");
                              } catch {
                                  revert("ERC1967Upgrade: new implementation is not UUPS");
                              }
                              _upgradeToAndCall(newImplementation, data, forceCall);
                          }
                      }
                      /**
                       * @dev Storage slot with the admin of the contract.
                       * This is the keccak-256 hash of "eip1967.proxy.admin" subtracted by 1, and is
                       * validated in the constructor.
                       */
                      bytes32 internal constant _ADMIN_SLOT = 0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103;
                      /**
                       * @dev Returns the current admin.
                       */
                      function _getAdmin() internal view returns (address) {
                          return StorageSlot.getAddressSlot(_ADMIN_SLOT).value;
                      }
                      /**
                       * @dev Stores a new address in the EIP1967 admin slot.
                       */
                      function _setAdmin(address newAdmin) private {
                          require(newAdmin != address(0), "ERC1967: new admin is the zero address");
                          StorageSlot.getAddressSlot(_ADMIN_SLOT).value = newAdmin;
                      }
                      /**
                       * @dev Changes the admin of the proxy.
                       *
                       * Emits an {AdminChanged} event.
                       */
                      function _changeAdmin(address newAdmin) internal {
                          emit AdminChanged(_getAdmin(), newAdmin);
                          _setAdmin(newAdmin);
                      }
                      /**
                       * @dev The storage slot of the UpgradeableBeacon contract which defines the implementation for this proxy.
                       * This is bytes32(uint256(keccak256('eip1967.proxy.beacon')) - 1)) and is validated in the constructor.
                       */
                      bytes32 internal constant _BEACON_SLOT = 0xa3f0ad74e5423aebfd80d3ef4346578335a9a72aeaee59ff6cb3582b35133d50;
                      /**
                       * @dev Returns the current beacon.
                       */
                      function _getBeacon() internal view returns (address) {
                          return StorageSlot.getAddressSlot(_BEACON_SLOT).value;
                      }
                      /**
                       * @dev Stores a new beacon in the EIP1967 beacon slot.
                       */
                      function _setBeacon(address newBeacon) private {
                          require(Address.isContract(newBeacon), "ERC1967: new beacon is not a contract");
                          require(
                              Address.isContract(IBeacon(newBeacon).implementation()),
                              "ERC1967: beacon implementation is not a contract"
                          );
                          StorageSlot.getAddressSlot(_BEACON_SLOT).value = newBeacon;
                      }
                      /**
                       * @dev Perform beacon upgrade with additional setup call. Note: This upgrades the address of the beacon, it does
                       * not upgrade the implementation contained in the beacon (see {UpgradeableBeacon-_setImplementation} for that).
                       *
                       * Emits a {BeaconUpgraded} event.
                       */
                      function _upgradeBeaconToAndCall(address newBeacon, bytes memory data, bool forceCall) internal {
                          _setBeacon(newBeacon);
                          emit BeaconUpgraded(newBeacon);
                          if (data.length > 0 || forceCall) {
                              Address.functionDelegateCall(IBeacon(newBeacon).implementation(), data);
                          }
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.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/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.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 8: TransparentUpgradeableProxy
                  // 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) (interfaces/IERC1967.sol)
                  pragma solidity ^0.8.0;
                  /**
                   * @dev ERC-1967: Proxy Storage Slots. This interface contains the events defined in the ERC.
                   *
                   * _Available since v4.8.3._
                   */
                  interface IERC1967 {
                      /**
                       * @dev Emitted when the implementation is upgraded.
                       */
                      event Upgraded(address indexed implementation);
                      /**
                       * @dev Emitted when the admin account has changed.
                       */
                      event AdminChanged(address previousAdmin, address newAdmin);
                      /**
                       * @dev Emitted when the beacon is changed.
                       */
                      event BeaconUpgraded(address indexed beacon);
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts v4.4.1 (proxy/beacon/IBeacon.sol)
                  pragma solidity ^0.8.0;
                  /**
                   * @dev This is the interface that {BeaconProxy} expects of its beacon.
                   */
                  interface IBeacon {
                      /**
                       * @dev Must return an address that can be used as a delegate call target.
                       *
                       * {BeaconProxy} will check that this address is a contract.
                       */
                      function implementation() external view returns (address);
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.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.9.0) (proxy/ERC1967/ERC1967Upgrade.sol)
                  pragma solidity ^0.8.2;
                  import "../beacon/IBeacon.sol";
                  import "../../interfaces/IERC1967.sol";
                  import "../../interfaces/draft-IERC1822.sol";
                  import "../../utils/Address.sol";
                  import "../../utils/StorageSlot.sol";
                  /**
                   * @dev This abstract contract provides getters and event emitting update functions for
                   * https://eips.ethereum.org/EIPS/eip-1967[EIP1967] slots.
                   *
                   * _Available since v4.1._
                   */
                  abstract contract ERC1967Upgrade is IERC1967 {
                      // This is the keccak-256 hash of "eip1967.proxy.rollback" subtracted by 1
                      bytes32 private constant _ROLLBACK_SLOT = 0x4910fdfa16fed3260ed0e7147f7cc6da11a60208b5b9406d12a635614ffd9143;
                      /**
                       * @dev Storage slot with the address of the current implementation.
                       * This is the keccak-256 hash of "eip1967.proxy.implementation" subtracted by 1, and is
                       * validated in the constructor.
                       */
                      bytes32 internal constant _IMPLEMENTATION_SLOT = 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc;
                      /**
                       * @dev Returns the current implementation address.
                       */
                      function _getImplementation() internal view returns (address) {
                          return StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value;
                      }
                      /**
                       * @dev Stores a new address in the EIP1967 implementation slot.
                       */
                      function _setImplementation(address newImplementation) private {
                          require(Address.isContract(newImplementation), "ERC1967: new implementation is not a contract");
                          StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value = newImplementation;
                      }
                      /**
                       * @dev Perform implementation upgrade
                       *
                       * Emits an {Upgraded} event.
                       */
                      function _upgradeTo(address newImplementation) internal {
                          _setImplementation(newImplementation);
                          emit Upgraded(newImplementation);
                      }
                      /**
                       * @dev Perform implementation upgrade with additional setup call.
                       *
                       * Emits an {Upgraded} event.
                       */
                      function _upgradeToAndCall(address newImplementation, bytes memory data, bool forceCall) internal {
                          _upgradeTo(newImplementation);
                          if (data.length > 0 || forceCall) {
                              Address.functionDelegateCall(newImplementation, data);
                          }
                      }
                      /**
                       * @dev Perform implementation upgrade with security checks for UUPS proxies, and additional setup call.
                       *
                       * Emits an {Upgraded} event.
                       */
                      function _upgradeToAndCallUUPS(address newImplementation, bytes memory data, bool forceCall) internal {
                          // Upgrades from old implementations will perform a rollback test. This test requires the new
                          // implementation to upgrade back to the old, non-ERC1822 compliant, implementation. Removing
                          // this special case will break upgrade paths from old UUPS implementation to new ones.
                          if (StorageSlot.getBooleanSlot(_ROLLBACK_SLOT).value) {
                              _setImplementation(newImplementation);
                          } else {
                              try IERC1822Proxiable(newImplementation).proxiableUUID() returns (bytes32 slot) {
                                  require(slot == _IMPLEMENTATION_SLOT, "ERC1967Upgrade: unsupported proxiableUUID");
                              } catch {
                                  revert("ERC1967Upgrade: new implementation is not UUPS");
                              }
                              _upgradeToAndCall(newImplementation, data, forceCall);
                          }
                      }
                      /**
                       * @dev Storage slot with the admin of the contract.
                       * This is the keccak-256 hash of "eip1967.proxy.admin" subtracted by 1, and is
                       * validated in the constructor.
                       */
                      bytes32 internal constant _ADMIN_SLOT = 0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103;
                      /**
                       * @dev Returns the current admin.
                       */
                      function _getAdmin() internal view returns (address) {
                          return StorageSlot.getAddressSlot(_ADMIN_SLOT).value;
                      }
                      /**
                       * @dev Stores a new address in the EIP1967 admin slot.
                       */
                      function _setAdmin(address newAdmin) private {
                          require(newAdmin != address(0), "ERC1967: new admin is the zero address");
                          StorageSlot.getAddressSlot(_ADMIN_SLOT).value = newAdmin;
                      }
                      /**
                       * @dev Changes the admin of the proxy.
                       *
                       * Emits an {AdminChanged} event.
                       */
                      function _changeAdmin(address newAdmin) internal {
                          emit AdminChanged(_getAdmin(), newAdmin);
                          _setAdmin(newAdmin);
                      }
                      /**
                       * @dev The storage slot of the UpgradeableBeacon contract which defines the implementation for this proxy.
                       * This is bytes32(uint256(keccak256('eip1967.proxy.beacon')) - 1)) and is validated in the constructor.
                       */
                      bytes32 internal constant _BEACON_SLOT = 0xa3f0ad74e5423aebfd80d3ef4346578335a9a72aeaee59ff6cb3582b35133d50;
                      /**
                       * @dev Returns the current beacon.
                       */
                      function _getBeacon() internal view returns (address) {
                          return StorageSlot.getAddressSlot(_BEACON_SLOT).value;
                      }
                      /**
                       * @dev Stores a new beacon in the EIP1967 beacon slot.
                       */
                      function _setBeacon(address newBeacon) private {
                          require(Address.isContract(newBeacon), "ERC1967: new beacon is not a contract");
                          require(
                              Address.isContract(IBeacon(newBeacon).implementation()),
                              "ERC1967: beacon implementation is not a contract"
                          );
                          StorageSlot.getAddressSlot(_BEACON_SLOT).value = newBeacon;
                      }
                      /**
                       * @dev Perform beacon upgrade with additional setup call. Note: This upgrades the address of the beacon, it does
                       * not upgrade the implementation contained in the beacon (see {UpgradeableBeacon-_setImplementation} for that).
                       *
                       * Emits a {BeaconUpgraded} event.
                       */
                      function _upgradeBeaconToAndCall(address newBeacon, bytes memory data, bool forceCall) internal {
                          _setBeacon(newBeacon);
                          emit BeaconUpgraded(newBeacon);
                          if (data.length > 0 || forceCall) {
                              Address.functionDelegateCall(IBeacon(newBeacon).implementation(), data);
                          }
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.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/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.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 8: L1GatewayRouter
                  // 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.
                      /// @param nonce The nonce of cross-chain messages sent at L1.
                      event DepositERC20(
                          address indexed l1Token,
                          address indexed l2Token,
                          address indexed from,
                          address to,
                          uint256 amount,
                          bytes data,
                          uint256 nonce
                      );
                      /// @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 L1CrossDomainMessenger.
                      ///      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;
                  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.
                      /// @param nonce The nonce of cross-chain messages sent at L1.
                      event DepositETH(address indexed from, address indexed to, uint256 amount, bytes data, uint256 nonce);
                      /// @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 L1CrossDomainMessenger.
                      ///      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.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[] memory _tokens, address[] memory _gateways) external;
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity =0.8.24;
                  import {OwnableUpgradeable} from "node_modules/@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol";
                  import {IERC20Upgradeable} from "node_modules/@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol";
                  import {SafeERC20Upgradeable} from "node_modules/@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");
                          // enter deposit context
                          gatewayInContext = _gateway;
                          // encode msg.sender with _data
                          bytes memory _routerData = abi.encode(_msgSender(), _data);
                          IL1ETHGateway(_gateway).depositETHAndCall{value: msg.value}(_to, _amount, _routerData, _gasLimit);
                          // leave deposit context
                          gatewayInContext = address(0);
                      }
                      /// @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 {Initializable} from "../proxy/utils/Initializable.sol";
                  /**
                   * @dev Contract module which provides a basic access control mechanism, where
                   * there is an account (an owner) that can be granted exclusive access to
                   * specific functions.
                   *
                   * By default, the owner account will be the one that deploys the contract. This
                   * can later be changed with {transferOwnership}.
                   *
                   * This module is used through inheritance. It will make available the modifier
                   * `onlyOwner`, which can be applied to your functions to restrict their use to
                   * the owner.
                   */
                  abstract contract OwnableUpgradeable is Initializable, ContextUpgradeable {
                      address private _owner;
                      event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
                      /**
                       * @dev Initializes the contract setting the deployer as the initial owner.
                       */
                      function __Ownable_init() internal onlyInitializing {
                          __Ownable_init_unchained();
                      }
                      function __Ownable_init_unchained() internal onlyInitializing {
                          _transferOwnership(_msgSender());
                      }
                      /**
                       * @dev Throws if called by any account other than the owner.
                       */
                      modifier onlyOwner() {
                          _checkOwner();
                          _;
                      }
                      /**
                       * @dev Returns the address of the current owner.
                       */
                      function owner() public view virtual returns (address) {
                          return _owner;
                      }
                      /**
                       * @dev Throws if the sender is not the owner.
                       */
                      function _checkOwner() internal view virtual {
                          require(owner() == _msgSender(), "Ownable: caller is not the owner");
                      }
                      /**
                       * @dev Leaves the contract without owner. It will not be possible to call
                       * `onlyOwner` functions. Can only be called by the current owner.
                       *
                       * NOTE: Renouncing ownership will leave the contract without an owner,
                       * thereby disabling any functionality that is only available to the owner.
                       */
                      function renounceOwnership() public virtual onlyOwner {
                          _transferOwnership(address(0));
                      }
                      /**
                       * @dev Transfers ownership of the contract to a new account (`newOwner`).
                       * Can only be called by the current owner.
                       */
                      function transferOwnership(address newOwner) public virtual onlyOwner {
                          require(newOwner != address(0), "Ownable: new owner is the zero address");
                          _transferOwnership(newOwner);
                      }
                      /**
                       * @dev Transfers ownership of the contract to a new account (`newOwner`).
                       * Internal function without access restriction.
                       */
                      function _transferOwnership(address newOwner) internal virtual {
                          address oldOwner = _owner;
                          _owner = newOwner;
                          emit OwnershipTransferred(oldOwner, newOwner);
                      }
                      /**
                       * @dev This empty reserved space is put in place to allow future versions to add new
                       * variables without shifting down storage in the inheritance chain.
                       * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
                       */
                      uint256[49] private __gap;
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.9.0) (proxy/utils/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.4) (token/ERC20/extensions/IERC20Permit.sol)
                  pragma solidity ^0.8.0;
                  /**
                   * @dev Interface of the ERC20 Permit extension allowing approvals to be made via signatures, as defined in
                   * https://eips.ethereum.org/EIPS/eip-2612[EIP-2612].
                   *
                   * Adds the {permit} method, which can be used to change an account's ERC20 allowance (see {IERC20-allowance}) by
                   * presenting a message signed by the account. By not relying on {IERC20-approve}, the token holder account doesn't
                   * need to send a transaction, and thus is not required to hold Ether at all.
                   *
                   * ==== Security Considerations
                   *
                   * There are two important considerations concerning the use of `permit`. The first is that a valid permit signature
                   * expresses an allowance, and it should not be assumed to convey additional meaning. In particular, it should not be
                   * considered as an intention to spend the allowance in any specific way. The second is that because permits have
                   * built-in replay protection and can be submitted by anyone, they can be frontrun. A protocol that uses permits should
                   * take this into consideration and allow a `permit` call to fail. Combining these two aspects, a pattern that may be
                   * generally recommended is:
                   *
                   * ```solidity
                   * function doThingWithPermit(..., uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s) public {
                   *     try token.permit(msg.sender, address(this), value, deadline, v, r, s) {} catch {}
                   *     doThing(..., value);
                   * }
                   *
                   * function doThing(..., uint256 value) public {
                   *     token.safeTransferFrom(msg.sender, address(this), value);
                   *     ...
                   * }
                   * ```
                   *
                   * Observe that: 1) `msg.sender` is used as the owner, leaving no ambiguity as to the signer intent, and 2) the use of
                   * `try/catch` allows the permit to fail and makes the code tolerant to frontrunning. (See also
                   * {SafeERC20-safeTransferFrom}).
                   *
                   * Additionally, note that smart contract wallets (such as Argent or Safe) are not able to produce permit signatures, so
                   * contracts should have entry points that don't rely on permit.
                   */
                  interface IERC20PermitUpgradeable {
                      /**
                       * @dev Sets `value` as the allowance of `spender` over ``owner``'s tokens,
                       * given ``owner``'s signed approval.
                       *
                       * IMPORTANT: The same issues {IERC20-approve} has related to transaction
                       * ordering also apply here.
                       *
                       * Emits an {Approval} event.
                       *
                       * Requirements:
                       *
                       * - `spender` cannot be the zero address.
                       * - `deadline` must be a timestamp in the future.
                       * - `v`, `r` and `s` must be a valid `secp256k1` signature from `owner`
                       * over the EIP712-formatted function arguments.
                       * - the signature must use ``owner``'s current nonce (see {nonces}).
                       *
                       * For more information on the signature format, see the
                       * https://eips.ethereum.org/EIPS/eip-2612#specification[relevant EIP
                       * section].
                       *
                       * CAUTION: See Security Considerations above.
                       */
                      function permit(
                          address owner,
                          address spender,
                          uint256 value,
                          uint256 deadline,
                          uint8 v,
                          bytes32 r,
                          bytes32 s
                      ) external;
                      /**
                       * @dev Returns the current nonce for `owner`. This value must be
                       * included whenever a signature is generated for {permit}.
                       *
                       * Every successful call to {permit} increases ``owner``'s nonce by one. This
                       * prevents a signature from being used multiple times.
                       */
                      function nonces(address owner) external view returns (uint256);
                      /**
                       * @dev Returns the domain separator used in the encoding of the signature for {permit}, as defined by {EIP712}.
                       */
                      // solhint-disable-next-line func-name-mixedcase
                      function DOMAIN_SEPARATOR() external view returns (bytes32);
                  }
                  // SPDX-License-Identifier: MIT
                  // 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
                  // OpenZeppelin Contracts (last updated v4.9.0) (utils/Address.sol)
                  pragma solidity ^0.8.1;
                  /**
                   * @dev Collection of functions related to the address type
                   */
                  library AddressUpgradeable {
                      /**
                       * @dev Returns true if `account` is a contract.
                       *
                       * [IMPORTANT]
                       * ====
                       * It is unsafe to assume that an address for which this function returns
                       * false is an externally-owned account (EOA) and not a contract.
                       *
                       * Among others, `isContract` will return false for the following
                       * types of addresses:
                       *
                       *  - an externally-owned account
                       *  - a contract in construction
                       *  - an address where a contract will be created
                       *  - an address where a contract lived, but was destroyed
                       *
                       * Furthermore, `isContract` will also return true if the target contract within
                       * the same transaction is already scheduled for destruction by `SELFDESTRUCT`,
                       * which only has an effect at the end of a transaction.
                       * ====
                       *
                       * [IMPORTANT]
                       * ====
                       * You shouldn't rely on `isContract` to protect against flash loan attacks!
                       *
                       * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets
                       * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract
                       * constructor.
                       * ====
                       */
                      function isContract(address account) internal view returns (bool) {
                          // This method relies on extcodesize/address.code.length, which returns 0
                          // for contracts in construction, since the code is only stored at the end
                          // of the constructor execution.
                          return account.code.length > 0;
                      }
                      /**
                       * @dev Replacement for Solidity's `transfer`: sends `amount` wei to
                       * `recipient`, forwarding all available gas and reverting on errors.
                       *
                       * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
                       * of certain opcodes, possibly making contracts go over the 2300 gas limit
                       * imposed by `transfer`, making them unable to receive funds via
                       * `transfer`. {sendValue} removes this limitation.
                       *
                       * https://consensys.net/diligence/blog/2019/09/stop-using-soliditys-transfer-now/[Learn more].
                       *
                       * IMPORTANT: because control is transferred to `recipient`, care must be
                       * taken to not create reentrancy vulnerabilities. Consider using
                       * {ReentrancyGuard} or the
                       * https://solidity.readthedocs.io/en/v0.8.0/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
                       */
                      function sendValue(address payable recipient, uint256 amount) internal {
                          require(address(this).balance >= amount, "Address: insufficient balance");
                          (bool success, ) = recipient.call{value: amount}("");
                          require(success, "Address: unable to send value, recipient may have reverted");
                      }
                      /**
                       * @dev Performs a Solidity function call using a low level `call`. A
                       * plain `call` is an unsafe replacement for a function call: use this
                       * function instead.
                       *
                       * If `target` reverts with a revert reason, it is bubbled up by this
                       * function (like regular Solidity function calls).
                       *
                       * Returns the raw returned data. To convert to the expected return value,
                       * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
                       *
                       * Requirements:
                       *
                       * - `target` must be a contract.
                       * - calling `target` with `data` must not revert.
                       *
                       * _Available since v3.1._
                       */
                      function functionCall(address target, bytes memory data) internal returns (bytes memory) {
                          return functionCallWithValue(target, data, 0, "Address: low-level call failed");
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
                       * `errorMessage` as a fallback revert reason when `target` reverts.
                       *
                       * _Available since v3.1._
                       */
                      function functionCall(
                          address target,
                          bytes memory data,
                          string memory errorMessage
                      ) internal returns (bytes memory) {
                          return functionCallWithValue(target, data, 0, errorMessage);
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                       * but also transferring `value` wei to `target`.
                       *
                       * Requirements:
                       *
                       * - the calling contract must have an ETH balance of at least `value`.
                       * - the called Solidity function must be `payable`.
                       *
                       * _Available since v3.1._
                       */
                      function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
                          return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
                      }
                      /**
                       * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
                       * with `errorMessage` as a fallback revert reason when `target` reverts.
                       *
                       * _Available since v3.1._
                       */
                      function functionCallWithValue(
                          address target,
                          bytes memory data,
                          uint256 value,
                          string memory errorMessage
                      ) internal returns (bytes memory) {
                          require(address(this).balance >= value, "Address: insufficient balance for call");
                          (bool success, bytes memory returndata) = target.call{value: value}(data);
                          return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                       * but performing a static call.
                       *
                       * _Available since v3.3._
                       */
                      function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
                          return functionStaticCall(target, data, "Address: low-level static call failed");
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                       * but performing a static call.
                       *
                       * _Available since v3.3._
                       */
                      function functionStaticCall(
                          address target,
                          bytes memory data,
                          string memory errorMessage
                      ) internal view returns (bytes memory) {
                          (bool success, bytes memory returndata) = target.staticcall(data);
                          return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                       * but performing a delegate call.
                       *
                       * _Available since v3.4._
                       */
                      function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
                          return functionDelegateCall(target, data, "Address: low-level delegate call failed");
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                       * but performing a delegate call.
                       *
                       * _Available since v3.4._
                       */
                      function functionDelegateCall(
                          address target,
                          bytes memory data,
                          string memory errorMessage
                      ) internal returns (bytes memory) {
                          (bool success, bytes memory returndata) = target.delegatecall(data);
                          return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                      }
                      /**
                       * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling
                       * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract.
                       *
                       * _Available since v4.8._
                       */
                      function verifyCallResultFromTarget(
                          address target,
                          bool success,
                          bytes memory returndata,
                          string memory errorMessage
                      ) internal view returns (bytes memory) {
                          if (success) {
                              if (returndata.length == 0) {
                                  // only check isContract if the call was successful and the return data is empty
                                  // otherwise we already know that it was a contract
                                  require(isContract(target), "Address: call to non-contract");
                              }
                              return returndata;
                          } else {
                              _revert(returndata, errorMessage);
                          }
                      }
                      /**
                       * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the
                       * revert reason or using the provided one.
                       *
                       * _Available since v4.3._
                       */
                      function verifyCallResult(
                          bool success,
                          bytes memory returndata,
                          string memory errorMessage
                      ) internal pure returns (bytes memory) {
                          if (success) {
                              return returndata;
                          } else {
                              _revert(returndata, errorMessage);
                          }
                      }
                      function _revert(bytes memory returndata, string memory errorMessage) private pure {
                          // Look for revert reason and bubble it up if present
                          if (returndata.length > 0) {
                              // The easiest way to bubble the revert reason is using memory via assembly
                              /// @solidity memory-safe-assembly
                              assembly {
                                  let returndata_size := mload(returndata)
                                  revert(add(32, returndata), returndata_size)
                              }
                          } else {
                              revert(errorMessage);
                          }
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.9.4) (utils/Context.sol)
                  pragma solidity ^0.8.0;
                  import {Initializable} from "../proxy/utils/Initializable.sol";
                  /**
                   * @dev Provides information about the current execution context, including the
                   * sender of the transaction and its data. While these are generally available
                   * via msg.sender and msg.data, they should not be accessed in such a direct
                   * manner, since when dealing with meta-transactions the account sending and
                   * paying for execution may not be the actual sender (as far as an application
                   * is concerned).
                   *
                   * This contract is only required for intermediate, library-like contracts.
                   */
                  abstract contract ContextUpgradeable is Initializable {
                      function __Context_init() internal onlyInitializing {
                      }
                      function __Context_init_unchained() internal onlyInitializing {
                      }
                      function _msgSender() internal view virtual returns (address) {
                          return msg.sender;
                      }
                      function _msgData() internal view virtual returns (bytes calldata) {
                          return msg.data;
                      }
                      function _contextSuffixLength() internal view virtual returns (uint256) {
                          return 0;
                      }
                      /**
                       * @dev This empty reserved space is put in place to allow future versions to add new
                       * variables without shifting down storage in the inheritance chain.
                       * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
                       */
                      uint256[50] private __gap;
                  }
                  

                  File 7 of 8: L1ETHGateway
                  // 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.
                      /// @param nonce The nonce of cross-chain messages sent at L1.
                      event DepositETH(address indexed from, address indexed to, uint256 amount, bytes data, uint256 nonce);
                      /// @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 L1CrossDomainMessenger.
                      ///      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.24;
                  import {IL2ETHGateway} from "../../l2/gateways/IL2ETHGateway.sol";
                  import {IL1CrossDomainMessenger} from "../IL1CrossDomainMessenger.sol";
                  import {IL1ETHGateway} from "./IL1ETHGateway.sol";
                  import {IMessageDropCallback} from "../../libraries/callbacks/IMessageDropCallback.sol";
                  import {GatewayBase} from "../../libraries/gateway/GatewayBase.sol";
                  // solhint-disable avoid-low-level-calls
                  /// @title L1ETHGateway
                  /// @notice The `L1ETHGateway` is used to deposit ETH on layer 1 and
                  /// finalize withdraw ETH from layer 2.
                  /// @dev The deposited ETH tokens are held in this gateway. On finalizing withdraw, the corresponding
                  /// ETH will be transfer to the recipient directly.
                  contract L1ETHGateway is GatewayBase, IL1ETHGateway, IMessageDropCallback {
                      /***************
                       * Constructor *
                       ***************/
                      constructor() {
                          _disableInitializers();
                      }
                      /// @notice Initialize the storage of L1ETHGateway.
                      /// @param _counterpart The address of L2ETHGateway in L2.
                      /// @param _router The address of L1GatewayRouter.
                      /// @param _messenger The address of L1CrossDomainMessenger.
                      function initialize(address _counterpart, address _router, address _messenger) external initializer {
                          require(_router != address(0), "zero router address");
                          GatewayBase._initialize(_counterpart, _router, _messenger);
                      }
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @inheritdoc IL1ETHGateway
                      function depositETH(uint256 _amount, uint256 _gasLimit) external payable override {
                          _deposit(_msgSender(), _amount, new bytes(0), _gasLimit);
                      }
                      /// @inheritdoc IL1ETHGateway
                      function depositETH(address _to, uint256 _amount, uint256 _gasLimit) external payable override {
                          _deposit(_to, _amount, new bytes(0), _gasLimit);
                      }
                      /// @inheritdoc IL1ETHGateway
                      function depositETHAndCall(
                          address _to,
                          uint256 _amount,
                          bytes calldata _data,
                          uint256 _gasLimit
                      ) external payable override {
                          _deposit(_to, _amount, _data, _gasLimit);
                      }
                      /// @inheritdoc IL1ETHGateway
                      function finalizeWithdrawETH(
                          address _from,
                          address _to,
                          uint256 _amount,
                          bytes calldata _data
                      ) external payable override onlyCallByCounterpart nonReentrant {
                          require(msg.value == _amount, "msg.value mismatch");
                          // @note can possible trigger reentrant call to messenger,
                          // but it seems not a big problem.
                          (bool _success, ) = _to.call{value: _amount}("");
                          require(_success, "ETH transfer failed");
                          _doCallback(_to, _data);
                          emit FinalizeWithdrawETH(_from, _to, _amount, _data);
                      }
                      /// @inheritdoc IMessageDropCallback
                      function onDropMessage(bytes calldata _message) external payable virtual onlyInDropContext nonReentrant {
                          // _message should start with 0x232e8748  =>  finalizeDepositETH(address,address,uint256,bytes)
                          require(bytes4(_message[0:4]) == IL2ETHGateway.finalizeDepositETH.selector, "invalid selector");
                          // decode (receiver, amount)
                          (address _receiver, , uint256 _amount, ) = abi.decode(_message[4:], (address, address, uint256, bytes));
                          require(_amount == msg.value, "msg.value mismatch");
                          (bool _success, ) = _receiver.call{value: _amount}("");
                          require(_success, "ETH transfer failed");
                          emit RefundETH(_receiver, _amount);
                      }
                      /**********************
                       * Internal Functions *
                       **********************/
                      /// @dev The internal ETH deposit implementation.
                      /// @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 _deposit(
                          address _to,
                          uint256 _amount,
                          bytes memory _data,
                          uint256 _gasLimit
                      ) internal virtual nonReentrant {
                          require(_amount > 0, "deposit zero eth");
                          // 1. Extract real sender if this call is from L1GatewayRouter.
                          address _from = _msgSender();
                          if (router == _from) {
                              (_from, _data) = abi.decode(_data, (address, bytes));
                          }
                          // @note no rate limit here, since ETH is limited in messenger
                          // 2. Generate message passed to L1CrossDomainMessenger.
                          bytes memory _message = abi.encodeCall(IL2ETHGateway.finalizeDepositETH, (_from, _to, _amount, _data));
                          uint256 nonce = IL1CrossDomainMessenger(messenger).messageNonce();
                          IL1CrossDomainMessenger(messenger).sendMessage{value: msg.value}(
                              counterpart,
                              _amount,
                              _message,
                              _gasLimit,
                              _from
                          );
                          emit DepositETH(_from, _to, _amount, _data, nonce);
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity =0.8.24;
                  import {ICrossDomainMessenger} from "../libraries/ICrossDomainMessenger.sol";
                  interface IL1CrossDomainMessenger is ICrossDomainMessenger {
                      /**********
                       * Events *
                       **********/
                      /// @dev Emitted when the rollup contract address is updated.
                      /// @param oldRollup The address of the old rollup contract.
                      /// @param newRollup The address of the new rollup contract.
                      event UpdateRollup(address oldRollup, address newRollup);
                      /// @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);
                      /// @notice Emitted when have message relay.
                      /// @param oldNonce The index of the old message to be replayed.
                      /// @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 ReplayMessage(
                          uint256 indexed oldNonce,
                          address indexed sender,
                          address indexed target,
                          uint256 value,
                          uint256 messageNonce,
                          uint256 gasLimit,
                          bytes message
                      );
                      /// @notice Emitted when have message dropped.
                      /// @param nonce The index of the message to be dropped.
                      event DropMessage(uint256 indexed nonce);
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @notice Prove a L2 => L1 message with message proof and relay a L2 => L1 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 _nonce The nonce of the message to avoid replay attack.
                      /// @param _message The content of the message.
                      /// @param _withdrawalProof Merkle tree proof of the message.
                      /// @param _withdrawalRoot Merkle tree root of the proof.
                      function proveAndRelayMessage(
                          address _from,
                          address _to,
                          uint256 _value,
                          uint256 _nonce,
                          bytes memory _message,
                          bytes32[32] calldata _withdrawalProof,
                          bytes32 _withdrawalRoot
                      ) 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 IL2ETHGateway {
                      /**********
                       * Events *
                       **********/
                      /// @notice Emitted when someone withdraw ETH from L2 to L1.
                      /// @param from The address of sender in L2.
                      /// @param to The address of recipient in L1.
                      /// @param amount The amount of ETH will be deposited from L2 to L1.
                      /// @param data The optional calldata passed to recipient in L1.
                      /// @param nonce The nonce of cross-chain messages sent at L2.
                      event WithdrawETH(address indexed from, address indexed to, uint256 amount, bytes data, uint256 nonce);
                      /// @notice Emitted when ETH is deposited from L1 to L2 and transfer to recipient.
                      /// @param from The address of sender in L1.
                      /// @param to The address of recipient in L2.
                      /// @param amount The amount of ETH deposited from L1 to L2.
                      /// @param data The optional calldata passed to recipient in L2.
                      event FinalizeDepositETH(address indexed from, address indexed to, uint256 amount, bytes data);
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @notice Withdraw ETH to caller's account in L1.
                      /// @param amount The amount of ETH to be withdrawn.
                      /// @param gasLimit Optional, gas limit used to complete the withdraw on L1.
                      function withdrawETH(uint256 amount, uint256 gasLimit) external payable;
                      /// @notice Withdraw ETH to caller's account in L1.
                      /// @param to The address of recipient's account on L1.
                      /// @param amount The amount of ETH to be withdrawn.
                      /// @param gasLimit Optional, gas limit used to complete the withdraw on L1.
                      function withdrawETH(address to, uint256 amount, uint256 gasLimit) external payable;
                      /// @notice Withdraw ETH to caller's account in L1.
                      /// @param to The address of recipient's account on L1.
                      /// @param amount The amount of ETH to be withdrawn.
                      /// @param data Optional data to forward to recipient's account.
                      /// @param gasLimit Optional, gas limit used to complete the withdraw on L1.
                      function withdrawETHAndCall(address to, uint256 amount, bytes calldata data, uint256 gasLimit) external payable;
                      /// @notice Complete ETH deposit from L1 to L2 and send fund to recipient's account in L2.
                      /// @dev This function should only be called by L2CrossDomainMessenger.
                      ///      This function should also only be called by L1GatewayRouter in L1.
                      /// @param _from The address of account who deposit ETH in L1.
                      /// @param _to The address of recipient in L2 to receive ETH.
                      /// @param _amount The amount of ETH to deposit.
                      /// @param _data Optional data to forward to recipient's account.
                      function finalizeDepositETH(address _from, address _to, uint256 _amount, bytes calldata _data) external payable;
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.9;
                  interface IGatewayCallback {
                      function onGatewayCallback(bytes memory data) external;
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.9;
                  interface IMessageDropCallback {
                      function onDropMessage(bytes memory message) external payable;
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity =0.8.24;
                  /**
                   * @title Constants
                   * @notice Constants is a library for storing constants. Simple! Don't put everything in here, just
                   *         the stuff used in multiple contracts. Constants that only apply to a single contract
                   *         should be defined in that contract instead.
                   */
                  library Constants {
                      /**
                       * @notice Value used for the L2 sender storage slot in both the MorphPortal and the
                       *         CrossDomainMessenger contracts before an actual sender is set. This value is
                       *         non-zero to reduce the gas cost of message passing transactions.
                       */
                      address internal constant DEFAULT_XDOMAIN_MESSAGE_SENDER = 0x000000000000000000000000000000000000dEaD;
                      /// @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;
                  import {OwnableUpgradeable} from "node_modules/@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol";
                  import {ReentrancyGuardUpgradeable} from "node_modules/@openzeppelin/contracts-upgradeable/security/ReentrancyGuardUpgradeable.sol";
                  import {IGateway} from "./IGateway.sol";
                  import {ICrossDomainMessenger} from "../ICrossDomainMessenger.sol";
                  import {IGatewayCallback} from "../callbacks/IGatewayCallback.sol";
                  import {Constants} from "../constants/Constants.sol";
                  /// @title GatewayBase
                  /// @notice The `GatewayBase` is a base contract for gateway contracts used in both in L1 and L2.
                  abstract contract GatewayBase is ReentrancyGuardUpgradeable, OwnableUpgradeable, IGateway {
                      /*************
                       * Variables *
                       *************/
                      /// @inheritdoc IGateway
                      address public override counterpart;
                      /// @inheritdoc IGateway
                      address public override router;
                      /// @inheritdoc IGateway
                      address public override messenger;
                      /// @dev The storage slots for future usage.
                      uint256[46] private __gap;
                      /**********************
                       * Function Modifiers *
                       **********************/
                      modifier onlyCallByCounterpart() {
                          address _messenger = messenger; // gas saving
                          require(_msgSender() == _messenger, "only messenger can call");
                          require(counterpart == ICrossDomainMessenger(_messenger).xDomainMessageSender(), "only call by counterpart");
                          _;
                      }
                      modifier onlyInDropContext() {
                          address _messenger = messenger; // gas saving
                          require(_msgSender() == _messenger, "only messenger can call");
                          require(
                              Constants.DROP_XDOMAIN_MESSAGE_SENDER == ICrossDomainMessenger(_messenger).xDomainMessageSender(),
                              "only called in drop context"
                          );
                          _;
                      }
                      /***************
                       * Constructor *
                       ***************/
                      function _initialize(address _counterpart, address _router, address _messenger) internal {
                          require(_counterpart != address(0), "zero counterpart address");
                          require(_messenger != address(0), "zero messenger address");
                          ReentrancyGuardUpgradeable.__ReentrancyGuard_init();
                          OwnableUpgradeable.__Ownable_init();
                          counterpart = _counterpart;
                          messenger = _messenger;
                          // @note: the address of router could be zero, if this contract is GatewayRouter.
                          if (_router != address(0)) {
                              router = _router;
                          }
                      }
                      /**********************
                       * 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) {
                              IGatewayCallback(_to).onGatewayCallback(_data);
                          }
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity =0.8.24;
                  interface IGateway {
                      /// @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 L1CrossDomainMessenger/L2CrossDomainMessenger contract.
                      function messenger() external view returns (address);
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.9;
                  interface ICrossDomainMessenger {
                      /***********
                       * Errors *
                       ***********/
                      error ErrZeroAddress();
                      /**********
                       * 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);
                      /*************************
                       * Public View Functions *
                       *************************/
                      /// @notice Return the sender of a cross domain message.
                      function xDomainMessageSender() external view returns (address);
                      /// @notice Return the nonce of a cross domain message.
                      function messageNonce() external view returns (uint256);
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @notice Send cross chain message from L1 to L2 or L2 to L1.
                      /// @dev EOA addresses and contracts that have not implemented `onDropMessage`
                      /// cannot execute the `dropMessage` operation.
                      /// Please proceed with caution to control risk.
                      /// @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.
                      /// @dev EOA addresses and contracts that have not implemented `onDropMessage`
                      /// cannot execute the `dropMessage` operation.
                      /// Please proceed with caution to control risk.
                      /// @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 {Initializable} from "../proxy/utils/Initializable.sol";
                  /**
                   * @dev Contract module which provides a basic access control mechanism, where
                   * there is an account (an owner) that can be granted exclusive access to
                   * specific functions.
                   *
                   * By default, the owner account will be the one that deploys the contract. This
                   * can later be changed with {transferOwnership}.
                   *
                   * This module is used through inheritance. It will make available the modifier
                   * `onlyOwner`, which can be applied to your functions to restrict their use to
                   * the owner.
                   */
                  abstract contract OwnableUpgradeable is Initializable, ContextUpgradeable {
                      address private _owner;
                      event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
                      /**
                       * @dev Initializes the contract setting the deployer as the initial owner.
                       */
                      function __Ownable_init() internal onlyInitializing {
                          __Ownable_init_unchained();
                      }
                      function __Ownable_init_unchained() internal onlyInitializing {
                          _transferOwnership(_msgSender());
                      }
                      /**
                       * @dev Throws if called by any account other than the owner.
                       */
                      modifier onlyOwner() {
                          _checkOwner();
                          _;
                      }
                      /**
                       * @dev Returns the address of the current owner.
                       */
                      function owner() public view virtual returns (address) {
                          return _owner;
                      }
                      /**
                       * @dev Throws if the sender is not the owner.
                       */
                      function _checkOwner() internal view virtual {
                          require(owner() == _msgSender(), "Ownable: caller is not the owner");
                      }
                      /**
                       * @dev Leaves the contract without owner. It will not be possible to call
                       * `onlyOwner` functions. Can only be called by the current owner.
                       *
                       * NOTE: Renouncing ownership will leave the contract without an owner,
                       * thereby disabling any functionality that is only available to the owner.
                       */
                      function renounceOwnership() public virtual onlyOwner {
                          _transferOwnership(address(0));
                      }
                      /**
                       * @dev Transfers ownership of the contract to a new account (`newOwner`).
                       * Can only be called by the current owner.
                       */
                      function transferOwnership(address newOwner) public virtual onlyOwner {
                          require(newOwner != address(0), "Ownable: new owner is the zero address");
                          _transferOwnership(newOwner);
                      }
                      /**
                       * @dev Transfers ownership of the contract to a new account (`newOwner`).
                       * Internal function without access restriction.
                       */
                      function _transferOwnership(address newOwner) internal virtual {
                          address oldOwner = _owner;
                          _owner = newOwner;
                          emit OwnershipTransferred(oldOwner, newOwner);
                      }
                      /**
                       * @dev This empty reserved space is put in place to allow future versions to add new
                       * variables without shifting down storage in the inheritance chain.
                       * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
                       */
                      uint256[49] private __gap;
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.9.0) (proxy/utils/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) (security/ReentrancyGuard.sol)
                  pragma solidity ^0.8.0;
                  import {Initializable} from "../proxy/utils/Initializable.sol";
                  /**
                   * @dev Contract module that helps prevent reentrant calls to a function.
                   *
                   * Inheriting from `ReentrancyGuard` will make the {nonReentrant} modifier
                   * available, which can be applied to functions to make sure there are no nested
                   * (reentrant) calls to them.
                   *
                   * Note that because there is a single `nonReentrant` guard, functions marked as
                   * `nonReentrant` may not call one another. This can be worked around by making
                   * those functions `private`, and then adding `external` `nonReentrant` entry
                   * points to them.
                   *
                   * TIP: If you would like to learn more about reentrancy and alternative ways
                   * to protect against it, check out our blog post
                   * https://blog.openzeppelin.com/reentrancy-after-istanbul/[Reentrancy After Istanbul].
                   */
                  abstract contract ReentrancyGuardUpgradeable is Initializable {
                      // Booleans are more expensive than uint256 or any type that takes up a full
                      // word because each write operation emits an extra SLOAD to first read the
                      // slot's contents, replace the bits taken up by the boolean, and then write
                      // back. This is the compiler's defense against contract upgrades and
                      // pointer aliasing, and it cannot be disabled.
                      // The values being non-zero value makes deployment a bit more expensive,
                      // but in exchange the refund on every call to nonReentrant will be lower in
                      // amount. Since refunds are capped to a percentage of the total
                      // transaction's gas, it is best to keep them low in cases like this one, to
                      // increase the likelihood of the full refund coming into effect.
                      uint256 private constant _NOT_ENTERED = 1;
                      uint256 private constant _ENTERED = 2;
                      uint256 private _status;
                      function __ReentrancyGuard_init() internal onlyInitializing {
                          __ReentrancyGuard_init_unchained();
                      }
                      function __ReentrancyGuard_init_unchained() internal onlyInitializing {
                          _status = _NOT_ENTERED;
                      }
                      /**
                       * @dev Prevents a contract from calling itself, directly or indirectly.
                       * Calling a `nonReentrant` function from another `nonReentrant`
                       * function is not supported. It is possible to prevent this from happening
                       * by making the `nonReentrant` function external, and making it call a
                       * `private` function that does the actual work.
                       */
                      modifier nonReentrant() {
                          _nonReentrantBefore();
                          _;
                          _nonReentrantAfter();
                      }
                      function _nonReentrantBefore() private {
                          // On the first call to nonReentrant, _status will be _NOT_ENTERED
                          require(_status != _ENTERED, "ReentrancyGuard: reentrant call");
                          // Any calls to nonReentrant after this point will fail
                          _status = _ENTERED;
                      }
                      function _nonReentrantAfter() private {
                          // By storing the original value once again, a refund is triggered (see
                          // https://eips.ethereum.org/EIPS/eip-2200)
                          _status = _NOT_ENTERED;
                      }
                      /**
                       * @dev Returns true if the reentrancy guard is currently set to "entered", which indicates there is a
                       * `nonReentrant` function in the call stack.
                       */
                      function _reentrancyGuardEntered() internal view returns (bool) {
                          return _status == _ENTERED;
                      }
                      /**
                       * @dev This empty reserved space is put in place to allow future versions to add new
                       * variables without shifting down storage in the inheritance chain.
                       * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
                       */
                      uint256[49] private __gap;
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.9.0) (utils/Address.sol)
                  pragma solidity ^0.8.1;
                  /**
                   * @dev Collection of functions related to the address type
                   */
                  library AddressUpgradeable {
                      /**
                       * @dev Returns true if `account` is a contract.
                       *
                       * [IMPORTANT]
                       * ====
                       * It is unsafe to assume that an address for which this function returns
                       * false is an externally-owned account (EOA) and not a contract.
                       *
                       * Among others, `isContract` will return false for the following
                       * types of addresses:
                       *
                       *  - an externally-owned account
                       *  - a contract in construction
                       *  - an address where a contract will be created
                       *  - an address where a contract lived, but was destroyed
                       *
                       * Furthermore, `isContract` will also return true if the target contract within
                       * the same transaction is already scheduled for destruction by `SELFDESTRUCT`,
                       * which only has an effect at the end of a transaction.
                       * ====
                       *
                       * [IMPORTANT]
                       * ====
                       * You shouldn't rely on `isContract` to protect against flash loan attacks!
                       *
                       * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets
                       * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract
                       * constructor.
                       * ====
                       */
                      function isContract(address account) internal view returns (bool) {
                          // This method relies on extcodesize/address.code.length, which returns 0
                          // for contracts in construction, since the code is only stored at the end
                          // of the constructor execution.
                          return account.code.length > 0;
                      }
                      /**
                       * @dev Replacement for Solidity's `transfer`: sends `amount` wei to
                       * `recipient`, forwarding all available gas and reverting on errors.
                       *
                       * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
                       * of certain opcodes, possibly making contracts go over the 2300 gas limit
                       * imposed by `transfer`, making them unable to receive funds via
                       * `transfer`. {sendValue} removes this limitation.
                       *
                       * https://consensys.net/diligence/blog/2019/09/stop-using-soliditys-transfer-now/[Learn more].
                       *
                       * IMPORTANT: because control is transferred to `recipient`, care must be
                       * taken to not create reentrancy vulnerabilities. Consider using
                       * {ReentrancyGuard} or the
                       * https://solidity.readthedocs.io/en/v0.8.0/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
                       */
                      function sendValue(address payable recipient, uint256 amount) internal {
                          require(address(this).balance >= amount, "Address: insufficient balance");
                          (bool success, ) = recipient.call{value: amount}("");
                          require(success, "Address: unable to send value, recipient may have reverted");
                      }
                      /**
                       * @dev Performs a Solidity function call using a low level `call`. A
                       * plain `call` is an unsafe replacement for a function call: use this
                       * function instead.
                       *
                       * If `target` reverts with a revert reason, it is bubbled up by this
                       * function (like regular Solidity function calls).
                       *
                       * Returns the raw returned data. To convert to the expected return value,
                       * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
                       *
                       * Requirements:
                       *
                       * - `target` must be a contract.
                       * - calling `target` with `data` must not revert.
                       *
                       * _Available since v3.1._
                       */
                      function functionCall(address target, bytes memory data) internal returns (bytes memory) {
                          return functionCallWithValue(target, data, 0, "Address: low-level call failed");
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
                       * `errorMessage` as a fallback revert reason when `target` reverts.
                       *
                       * _Available since v3.1._
                       */
                      function functionCall(
                          address target,
                          bytes memory data,
                          string memory errorMessage
                      ) internal returns (bytes memory) {
                          return functionCallWithValue(target, data, 0, errorMessage);
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                       * but also transferring `value` wei to `target`.
                       *
                       * Requirements:
                       *
                       * - the calling contract must have an ETH balance of at least `value`.
                       * - the called Solidity function must be `payable`.
                       *
                       * _Available since v3.1._
                       */
                      function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
                          return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
                      }
                      /**
                       * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
                       * with `errorMessage` as a fallback revert reason when `target` reverts.
                       *
                       * _Available since v3.1._
                       */
                      function functionCallWithValue(
                          address target,
                          bytes memory data,
                          uint256 value,
                          string memory errorMessage
                      ) internal returns (bytes memory) {
                          require(address(this).balance >= value, "Address: insufficient balance for call");
                          (bool success, bytes memory returndata) = target.call{value: value}(data);
                          return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                       * but performing a static call.
                       *
                       * _Available since v3.3._
                       */
                      function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
                          return functionStaticCall(target, data, "Address: low-level static call failed");
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                       * but performing a static call.
                       *
                       * _Available since v3.3._
                       */
                      function functionStaticCall(
                          address target,
                          bytes memory data,
                          string memory errorMessage
                      ) internal view returns (bytes memory) {
                          (bool success, bytes memory returndata) = target.staticcall(data);
                          return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                       * but performing a delegate call.
                       *
                       * _Available since v3.4._
                       */
                      function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
                          return functionDelegateCall(target, data, "Address: low-level delegate call failed");
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                       * but performing a delegate call.
                       *
                       * _Available since v3.4._
                       */
                      function functionDelegateCall(
                          address target,
                          bytes memory data,
                          string memory errorMessage
                      ) internal returns (bytes memory) {
                          (bool success, bytes memory returndata) = target.delegatecall(data);
                          return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                      }
                      /**
                       * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling
                       * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract.
                       *
                       * _Available since v4.8._
                       */
                      function verifyCallResultFromTarget(
                          address target,
                          bool success,
                          bytes memory returndata,
                          string memory errorMessage
                      ) internal view returns (bytes memory) {
                          if (success) {
                              if (returndata.length == 0) {
                                  // only check isContract if the call was successful and the return data is empty
                                  // otherwise we already know that it was a contract
                                  require(isContract(target), "Address: call to non-contract");
                              }
                              return returndata;
                          } else {
                              _revert(returndata, errorMessage);
                          }
                      }
                      /**
                       * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the
                       * revert reason or using the provided one.
                       *
                       * _Available since v4.3._
                       */
                      function verifyCallResult(
                          bool success,
                          bytes memory returndata,
                          string memory errorMessage
                      ) internal pure returns (bytes memory) {
                          if (success) {
                              return returndata;
                          } else {
                              _revert(returndata, errorMessage);
                          }
                      }
                      function _revert(bytes memory returndata, string memory errorMessage) private pure {
                          // Look for revert reason and bubble it up if present
                          if (returndata.length > 0) {
                              // The easiest way to bubble the revert reason is using memory via assembly
                              /// @solidity memory-safe-assembly
                              assembly {
                                  let returndata_size := mload(returndata)
                                  revert(add(32, returndata), returndata_size)
                              }
                          } else {
                              revert(errorMessage);
                          }
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.9.4) (utils/Context.sol)
                  pragma solidity ^0.8.0;
                  import {Initializable} from "../proxy/utils/Initializable.sol";
                  /**
                   * @dev Provides information about the current execution context, including the
                   * sender of the transaction and its data. While these are generally available
                   * via msg.sender and msg.data, they should not be accessed in such a direct
                   * manner, since when dealing with meta-transactions the account sending and
                   * paying for execution may not be the actual sender (as far as an application
                   * is concerned).
                   *
                   * This contract is only required for intermediate, library-like contracts.
                   */
                  abstract contract ContextUpgradeable is Initializable {
                      function __Context_init() internal onlyInitializing {
                      }
                      function __Context_init_unchained() internal onlyInitializing {
                      }
                      function _msgSender() internal view virtual returns (address) {
                          return msg.sender;
                      }
                      function _msgData() internal view virtual returns (bytes calldata) {
                          return msg.data;
                      }
                      function _contextSuffixLength() internal view virtual returns (uint256) {
                          return 0;
                      }
                      /**
                       * @dev This empty reserved space is put in place to allow future versions to add new
                       * variables without shifting down storage in the inheritance chain.
                       * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
                       */
                      uint256[50] private __gap;
                  }
                  

                  File 8 of 8: L1CrossDomainMessenger
                  // SPDX-License-Identifier: MIT
                  pragma solidity =0.8.24;
                  import {ICrossDomainMessenger} from "../libraries/ICrossDomainMessenger.sol";
                  interface IL1CrossDomainMessenger is ICrossDomainMessenger {
                      /**********
                       * Events *
                       **********/
                      /// @dev Emitted when the rollup contract address is updated.
                      /// @param oldRollup The address of the old rollup contract.
                      /// @param newRollup The address of the new rollup contract.
                      event UpdateRollup(address oldRollup, address newRollup);
                      /// @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);
                      /// @notice Emitted when have message relay.
                      /// @param oldNonce The index of the old message to be replayed.
                      /// @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 ReplayMessage(
                          uint256 indexed oldNonce,
                          address indexed sender,
                          address indexed target,
                          uint256 value,
                          uint256 messageNonce,
                          uint256 gasLimit,
                          bytes message
                      );
                      /// @notice Emitted when have message dropped.
                      /// @param nonce The index of the message to be dropped.
                      event DropMessage(uint256 indexed nonce);
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @notice Prove a L2 => L1 message with message proof and relay a L2 => L1 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 _nonce The nonce of the message to avoid replay attack.
                      /// @param _message The content of the message.
                      /// @param _withdrawalProof Merkle tree proof of the message.
                      /// @param _withdrawalRoot Merkle tree root of the proof.
                      function proveAndRelayMessage(
                          address _from,
                          address _to,
                          uint256 _value,
                          uint256 _nonce,
                          bytes memory _message,
                          bytes32[32] calldata _withdrawalProof,
                          bytes32 _withdrawalRoot
                      ) 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;
                  import {IMessageDropCallback} from "../libraries/callbacks/IMessageDropCallback.sol";
                  import {Predeploys} from "../libraries/constants/Predeploys.sol";
                  import {Constants} from "../libraries/constants/Constants.sol";
                  import {CrossDomainMessenger} from "../libraries/CrossDomainMessenger.sol";
                  import {ICrossDomainMessenger} from "../libraries/ICrossDomainMessenger.sol";
                  import {IL1MessageQueue} from "./rollup/IL1MessageQueue.sol";
                  import {IRollup} from "./rollup/IRollup.sol";
                  import {Verify} from "../libraries/common/Verify.sol";
                  import {IL1CrossDomainMessenger} from "./IL1CrossDomainMessenger.sol";
                  /**
                   * @custom:proxied
                   * @title L1CrossDomainMessenger
                   * @notice The L1CrossDomainMessenger is a message passing interface between L1 and L2 responsible
                   *         for sending and receiving data on the L1 side. Users are encouraged to use this
                   *         interface instead of interacting with lower-level contracts directly.
                   */
                  contract L1CrossDomainMessenger is IL1CrossDomainMessenger, CrossDomainMessenger, Verify {
                      /***********
                       * Structs *
                       ***********/
                      struct ReplayState {
                          // The number of replayed times.
                          uint128 times;
                          // The queue index of latest replayed one. If it is zero, it means the message has not been replayed.
                          uint128 lastIndex;
                      }
                      /*************
                       * Variables *
                       *************/
                      /**
                       * @notice A list of withdrawal hashes which have been successfully finalized.
                       */
                      mapping(bytes32 => bool) public finalizedWithdrawals;
                      /// @notice Mapping from L1 message hash to the timestamp when the message is sent.
                      mapping(bytes32 => uint256) public messageSendTimestamp;
                      /// @notice Mapping from L1 message hash to drop status.
                      mapping(bytes32 => bool) public isL1MessageDropped;
                      /// @notice The address of Rollup contract.
                      address public rollup;
                      /// @notice The address of L1MessageQueue contract.
                      address public 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() {
                          _disableInitializers();
                      }
                      /// @notice Initialize the storage of L1CrossDomainMessenger.
                      /// @param _feeVault The address of fee vault, which will be used to collect relayer fee.
                      /// @param _rollup The address of rollup contract.
                      /// @param _messageQueue The address of L1MessageQueue contract.
                      function initialize(address _feeVault, address _rollup, address _messageQueue) public initializer {
                          if (_rollup == address(0) || _messageQueue == address(0) || _feeVault == address(0)) {
                              revert ErrZeroAddress();
                          }
                          CrossDomainMessenger.__Messenger_init(Predeploys.L2_CROSS_DOMAIN_MESSENGER, _feeVault);
                          rollup = _rollup;
                          messageQueue = _messageQueue;
                          maxReplayTimes = 3;
                          emit UpdateMaxReplayTimes(0, 3);
                      }
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @inheritdoc ICrossDomainMessenger
                      function sendMessage(
                          address _to,
                          uint256 _value,
                          bytes memory _message,
                          uint256 _gasLimit
                      ) external payable override whenNotPaused {
                          _sendMessage(_to, _value, _message, _gasLimit, _msgSender());
                      }
                      /// @inheritdoc ICrossDomainMessenger
                      function sendMessage(
                          address _to,
                          uint256 _value,
                          bytes calldata _message,
                          uint256 _gasLimit,
                          address _refundAddress
                      ) external payable override whenNotPaused {
                          _sendMessage(_to, _value, _message, _gasLimit, _refundAddress);
                      }
                      function proveAndRelayMessage(
                          address _from,
                          address _to,
                          uint256 _value,
                          uint256 _nonce,
                          bytes memory _message,
                          bytes32[32] calldata _withdrawalProof,
                          bytes32 _withdrawalRoot
                      ) external override whenNotPaused notInExecution {
                          // @note check more `_to` address to avoid attack in the future when we add more gateways.
                          require(_to != messageQueue, "Messenger: Forbid to call message queue");
                          _validateTargetAddress(_to);
                          // @note This usually will never happen, just in case.
                          require(_from != xDomainMessageSender, "Messenger: Invalid message sender");
                          bytes32 _xDomainCalldataHash = keccak256(_encodeXDomainCalldata(_from, _to, _value, _nonce, _message));
                          // Check that this withdrawal has not already been finalized, this is replay protection.
                          require(!finalizedWithdrawals[_xDomainCalldataHash], "Messenger: withdrawal has already been finalized");
                          address _rollup = rollup;
                          // prove message
                          {
                              // withdrawalRoot for withdraw proof verify
                              bool finalized = IRollup(_rollup).withdrawalRoots(_withdrawalRoot);
                              require(finalized, "Messenger: withdrawalRoot not finalized");
                              // Verify that the hash of this withdrawal was stored in the L2toL1MessagePasser contract
                              // on L2. If this is true, under the assumption that the Tree does not have
                              // bugs, then we know that this withdrawal was actually triggered on L2 and can therefore
                              // be relayed on L1.
                              require(
                                  verifyMerkleProof(_xDomainCalldataHash, _withdrawalProof, _nonce, _withdrawalRoot),
                                  "Messenger: invalid withdrawal inclusion proof"
                              );
                          }
                          // relay message
                          xDomainMessageSender = _from;
                          (bool success, ) = _to.call{value: _value}(_message);
                          // reset value to refund gas.
                          xDomainMessageSender = Constants.DEFAULT_XDOMAIN_MESSAGE_SENDER;
                          if (success) {
                              // Mark the withdrawal as finalized so it can't be replayed.
                              finalizedWithdrawals[_xDomainCalldataHash] = true;
                              emit RelayedMessage(_xDomainCalldataHash);
                          } else {
                              emit FailedRelayedMessage(_xDomainCalldataHash);
                          }
                      }
                      /// @inheritdoc IL1CrossDomainMessenger
                      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".
                          address _messageQueue = messageQueue;
                          address _counterpart = counterpart;
                          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(_from, _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);
                          {
                              // avoid stack too deep
                              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;
                          }
                          emit ReplayMessage(_messageNonce, _msgSender(), _to, _value, _nextQueueIndex, _newGasLimit, _message);
                          // 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 IL1CrossDomainMessenger
                      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.
                          address _messageQueue = messageQueue;
                          // 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;
                          emit DropMessage(_messageNonce);
                          // set execution context
                          xDomainMessageSender = Constants.DROP_XDOMAIN_MESSAGE_SENDER;
                          IMessageDropCallback(_from).onDropMessage{value: _value}(_message);
                          // clear execution context
                          xDomainMessageSender = Constants.DEFAULT_XDOMAIN_MESSAGE_SENDER;
                      }
                      /************************
                       * Restricted Functions *
                       ************************/
                      /// @dev Updates the rollup contract address.
                      /// @param _newRollup The address of the new rollup contract.
                      function updateRollup(address _newRollup) external onlyOwner {
                          require(_newRollup != address(0), "rollup address cannot be address(0)");
                          address _oldRollup = rollup;
                          rollup = _newRollup;
                          emit UpdateRollup(_oldRollup, _newRollup);
                      }
                      /// @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 {
                          require(_newMaxReplayTimes > 0, "replay times must be greater than 0");
                          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 {
                          address _messageQueue = messageQueue; // gas saving
                          address _counterpart = counterpart; // gas saving
                          // 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(_msgSender(), _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");
                              }
                          }
                      }
                      function messageNonce() external view override(ICrossDomainMessenger, CrossDomainMessenger) returns (uint256) {
                          return IL1MessageQueue(messageQueue).nextCrossDomainMessageIndex();
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.16;
                  interface IL1MessageQueue {
                      /**********
                       * Errors *
                       **********/
                      /// @dev Thrown when the given address is `address(0)`.
                      error ErrZeroAddress();
                      /**********
                       * 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 EnforcedTxGateway contract.
                      /// @param _oldGateway The address of old EnforcedTxGateway contract.
                      /// @param _newGateway The address of new EnforcedTxGateway contract.
                      event UpdateEnforcedTxGateway(address indexed _oldGateway, address indexed _newGateway);
                      /// @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);
                      /*************************
                       * 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 sender The address of the message sender.
                      /// @param gasLimit Gas limit required to complete the message relay on L2.
                      /// @dev Estimates the fee for a cross-domain message.
                      function estimateCrossDomainMessageFee(address sender, 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 memory _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.24;
                  interface IRollup {
                      /***********
                       * Structs *
                       ***********/
                      /// @param version                  The version of current batch.
                      /// @param parentBatchHeader        The header of parent batch, see the comments of `BatchHeaderV0Codec`.
                      /// @param blockContexts            The block contexts of current batch.
                      /// @param skippedL1MessageBitmap   The bitmap indicates whether each L1 message is skipped or not.
                      /// @param prevStateRoot            The state root of parent batch.
                      /// @param postStateRoot            The state root of current batch.
                      /// @param withdrawalRoot           The withdraw trie root of current batch.
                      struct BatchDataInput {
                          uint8 version;
                          bytes parentBatchHeader;
                          bytes blockContexts;
                          bytes skippedL1MessageBitmap;
                          bytes32 prevStateRoot;
                          bytes32 postStateRoot;
                          bytes32 withdrawalRoot;
                      }
                      /// @param signedSequencers The bitmap of signed sequencers
                      /// @param sequencerSets    The latest 3 sequencer sets
                      /// @param signature        The BLS signature
                      struct BatchSignatureInput {
                          uint256 signedSequencersBitmap;
                          bytes sequencerSets;
                          bytes signature;
                      }
                      /// @param originTimestamp
                      /// @param finalizeTimestamp
                      /// @param blockNumber
                      struct BatchData {
                          uint256 originTimestamp;
                          uint256 finalizeTimestamp;
                          uint256 blockNumber;
                          uint256 signedSequencersBitmap;
                      }
                      /// @dev Structure to store information about a batch challenge.
                      /// @param batchIndex The index of the challenged batch.
                      /// @param challenger The address of the challenger.
                      /// @param challengeDeposit The amount of deposit put up by the challenger.
                      /// @param startTime The timestamp when the challenge started.
                      /// @param challengeSuccess Flag indicating whether the challenge was successful.
                      /// @param finished Flag indicating whether the challenge has been resolved.
                      struct BatchChallenge {
                          uint64 batchIndex;
                          address challenger;
                          uint256 challengeDeposit;
                          uint256 startTime;
                          bool challengeSuccess;
                          bool finished;
                      }
                      /// @param receiver
                      /// @param amount
                      struct BatchChallengeReward {
                          address receiver;
                          uint256 amount;
                      }
                      /***********
                       * Errors *
                       ***********/
                      /// @notice error zero address
                      error ErrZeroAddress();
                      /**********
                       * 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);
                      /// @notice Emitted when owner updates the proofWindow parameter.
                      /// @param oldWindow    The old proofWindow.
                      /// @param newWindow    The new proofWindow.
                      event UpdateProofWindow(uint256 oldWindow, uint256 newWindow);
                      /// @notice Emitted when owner updates the finalizationPeriodSeconds parameter.
                      /// @param oldPeriod    The old finalizationPeriodSeconds.
                      /// @param newPeriod    The new finalizationPeriodSeconds.
                      event UpdateFinalizationPeriodSeconds(uint256 oldPeriod, uint256 newPeriod);
                      /// @notice Emitted when owner updates the status of challenger.
                      /// @param account  The address of account updated.
                      /// @param status   The status of the account updated.
                      event UpdateChallenger(address indexed account, bool status);
                      /// @notice Emitted when the address of rollup verifier is updated.
                      /// @param oldVerifier  The address of old rollup verifier.
                      /// @param newVerifier  The address of new rollup verifier.
                      event UpdateVerifier(address indexed oldVerifier, address indexed newVerifier);
                      /// @notice Emitted when the proof reward percent is updated.
                      /// @param oldPercent  The old proofRewardPercent.
                      /// @param newPercent  The new proofRewardPercent.
                      event UpdateProofRewardPercent(uint256 oldPercent, uint256 newPercent);
                      /// @notice Emit when prove remaining claimed.
                      /// @param receiver  receiver address.
                      /// @param amount    claimed amount.
                      event ProveRemainingClaimed(address receiver, uint256 amount);
                      /// @notice Emitted when the state of Challenge is updated.
                      /// @param batchIndex       The index of the batch.
                      /// @param challenger       The address of challenger.
                      /// @param challengeDeposit The deposit of challenger.
                      event ChallengeState(uint64 indexed batchIndex, address indexed challenger, uint256 challengeDeposit);
                      /// @notice Emitted when the result of Challenge is updated.
                      /// @param batchIndex   The index of the batch.
                      /// @param winner       The address of winner.
                      /// @param res          The result of challenge.
                      event ChallengeRes(uint256 indexed batchIndex, address indexed winner, string indexed res);
                      /// @notice Emitted when the challenger claim the challenge reward.
                      /// @param receiver  receiver address
                      /// @param amount    claimed amount
                      event ChallengeRewardClaim(address indexed receiver, uint256 amount);
                      /*************************
                       * Public View Functions *
                       *************************/
                      /// @notice The latest finalized batch index.
                      function lastFinalizedBatchIndex() external view returns (uint256);
                      /// @notice The latest finalized batch index.
                      function lastCommittedBatchIndex() 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 the committed batch of withdrawalRoot.
                      /// @param withdrawalRoot The withdrawal root.
                      function withdrawalRoots(bytes32 withdrawalRoot) external view returns (bool);
                      /// @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);
                      /// @notice Return the rollup config of finalizationPeriodSeconds.
                      function finalizationPeriodSeconds() external view returns (uint256);
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @notice Commit a batch of transactions on layer 1.
                      ///
                      /// @param batchDataInput       The BatchDataInput struct
                      /// @param batchSignatureInput  The BatchSignatureInput struct
                      function commitBatch(
                          BatchDataInput calldata batchDataInput,
                          BatchSignatureInput calldata batchSignatureInput
                      ) external payable;
                      /// @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 Claim challenge reward
                      /// @param receiver The receiver address
                      function claimReward(address receiver) external;
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.9;
                  interface IMessageDropCallback {
                      function onDropMessage(bytes memory message) external payable;
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity =0.8.24;
                  contract Verify {
                      function verifyMerkleProof(
                          bytes32 leafHash,
                          bytes32[32] calldata smtProof,
                          uint256 index,
                          bytes32 root
                      ) public pure returns (bool) {
                          bytes32 node = leafHash;
                          for (uint256 height = 0; height < 32; height++) {
                              if (((index >> height) & 1) == 1) node = keccak256(abi.encodePacked(smtProof[height], node));
                              else node = keccak256(abi.encodePacked(node, smtProof[height]));
                          }
                          return node == root;
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity =0.8.24;
                  /**
                   * @title Constants
                   * @notice Constants is a library for storing constants. Simple! Don't put everything in here, just
                   *         the stuff used in multiple contracts. Constants that only apply to a single contract
                   *         should be defined in that contract instead.
                   */
                  library Constants {
                      /**
                       * @notice Value used for the L2 sender storage slot in both the MorphPortal and the
                       *         CrossDomainMessenger contracts before an actual sender is set. This value is
                       *         non-zero to reduce the gas cost of message passing transactions.
                       */
                      address internal constant DEFAULT_XDOMAIN_MESSAGE_SENDER = 0x000000000000000000000000000000000000dEaD;
                      /// @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;
                  /**
                   * @title Predeploys
                   * @notice Contains constant addresses for contracts that are pre-deployed to the L2 system.
                   */
                  library Predeploys {
                      /**
                       * @notice Address of the L2_TO_L1_MESSAGE_PASSER predeploy.
                       */
                      address internal constant L2_TO_L1_MESSAGE_PASSER = 0x5300000000000000000000000000000000000001;
                      /**
                       * @notice Address of the L2_GATEWAY_ROUTER predeploy.
                       */
                      address internal constant L2_GATEWAY_ROUTER = 0x5300000000000000000000000000000000000002;
                      /**
                       * @notice Address of the Gov predeploy.
                       */
                      address internal constant GOV = 0x5300000000000000000000000000000000000004;
                      /**
                       * @notice Address of the L2_ETH_GATEWAY predeploy.
                       */
                      address internal constant L2_ETH_GATEWAY = 0x5300000000000000000000000000000000000006;
                      /**
                       * @notice Address of the L2_CROSS_DOMAIN_MESSENGER predeploy.
                       */
                      address internal constant L2_CROSS_DOMAIN_MESSENGER = 0x5300000000000000000000000000000000000007;
                      /**
                       * @notice Address of the L2_STANDARD_ERC20_GATEWAY predeploy.
                       */
                      address internal constant L2_STANDARD_ERC20_GATEWAY = 0x5300000000000000000000000000000000000008;
                      /**
                       * @notice Address of the L2_ERC721_GATEWAY predeploy.
                       */
                      address internal constant L2_ERC721_GATEWAY = 0x5300000000000000000000000000000000000009;
                      /**
                       * @notice Address of the L2_TX_FEE_VAULT predeploy.
                       */
                      address internal constant L2_TX_FEE_VAULT = 0x530000000000000000000000000000000000000a;
                      /**
                       * @notice Address of the PROXY_ADMIN predeploy.
                       */
                      address internal constant PROXY_ADMIN = 0x530000000000000000000000000000000000000b;
                      /**
                       * @notice Address of the L2_ERC1155_GATEWAY predeploy.
                       */
                      address internal constant L2_ERC1155_GATEWAY = 0x530000000000000000000000000000000000000c;
                      /**
                       * @notice Address of the MORPH_STANDARD_ERC20 predeploy.
                       */
                      address internal constant MORPH_STANDARD_ERC20 = 0x530000000000000000000000000000000000000D;
                      /**
                       * @notice Address of the MORPH_STANDARD_ERC20_FACTORY predeploy.
                       */
                      address internal constant MORPH_STANDARD_ERC20_FACTORY = 0x530000000000000000000000000000000000000e;
                      /**
                       * @notice Address of the GAS_PRICE_ORACLE predeploy. Includes fee information
                       *         and helpers for computing the L1 portion of the transaction fee.
                       */
                      address internal constant GAS_PRICE_ORACLE = 0x530000000000000000000000000000000000000f;
                      /**
                       * @notice Address of the L2_WETH_GATEWAY predeploy.
                       */
                      address internal constant L2_WETH_GATEWAY = 0x5300000000000000000000000000000000000010;
                      /**
                       * @notice Address of the L2_WETH predeploy.
                       */
                      address internal constant L2_WETH = 0x5300000000000000000000000000000000000011;
                      /**
                       * @notice Address of the RECORD predeploy.
                       */
                      address internal constant RECORD = 0x5300000000000000000000000000000000000012;
                      /**
                       * @notice Address of the MORPH_TOKEN predeploy.
                       */
                      address internal constant MORPH_TOKEN = 0x5300000000000000000000000000000000000013;
                      /**
                       * @notice Address of the DISTRIBUTE predeploy.
                       */
                      address internal constant DISTRIBUTE = 0x5300000000000000000000000000000000000014;
                      /**
                       * @notice Address of the L2_STAKING predeploy.
                       */
                      address internal constant L2_STAKING = 0x5300000000000000000000000000000000000015;
                      /**
                       * @notice Address of the L2_CUSTOM_ERC20_GATEWAY predeploy.
                       */
                      address internal constant L2_CUSTOM_ERC20_GATEWAY = 0x5300000000000000000000000000000000000016;
                      /**
                       * @notice Address of the SEQUENCER predeploy.
                       */
                      address internal constant SEQUENCER = 0x5300000000000000000000000000000000000017;
                      /**
                       * @notice Address of the L2_REVERSE_ERC20_GATEWAY predeploy.
                       */
                      address internal constant L2_REVERSE_ERC20_GATEWAY = 0x5300000000000000000000000000000000000018;
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity =0.8.24;
                  import {OwnableUpgradeable} from "node_modules/@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol";
                  import {PausableUpgradeable} from "node_modules/@openzeppelin/contracts-upgradeable/security/PausableUpgradeable.sol";
                  import {ReentrancyGuardUpgradeable} from "node_modules/@openzeppelin/contracts-upgradeable/security/ReentrancyGuardUpgradeable.sol";
                  import {Constants} from "./constants/Constants.sol";
                  import {ICrossDomainMessenger} from "./ICrossDomainMessenger.sol";
                  /**
                   * @custom:upgradeable
                   * @title CrossDomainMessenger
                   * @notice CrossDomainMessenger is a base contract that provides the core logic for the L1 and L2
                   *         cross-chain messenger contracts. It's designed to be a universal interface that only
                   *         needs to be extended slightly to provide low-level message passing functionality on each
                   *         chain it's deployed on. Currently only designed for message passing between two paired
                   *         chains and does not support one-to-many interactions.
                   *
                   */
                  abstract contract CrossDomainMessenger is
                      OwnableUpgradeable,
                      PausableUpgradeable,
                      ReentrancyGuardUpgradeable,
                      ICrossDomainMessenger
                  {
                      /**********
                       * 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 indexed _oldFeeVault, address indexed _newFeeVault);
                      /*************
                       * Variables *
                       *************/
                      /**
                       * @notice Address of the sender of the currently executing message on the other chain. If the
                       *         value of this variable is the default value (0x00000000...dead) then no message is
                       *         currently being executed. Use the xDomainMessageSender getter which will throw an
                       *         error if this is the case.
                       */
                      address public override xDomainMessageSender;
                      /**
                       * @notice Address of the paired CrossDomainMessenger contract on the other chain.
                       */
                      address public counterpart;
                      /// @notice The address of fee vault, collecting cross domain messaging fee.
                      address public feeVault;
                      /// @dev The storage slots for future usage.
                      uint256[46] private __gap;
                      /**********************
                       * Function Modifiers *
                       **********************/
                      modifier notInExecution() {
                          require(xDomainMessageSender == Constants.DEFAULT_XDOMAIN_MESSAGE_SENDER, "Message is already in execution");
                          _;
                      }
                      /***************
                       * Constructor *
                       ***************/
                      /* solhint-disable */
                      function __Messenger_init(address _counterpart, address _feeVault) internal onlyInitializing {
                          OwnableUpgradeable.__Ownable_init();
                          PausableUpgradeable.__Pausable_init();
                          ReentrancyGuardUpgradeable.__ReentrancyGuard_init();
                          // initialize to a nonzero value
                          xDomainMessageSender = Constants.DEFAULT_XDOMAIN_MESSAGE_SENDER;
                          counterpart = _counterpart;
                          if (_feeVault != address(0)) {
                              feeVault = _feeVault;
                          }
                      }
                      /* solhint-enable */
                      // 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 {
                          require(_newFeeVault != address(0), "feeVault cannot be address(0)");
                          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), "Messenger: Forbid to call self");
                      }
                      function messageNonce() external view virtual returns (uint256);
                  }
                  // SPDX-License-Identifier: MIT
                  pragma solidity ^0.8.9;
                  interface ICrossDomainMessenger {
                      /***********
                       * Errors *
                       ***********/
                      error ErrZeroAddress();
                      /**********
                       * 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);
                      /*************************
                       * Public View Functions *
                       *************************/
                      /// @notice Return the sender of a cross domain message.
                      function xDomainMessageSender() external view returns (address);
                      /// @notice Return the nonce of a cross domain message.
                      function messageNonce() external view returns (uint256);
                      /*****************************
                       * Public Mutating Functions *
                       *****************************/
                      /// @notice Send cross chain message from L1 to L2 or L2 to L1.
                      /// @dev EOA addresses and contracts that have not implemented `onDropMessage`
                      /// cannot execute the `dropMessage` operation.
                      /// Please proceed with caution to control risk.
                      /// @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.
                      /// @dev EOA addresses and contracts that have not implemented `onDropMessage`
                      /// cannot execute the `dropMessage` operation.
                      /// Please proceed with caution to control risk.
                      /// @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 {Initializable} from "../proxy/utils/Initializable.sol";
                  /**
                   * @dev Contract module which provides a basic access control mechanism, where
                   * there is an account (an owner) that can be granted exclusive access to
                   * specific functions.
                   *
                   * By default, the owner account will be the one that deploys the contract. This
                   * can later be changed with {transferOwnership}.
                   *
                   * This module is used through inheritance. It will make available the modifier
                   * `onlyOwner`, which can be applied to your functions to restrict their use to
                   * the owner.
                   */
                  abstract contract OwnableUpgradeable is Initializable, ContextUpgradeable {
                      address private _owner;
                      event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
                      /**
                       * @dev Initializes the contract setting the deployer as the initial owner.
                       */
                      function __Ownable_init() internal onlyInitializing {
                          __Ownable_init_unchained();
                      }
                      function __Ownable_init_unchained() internal onlyInitializing {
                          _transferOwnership(_msgSender());
                      }
                      /**
                       * @dev Throws if called by any account other than the owner.
                       */
                      modifier onlyOwner() {
                          _checkOwner();
                          _;
                      }
                      /**
                       * @dev Returns the address of the current owner.
                       */
                      function owner() public view virtual returns (address) {
                          return _owner;
                      }
                      /**
                       * @dev Throws if the sender is not the owner.
                       */
                      function _checkOwner() internal view virtual {
                          require(owner() == _msgSender(), "Ownable: caller is not the owner");
                      }
                      /**
                       * @dev Leaves the contract without owner. It will not be possible to call
                       * `onlyOwner` functions. Can only be called by the current owner.
                       *
                       * NOTE: Renouncing ownership will leave the contract without an owner,
                       * thereby disabling any functionality that is only available to the owner.
                       */
                      function renounceOwnership() public virtual onlyOwner {
                          _transferOwnership(address(0));
                      }
                      /**
                       * @dev Transfers ownership of the contract to a new account (`newOwner`).
                       * Can only be called by the current owner.
                       */
                      function transferOwnership(address newOwner) public virtual onlyOwner {
                          require(newOwner != address(0), "Ownable: new owner is the zero address");
                          _transferOwnership(newOwner);
                      }
                      /**
                       * @dev Transfers ownership of the contract to a new account (`newOwner`).
                       * Internal function without access restriction.
                       */
                      function _transferOwnership(address newOwner) internal virtual {
                          address oldOwner = _owner;
                          _owner = newOwner;
                          emit OwnershipTransferred(oldOwner, newOwner);
                      }
                      /**
                       * @dev This empty reserved space is put in place to allow future versions to add new
                       * variables without shifting down storage in the inheritance chain.
                       * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
                       */
                      uint256[49] private __gap;
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.9.0) (proxy/utils/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 {Initializable} from "../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 {Initializable} from "../proxy/utils/Initializable.sol";
                  /**
                   * @dev Contract module that helps prevent reentrant calls to a function.
                   *
                   * Inheriting from `ReentrancyGuard` will make the {nonReentrant} modifier
                   * available, which can be applied to functions to make sure there are no nested
                   * (reentrant) calls to them.
                   *
                   * Note that because there is a single `nonReentrant` guard, functions marked as
                   * `nonReentrant` may not call one another. This can be worked around by making
                   * those functions `private`, and then adding `external` `nonReentrant` entry
                   * points to them.
                   *
                   * TIP: If you would like to learn more about reentrancy and alternative ways
                   * to protect against it, check out our blog post
                   * https://blog.openzeppelin.com/reentrancy-after-istanbul/[Reentrancy After Istanbul].
                   */
                  abstract contract ReentrancyGuardUpgradeable is Initializable {
                      // Booleans are more expensive than uint256 or any type that takes up a full
                      // word because each write operation emits an extra SLOAD to first read the
                      // slot's contents, replace the bits taken up by the boolean, and then write
                      // back. This is the compiler's defense against contract upgrades and
                      // pointer aliasing, and it cannot be disabled.
                      // The values being non-zero value makes deployment a bit more expensive,
                      // but in exchange the refund on every call to nonReentrant will be lower in
                      // amount. Since refunds are capped to a percentage of the total
                      // transaction's gas, it is best to keep them low in cases like this one, to
                      // increase the likelihood of the full refund coming into effect.
                      uint256 private constant _NOT_ENTERED = 1;
                      uint256 private constant _ENTERED = 2;
                      uint256 private _status;
                      function __ReentrancyGuard_init() internal onlyInitializing {
                          __ReentrancyGuard_init_unchained();
                      }
                      function __ReentrancyGuard_init_unchained() internal onlyInitializing {
                          _status = _NOT_ENTERED;
                      }
                      /**
                       * @dev Prevents a contract from calling itself, directly or indirectly.
                       * Calling a `nonReentrant` function from another `nonReentrant`
                       * function is not supported. It is possible to prevent this from happening
                       * by making the `nonReentrant` function external, and making it call a
                       * `private` function that does the actual work.
                       */
                      modifier nonReentrant() {
                          _nonReentrantBefore();
                          _;
                          _nonReentrantAfter();
                      }
                      function _nonReentrantBefore() private {
                          // On the first call to nonReentrant, _status will be _NOT_ENTERED
                          require(_status != _ENTERED, "ReentrancyGuard: reentrant call");
                          // Any calls to nonReentrant after this point will fail
                          _status = _ENTERED;
                      }
                      function _nonReentrantAfter() private {
                          // By storing the original value once again, a refund is triggered (see
                          // https://eips.ethereum.org/EIPS/eip-2200)
                          _status = _NOT_ENTERED;
                      }
                      /**
                       * @dev Returns true if the reentrancy guard is currently set to "entered", which indicates there is a
                       * `nonReentrant` function in the call stack.
                       */
                      function _reentrancyGuardEntered() internal view returns (bool) {
                          return _status == _ENTERED;
                      }
                      /**
                       * @dev This empty reserved space is put in place to allow future versions to add new
                       * variables without shifting down storage in the inheritance chain.
                       * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
                       */
                      uint256[49] private __gap;
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.9.0) (utils/Address.sol)
                  pragma solidity ^0.8.1;
                  /**
                   * @dev Collection of functions related to the address type
                   */
                  library AddressUpgradeable {
                      /**
                       * @dev Returns true if `account` is a contract.
                       *
                       * [IMPORTANT]
                       * ====
                       * It is unsafe to assume that an address for which this function returns
                       * false is an externally-owned account (EOA) and not a contract.
                       *
                       * Among others, `isContract` will return false for the following
                       * types of addresses:
                       *
                       *  - an externally-owned account
                       *  - a contract in construction
                       *  - an address where a contract will be created
                       *  - an address where a contract lived, but was destroyed
                       *
                       * Furthermore, `isContract` will also return true if the target contract within
                       * the same transaction is already scheduled for destruction by `SELFDESTRUCT`,
                       * which only has an effect at the end of a transaction.
                       * ====
                       *
                       * [IMPORTANT]
                       * ====
                       * You shouldn't rely on `isContract` to protect against flash loan attacks!
                       *
                       * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets
                       * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract
                       * constructor.
                       * ====
                       */
                      function isContract(address account) internal view returns (bool) {
                          // This method relies on extcodesize/address.code.length, which returns 0
                          // for contracts in construction, since the code is only stored at the end
                          // of the constructor execution.
                          return account.code.length > 0;
                      }
                      /**
                       * @dev Replacement for Solidity's `transfer`: sends `amount` wei to
                       * `recipient`, forwarding all available gas and reverting on errors.
                       *
                       * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
                       * of certain opcodes, possibly making contracts go over the 2300 gas limit
                       * imposed by `transfer`, making them unable to receive funds via
                       * `transfer`. {sendValue} removes this limitation.
                       *
                       * https://consensys.net/diligence/blog/2019/09/stop-using-soliditys-transfer-now/[Learn more].
                       *
                       * IMPORTANT: because control is transferred to `recipient`, care must be
                       * taken to not create reentrancy vulnerabilities. Consider using
                       * {ReentrancyGuard} or the
                       * https://solidity.readthedocs.io/en/v0.8.0/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
                       */
                      function sendValue(address payable recipient, uint256 amount) internal {
                          require(address(this).balance >= amount, "Address: insufficient balance");
                          (bool success, ) = recipient.call{value: amount}("");
                          require(success, "Address: unable to send value, recipient may have reverted");
                      }
                      /**
                       * @dev Performs a Solidity function call using a low level `call`. A
                       * plain `call` is an unsafe replacement for a function call: use this
                       * function instead.
                       *
                       * If `target` reverts with a revert reason, it is bubbled up by this
                       * function (like regular Solidity function calls).
                       *
                       * Returns the raw returned data. To convert to the expected return value,
                       * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
                       *
                       * Requirements:
                       *
                       * - `target` must be a contract.
                       * - calling `target` with `data` must not revert.
                       *
                       * _Available since v3.1._
                       */
                      function functionCall(address target, bytes memory data) internal returns (bytes memory) {
                          return functionCallWithValue(target, data, 0, "Address: low-level call failed");
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
                       * `errorMessage` as a fallback revert reason when `target` reverts.
                       *
                       * _Available since v3.1._
                       */
                      function functionCall(
                          address target,
                          bytes memory data,
                          string memory errorMessage
                      ) internal returns (bytes memory) {
                          return functionCallWithValue(target, data, 0, errorMessage);
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                       * but also transferring `value` wei to `target`.
                       *
                       * Requirements:
                       *
                       * - the calling contract must have an ETH balance of at least `value`.
                       * - the called Solidity function must be `payable`.
                       *
                       * _Available since v3.1._
                       */
                      function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
                          return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
                      }
                      /**
                       * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
                       * with `errorMessage` as a fallback revert reason when `target` reverts.
                       *
                       * _Available since v3.1._
                       */
                      function functionCallWithValue(
                          address target,
                          bytes memory data,
                          uint256 value,
                          string memory errorMessage
                      ) internal returns (bytes memory) {
                          require(address(this).balance >= value, "Address: insufficient balance for call");
                          (bool success, bytes memory returndata) = target.call{value: value}(data);
                          return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                       * but performing a static call.
                       *
                       * _Available since v3.3._
                       */
                      function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
                          return functionStaticCall(target, data, "Address: low-level static call failed");
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                       * but performing a static call.
                       *
                       * _Available since v3.3._
                       */
                      function functionStaticCall(
                          address target,
                          bytes memory data,
                          string memory errorMessage
                      ) internal view returns (bytes memory) {
                          (bool success, bytes memory returndata) = target.staticcall(data);
                          return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
                       * but performing a delegate call.
                       *
                       * _Available since v3.4._
                       */
                      function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
                          return functionDelegateCall(target, data, "Address: low-level delegate call failed");
                      }
                      /**
                       * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
                       * but performing a delegate call.
                       *
                       * _Available since v3.4._
                       */
                      function functionDelegateCall(
                          address target,
                          bytes memory data,
                          string memory errorMessage
                      ) internal returns (bytes memory) {
                          (bool success, bytes memory returndata) = target.delegatecall(data);
                          return verifyCallResultFromTarget(target, success, returndata, errorMessage);
                      }
                      /**
                       * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling
                       * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract.
                       *
                       * _Available since v4.8._
                       */
                      function verifyCallResultFromTarget(
                          address target,
                          bool success,
                          bytes memory returndata,
                          string memory errorMessage
                      ) internal view returns (bytes memory) {
                          if (success) {
                              if (returndata.length == 0) {
                                  // only check isContract if the call was successful and the return data is empty
                                  // otherwise we already know that it was a contract
                                  require(isContract(target), "Address: call to non-contract");
                              }
                              return returndata;
                          } else {
                              _revert(returndata, errorMessage);
                          }
                      }
                      /**
                       * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the
                       * revert reason or using the provided one.
                       *
                       * _Available since v4.3._
                       */
                      function verifyCallResult(
                          bool success,
                          bytes memory returndata,
                          string memory errorMessage
                      ) internal pure returns (bytes memory) {
                          if (success) {
                              return returndata;
                          } else {
                              _revert(returndata, errorMessage);
                          }
                      }
                      function _revert(bytes memory returndata, string memory errorMessage) private pure {
                          // Look for revert reason and bubble it up if present
                          if (returndata.length > 0) {
                              // The easiest way to bubble the revert reason is using memory via assembly
                              /// @solidity memory-safe-assembly
                              assembly {
                                  let returndata_size := mload(returndata)
                                  revert(add(32, returndata), returndata_size)
                              }
                          } else {
                              revert(errorMessage);
                          }
                      }
                  }
                  // SPDX-License-Identifier: MIT
                  // OpenZeppelin Contracts (last updated v4.9.4) (utils/Context.sol)
                  pragma solidity ^0.8.0;
                  import {Initializable} from "../proxy/utils/Initializable.sol";
                  /**
                   * @dev Provides information about the current execution context, including the
                   * sender of the transaction and its data. While these are generally available
                   * via msg.sender and msg.data, they should not be accessed in such a direct
                   * manner, since when dealing with meta-transactions the account sending and
                   * paying for execution may not be the actual sender (as far as an application
                   * is concerned).
                   *
                   * This contract is only required for intermediate, library-like contracts.
                   */
                  abstract contract ContextUpgradeable is Initializable {
                      function __Context_init() internal onlyInitializing {
                      }
                      function __Context_init_unchained() internal onlyInitializing {
                      }
                      function _msgSender() internal view virtual returns (address) {
                          return msg.sender;
                      }
                      function _msgData() internal view virtual returns (bytes calldata) {
                          return msg.data;
                      }
                      function _contextSuffixLength() internal view virtual returns (uint256) {
                          return 0;
                      }
                      /**
                       * @dev This empty reserved space is put in place to allow future versions to add new
                       * variables without shifting down storage in the inheritance chain.
                       * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
                       */
                      uint256[50] private __gap;
                  }