draft-ERC4337Utils.sol 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. // SPDX-License-Identifier: MIT
  2. // OpenZeppelin Contracts (last updated v5.3.0) (account/utils/draft-ERC4337Utils.sol)
  3. pragma solidity ^0.8.20;
  4. import {IEntryPoint, PackedUserOperation} from "../../interfaces/draft-IERC4337.sol";
  5. import {Math} from "../../utils/math/Math.sol";
  6. import {Calldata} from "../../utils/Calldata.sol";
  7. import {Packing} from "../../utils/Packing.sol";
  8. /// @dev This is available on all entrypoint since v0.4.0, but is not formally part of the ERC.
  9. interface IEntryPointExtra {
  10. function getUserOpHash(PackedUserOperation calldata userOp) external view returns (bytes32);
  11. }
  12. /**
  13. * @dev Library with common ERC-4337 utility functions.
  14. *
  15. * See https://eips.ethereum.org/EIPS/eip-4337[ERC-4337].
  16. */
  17. library ERC4337Utils {
  18. using Packing for *;
  19. /// @dev Address of the entrypoint v0.7.0
  20. IEntryPoint internal constant ENTRYPOINT_V07 = IEntryPoint(0x0000000071727De22E5E9d8BAf0edAc6f37da032);
  21. /// @dev Address of the entrypoint v0.8.0
  22. IEntryPoint internal constant ENTRYPOINT_V08 = IEntryPoint(0x4337084D9E255Ff0702461CF8895CE9E3b5Ff108);
  23. /// @dev For simulation purposes, validateUserOp (and validatePaymasterUserOp) return this value on success.
  24. uint256 internal constant SIG_VALIDATION_SUCCESS = 0;
  25. /// @dev For simulation purposes, validateUserOp (and validatePaymasterUserOp) must return this value in case of signature failure, instead of revert.
  26. uint256 internal constant SIG_VALIDATION_FAILED = 1;
  27. /// @dev Parses the validation data into its components. See {packValidationData}.
  28. function parseValidationData(
  29. uint256 validationData
  30. ) internal pure returns (address aggregator, uint48 validAfter, uint48 validUntil) {
  31. validAfter = uint48(bytes32(validationData).extract_32_6(0));
  32. validUntil = uint48(bytes32(validationData).extract_32_6(6));
  33. aggregator = address(bytes32(validationData).extract_32_20(12));
  34. if (validUntil == 0) validUntil = type(uint48).max;
  35. }
  36. /// @dev Packs the validation data into a single uint256. See {parseValidationData}.
  37. function packValidationData(
  38. address aggregator,
  39. uint48 validAfter,
  40. uint48 validUntil
  41. ) internal pure returns (uint256) {
  42. return uint256(bytes6(validAfter).pack_6_6(bytes6(validUntil)).pack_12_20(bytes20(aggregator)));
  43. }
  44. /// @dev Same as {packValidationData}, but with a boolean signature success flag.
  45. function packValidationData(bool sigSuccess, uint48 validAfter, uint48 validUntil) internal pure returns (uint256) {
  46. return
  47. packValidationData(
  48. address(uint160(Math.ternary(sigSuccess, SIG_VALIDATION_SUCCESS, SIG_VALIDATION_FAILED))),
  49. validAfter,
  50. validUntil
  51. );
  52. }
  53. /**
  54. * @dev Combines two validation data into a single one.
  55. *
  56. * The `aggregator` is set to {SIG_VALIDATION_SUCCESS} if both are successful, while
  57. * the `validAfter` is the maximum and the `validUntil` is the minimum of both.
  58. */
  59. function combineValidationData(uint256 validationData1, uint256 validationData2) internal pure returns (uint256) {
  60. (address aggregator1, uint48 validAfter1, uint48 validUntil1) = parseValidationData(validationData1);
  61. (address aggregator2, uint48 validAfter2, uint48 validUntil2) = parseValidationData(validationData2);
  62. bool success = aggregator1 == address(uint160(SIG_VALIDATION_SUCCESS)) &&
  63. aggregator2 == address(uint160(SIG_VALIDATION_SUCCESS));
  64. uint48 validAfter = uint48(Math.max(validAfter1, validAfter2));
  65. uint48 validUntil = uint48(Math.min(validUntil1, validUntil2));
  66. return packValidationData(success, validAfter, validUntil);
  67. }
  68. /// @dev Returns the aggregator of the `validationData` and whether it is out of time range.
  69. function getValidationData(uint256 validationData) internal view returns (address aggregator, bool outOfTimeRange) {
  70. (address aggregator_, uint48 validAfter, uint48 validUntil) = parseValidationData(validationData);
  71. return (aggregator_, block.timestamp < validAfter || validUntil < block.timestamp);
  72. }
  73. /// @dev Get the hash of a user operation for a given entrypoint
  74. function hash(PackedUserOperation calldata self, address entrypoint) internal view returns (bytes32) {
  75. // NOTE: getUserOpHash is available since v0.4.0
  76. //
  77. // Prior to v0.8.0, this was easy to replicate for any entrypoint and chainId. Since v0.8.0 of the
  78. // entrypoint, this depends on the Entrypoint's domain separator, which cannot be hardcoded and is complex
  79. // to recompute. Domain separator could be fetch using the `getDomainSeparatorV4` getter, or recomputed from
  80. // the ERC-5267 getter, but both operation would require doing a view call to the entrypoint. Overall it feels
  81. // simpler and less error prone to get that functionality from the entrypoint directly.
  82. return IEntryPointExtra(entrypoint).getUserOpHash(self);
  83. }
  84. /// @dev Returns `factory` from the {PackedUserOperation}, or address(0) if the initCode is empty or not properly formatted.
  85. function factory(PackedUserOperation calldata self) internal pure returns (address) {
  86. return self.initCode.length < 20 ? address(0) : address(bytes20(self.initCode[0:20]));
  87. }
  88. /// @dev Returns `factoryData` from the {PackedUserOperation}, or empty bytes if the initCode is empty or not properly formatted.
  89. function factoryData(PackedUserOperation calldata self) internal pure returns (bytes calldata) {
  90. return self.initCode.length < 20 ? Calldata.emptyBytes() : self.initCode[20:];
  91. }
  92. /// @dev Returns `verificationGasLimit` from the {PackedUserOperation}.
  93. function verificationGasLimit(PackedUserOperation calldata self) internal pure returns (uint256) {
  94. return uint128(self.accountGasLimits.extract_32_16(0));
  95. }
  96. /// @dev Returns `callGasLimit` from the {PackedUserOperation}.
  97. function callGasLimit(PackedUserOperation calldata self) internal pure returns (uint256) {
  98. return uint128(self.accountGasLimits.extract_32_16(16));
  99. }
  100. /// @dev Returns the first section of `gasFees` from the {PackedUserOperation}.
  101. function maxPriorityFeePerGas(PackedUserOperation calldata self) internal pure returns (uint256) {
  102. return uint128(self.gasFees.extract_32_16(0));
  103. }
  104. /// @dev Returns the second section of `gasFees` from the {PackedUserOperation}.
  105. function maxFeePerGas(PackedUserOperation calldata self) internal pure returns (uint256) {
  106. return uint128(self.gasFees.extract_32_16(16));
  107. }
  108. /// @dev Returns the total gas price for the {PackedUserOperation} (ie. `maxFeePerGas` or `maxPriorityFeePerGas + basefee`).
  109. function gasPrice(PackedUserOperation calldata self) internal view returns (uint256) {
  110. unchecked {
  111. // Following values are "per gas"
  112. uint256 maxPriorityFee = maxPriorityFeePerGas(self);
  113. uint256 maxFee = maxFeePerGas(self);
  114. return Math.min(maxFee, maxPriorityFee + block.basefee);
  115. }
  116. }
  117. /// @dev Returns the first section of `paymasterAndData` from the {PackedUserOperation}.
  118. function paymaster(PackedUserOperation calldata self) internal pure returns (address) {
  119. return self.paymasterAndData.length < 52 ? address(0) : address(bytes20(self.paymasterAndData[0:20]));
  120. }
  121. /// @dev Returns the second section of `paymasterAndData` from the {PackedUserOperation}.
  122. function paymasterVerificationGasLimit(PackedUserOperation calldata self) internal pure returns (uint256) {
  123. return self.paymasterAndData.length < 52 ? 0 : uint128(bytes16(self.paymasterAndData[20:36]));
  124. }
  125. /// @dev Returns the third section of `paymasterAndData` from the {PackedUserOperation}.
  126. function paymasterPostOpGasLimit(PackedUserOperation calldata self) internal pure returns (uint256) {
  127. return self.paymasterAndData.length < 52 ? 0 : uint128(bytes16(self.paymasterAndData[36:52]));
  128. }
  129. /// @dev Returns the fourth section of `paymasterAndData` from the {PackedUserOperation}.
  130. function paymasterData(PackedUserOperation calldata self) internal pure returns (bytes calldata) {
  131. return self.paymasterAndData.length < 52 ? Calldata.emptyBytes() : self.paymasterAndData[52:];
  132. }
  133. }