From f0c6d1edbd9bbfc233a2f2b6dad767b142d22ade Mon Sep 17 00:00:00 2001 From: Disco <131301107+0xDiscotech@users.noreply.github.com> Date: Tue, 15 Oct 2024 18:17:52 -0300 Subject: [PATCH] feat: support permit2 on optimism superchain erc20 and upgrade solady's erc20 implementation (#97) --- Co-Authored-by: AgusDuha <81362284+agusduha@users.noreply.github.com> --- .gitmodules | 3 + packages/contracts-bedrock/foundry.toml | 1 + .../contracts-bedrock/lib/solady-v0.0.245 | 1 + packages/contracts-bedrock/semver-lock.json | 6 +- .../abi/OptimismSuperchainERC20.json | 5 ++ .../src/L2/OptimismSuperchainERC20.sol | 9 ++- .../src/L2/SuperchainERC20.sol | 6 +- .../src/vendor/interfaces/IERC20Solady.sol | 3 + .../test/L2/OptimismSuperchainERC20.t.sol | 72 +++++++++++++++++-- .../test/L2/SuperchainERC20.t.sol | 6 +- 10 files changed, 96 insertions(+), 16 deletions(-) create mode 160000 packages/contracts-bedrock/lib/solady-v0.0.245 diff --git a/.gitmodules b/.gitmodules index 21ecaedbb77a..5422163a9599 100644 --- a/.gitmodules +++ b/.gitmodules @@ -29,3 +29,6 @@ [submodule "packages/contracts-bedrock/lib/openzeppelin-contracts-v5"] path = packages/contracts-bedrock/lib/openzeppelin-contracts-v5 url = https://github.com/OpenZeppelin/openzeppelin-contracts +[submodule "packages/contracts-bedrock/lib/solady-v0.0.245"] + path = packages/contracts-bedrock/lib/solady-v0.0.245 + url = https://github.com/vectorized/solady diff --git a/packages/contracts-bedrock/foundry.toml b/packages/contracts-bedrock/foundry.toml index f9e44d6a5b63..f49a757d46b1 100644 --- a/packages/contracts-bedrock/foundry.toml +++ b/packages/contracts-bedrock/foundry.toml @@ -17,6 +17,7 @@ remappings = [ '@rari-capital/solmate/=lib/solmate', '@lib-keccak/=lib/lib-keccak/contracts/lib', '@solady/=lib/solady/src', + '@solady-v0.0.245/=lib/solady-v0.0.245/src', 'forge-std/=lib/forge-std/src', 'ds-test/=lib/forge-std/lib/ds-test/src', 'safe-contracts/=lib/safe-contracts/contracts', diff --git a/packages/contracts-bedrock/lib/solady-v0.0.245 b/packages/contracts-bedrock/lib/solady-v0.0.245 new file mode 160000 index 000000000000..e0ef35adb0cc --- /dev/null +++ b/packages/contracts-bedrock/lib/solady-v0.0.245 @@ -0,0 +1 @@ +Subproject commit e0ef35adb0ccd1032794731a995cb599bba7b537 diff --git a/packages/contracts-bedrock/semver-lock.json b/packages/contracts-bedrock/semver-lock.json index 9488435c1ca3..7ce8720a5483 100644 --- a/packages/contracts-bedrock/semver-lock.json +++ b/packages/contracts-bedrock/semver-lock.json @@ -108,8 +108,8 @@ "sourceCodeHash": "0xfea53344596d735eff3be945ed1300dc75a6f8b7b2c02c0043af5b0036f5f239" }, "src/L2/OptimismSuperchainERC20.sol": { - "initCodeHash": "0xd5c84e45746fd741d541a917ddc1cc0c7043c6b21d5c18040d4bc999d6a7b2db", - "sourceCodeHash": "0xf32130f0b46333daba062c50ff6dcfadce1f177ff753bed2374d499ea9c2d98a" + "initCodeHash": "0x4e25579079d73c93f1d494e1976334b77fc4ec181c67f376d8e2613c7b207f52", + "sourceCodeHash": "0xe41cf3b005f1ea007fc1b5f69f630be5f6ef12d6e5e94a50e3160b0ebe0a1613" }, "src/L2/OptimismSuperchainERC20Beacon.sol": { "initCodeHash": "0x99ce8095b23c124850d866cbc144fee6cee05dbc6bb5d83acadfe00b90cf42c7", @@ -125,7 +125,7 @@ }, "src/L2/SuperchainERC20.sol": { "initCodeHash": "0xc5d2460186f7233c927e7db2dcc703c0e500b653ca82273b7bfad8045d85a470", - "sourceCodeHash": "0x75d061633a141af11a19b86e599a1725dfae8d245dcddfb6bb244a50d5e53f96" + "sourceCodeHash": "0x6a384ccfb6f2f7316c1b33873a1630b5179e52475951d31771656e06d2b11519" }, "src/L2/SuperchainTokenBridge.sol": { "initCodeHash": "0x07fc1d495928d9c13bd945a049d17e1d105d01c2082a7719e5d18cbc0e1c7d9e", diff --git a/packages/contracts-bedrock/snapshots/abi/OptimismSuperchainERC20.json b/packages/contracts-bedrock/snapshots/abi/OptimismSuperchainERC20.json index f1b7f83e3b53..d6ad63fad9c3 100644 --- a/packages/contracts-bedrock/snapshots/abi/OptimismSuperchainERC20.json +++ b/packages/contracts-bedrock/snapshots/abi/OptimismSuperchainERC20.json @@ -570,6 +570,11 @@ "name": "NotInitializing", "type": "error" }, + { + "inputs": [], + "name": "Permit2AllowanceIsFixedAtInfinity", + "type": "error" + }, { "inputs": [], "name": "PermitExpired", diff --git a/packages/contracts-bedrock/src/L2/OptimismSuperchainERC20.sol b/packages/contracts-bedrock/src/L2/OptimismSuperchainERC20.sol index 6e8ef9057325..a097bb736c84 100644 --- a/packages/contracts-bedrock/src/L2/OptimismSuperchainERC20.sol +++ b/packages/contracts-bedrock/src/L2/OptimismSuperchainERC20.sol @@ -59,8 +59,8 @@ contract OptimismSuperchainERC20 is SuperchainERC20, Initializable, ERC165 { } /// @notice Semantic version. - /// @custom:semver 1.0.0-beta.6 - string public constant override version = "1.0.0-beta.6"; + /// @custom:semver 1.0.0-beta.7 + string public constant override version = "1.0.0-beta.7"; /// @notice Constructs the OptimismSuperchainERC20 contract. constructor() { @@ -141,4 +141,9 @@ contract OptimismSuperchainERC20 is SuperchainERC20, Initializable, ERC165 { function supportsInterface(bytes4 _interfaceId) public view virtual override returns (bool) { return _interfaceId == type(IOptimismSuperchainERC20).interfaceId || super.supportsInterface(_interfaceId); } + + /// @notice Sets Permit2 contract's allowance at infinity. + function _givePermit2InfiniteAllowance() internal view virtual override returns (bool) { + return true; + } } diff --git a/packages/contracts-bedrock/src/L2/SuperchainERC20.sol b/packages/contracts-bedrock/src/L2/SuperchainERC20.sol index 9ead2645828b..d723ed8d992b 100644 --- a/packages/contracts-bedrock/src/L2/SuperchainERC20.sol +++ b/packages/contracts-bedrock/src/L2/SuperchainERC20.sol @@ -4,7 +4,7 @@ pragma solidity 0.8.25; import { ICrosschainERC20 } from "src/L2/interfaces/ICrosschainERC20.sol"; import { ISemver } from "src/universal/interfaces/ISemver.sol"; import { Predeploys } from "src/libraries/Predeploys.sol"; -import { ERC20 } from "@solady/tokens/ERC20.sol"; +import { ERC20 } from "@solady-v0.0.245/tokens/ERC20.sol"; import { Unauthorized } from "src/libraries/errors/CommonErrors.sol"; /// @title SuperchainERC20 @@ -19,9 +19,9 @@ abstract contract SuperchainERC20 is ERC20, ICrosschainERC20, ISemver { } /// @notice Semantic version. - /// @custom:semver 1.0.0-beta.1 + /// @custom:semver 1.0.0-beta.2 function version() external view virtual returns (string memory) { - return "1.0.0-beta.1"; + return "1.0.0-beta.2"; } /// @notice Allows the SuperchainTokenBridge to mint tokens. diff --git a/packages/contracts-bedrock/src/vendor/interfaces/IERC20Solady.sol b/packages/contracts-bedrock/src/vendor/interfaces/IERC20Solady.sol index 1e696ad23ac3..b05b906eec97 100644 --- a/packages/contracts-bedrock/src/vendor/interfaces/IERC20Solady.sol +++ b/packages/contracts-bedrock/src/vendor/interfaces/IERC20Solady.sol @@ -23,6 +23,9 @@ interface IERC20Solady { /// @dev The permit has expired. error PermitExpired(); + /// @dev The allowance of Permit2 is fixed at infinity. + error Permit2AllowanceIsFixedAtInfinity(); + /// @dev Emitted when `amount` tokens is transferred from `from` to `to`. event Transfer(address indexed from, address indexed to, uint256 amount); diff --git a/packages/contracts-bedrock/test/L2/OptimismSuperchainERC20.t.sol b/packages/contracts-bedrock/test/L2/OptimismSuperchainERC20.t.sol index 87d5cbb74b23..9f6d73d8cd81 100644 --- a/packages/contracts-bedrock/test/L2/OptimismSuperchainERC20.t.sol +++ b/packages/contracts-bedrock/test/L2/OptimismSuperchainERC20.t.sol @@ -7,12 +7,14 @@ import { EIP1967Helper } from "test/mocks/EIP1967Helper.sol"; // Libraries import { Predeploys } from "src/libraries/Predeploys.sol"; -import { IERC20 } from "@openzeppelin/contracts-v5/token/ERC20/IERC20.sol"; +import { IERC20Solady } from "src/vendor/interfaces/IERC20Solady.sol"; + import { Initializable } from "@openzeppelin/contracts-v5/proxy/utils/Initializable.sol"; import { IERC165 } from "@openzeppelin/contracts-v5/utils/introspection/IERC165.sol"; import { IBeacon } from "@openzeppelin/contracts-v5/proxy/beacon/IBeacon.sol"; import { BeaconProxy } from "@openzeppelin/contracts-v5/proxy/beacon/BeaconProxy.sol"; import { Unauthorized } from "src/libraries/errors/CommonErrors.sol"; +import { Preinstalls } from "src/libraries/Preinstalls.sol"; // Target contract import { OptimismSuperchainERC20 } from "src/L2/OptimismSuperchainERC20.sol"; @@ -142,12 +144,12 @@ contract OptimismSuperchainERC20Test is Test { vm.assume(_to != ZERO_ADDRESS); // Get the total supply and balance of `_to` before the mint to compare later on the assertions - uint256 _totalSupplyBefore = IERC20(address(optimismSuperchainERC20)).totalSupply(); - uint256 _toBalanceBefore = IERC20(address(optimismSuperchainERC20)).balanceOf(_to); + uint256 _totalSupplyBefore = IERC20Solady(address(optimismSuperchainERC20)).totalSupply(); + uint256 _toBalanceBefore = IERC20Solady(address(optimismSuperchainERC20)).balanceOf(_to); // Look for the emit of the `Transfer` event vm.expectEmit(address(optimismSuperchainERC20)); - emit IERC20.Transfer(ZERO_ADDRESS, _to, _amount); + emit IERC20Solady.Transfer(ZERO_ADDRESS, _to, _amount); // Look for the emit of the `Mint` event vm.expectEmit(address(optimismSuperchainERC20)); @@ -200,7 +202,7 @@ contract OptimismSuperchainERC20Test is Test { // Look for the emit of the `Transfer` event vm.expectEmit(address(optimismSuperchainERC20)); - emit IERC20.Transfer(_from, ZERO_ADDRESS, _amount); + emit IERC20Solady.Transfer(_from, ZERO_ADDRESS, _amount); // Look for the emit of the `Burn` event vm.expectEmit(address(optimismSuperchainERC20)); @@ -252,4 +254,64 @@ contract OptimismSuperchainERC20Test is Test { vm.assume(_interfaceId != type(IOptimismSuperchainERC20).interfaceId); assertFalse(optimismSuperchainERC20.supportsInterface(_interfaceId)); } + + /// @notice Tests that the allowance function returns the max uint256 value when the spender is Permit. + /// @param _randomCaller The address that will call the function - used to fuzz better since the behaviour should be + /// the same regardless of the caller. + /// @param _owner The funds owner. + function testFuzz_allowance_fromPermit2_succeeds(address _randomCaller, address _owner) public { + vm.prank(_randomCaller); + uint256 _allowance = optimismSuperchainERC20.allowance(_owner, Preinstalls.Permit2); + + assertEq(_allowance, type(uint256).max); + } + + /// @notice Tests that the allowance function returns the correct allowance when the spender is not Permit. + /// @param _randomCaller The address that will call the function - used to fuzz better + /// since the behaviour should be the same regardless of the caller. + /// @param _owner The funds owner. + /// @param _guy The address of the spender - It cannot be Permit2. + function testFuzz_allowance_succeeds(address _randomCaller, address _owner, address _guy, uint256 _amount) public { + // Assume + vm.assume(_guy != Preinstalls.Permit2); + + // Arrange + vm.prank(_owner); + optimismSuperchainERC20.approve(_guy, _amount); + + // Act + vm.prank(_randomCaller); + uint256 _allowance = optimismSuperchainERC20.allowance(_owner, _guy); + + // Assert + assertEq(_allowance, _amount); + } + + /// @notice Tests that `transferFrom` works when the caller (spender) is Permit2, without any explicit approval. + /// @param _owner The funds owner. + /// @param _recipient The address of the recipient. + /// @param _amount The amount of tokens to transfer. + function testFuzz_transferFrom_whenPermit2IsCaller_succeeds( + address _owner, + address _recipient, + uint256 _amount + ) + public + { + // Arrange + deal(address(optimismSuperchainERC20), _owner, _amount); + + vm.expectEmit(address(optimismSuperchainERC20)); + emit IERC20Solady.Transfer(_owner, _recipient, _amount); + + // Act + vm.prank(Preinstalls.Permit2); + optimismSuperchainERC20.transferFrom(_owner, _recipient, _amount); + + // Assert + assertEq(optimismSuperchainERC20.balanceOf(_recipient), _amount); + // Handle the case where the source and destination are the same to check the source balance. + if (_owner != _recipient) assertEq(optimismSuperchainERC20.balanceOf(_owner), 0); + else assertEq(optimismSuperchainERC20.balanceOf(_owner), _amount); + } } diff --git a/packages/contracts-bedrock/test/L2/SuperchainERC20.t.sol b/packages/contracts-bedrock/test/L2/SuperchainERC20.t.sol index 999b0ad4ee88..cac83380c3fb 100644 --- a/packages/contracts-bedrock/test/L2/SuperchainERC20.t.sol +++ b/packages/contracts-bedrock/test/L2/SuperchainERC20.t.sol @@ -6,7 +6,7 @@ import { Test } from "forge-std/Test.sol"; // Libraries import { Predeploys } from "src/libraries/Predeploys.sol"; -import { IERC20 } from "@openzeppelin/contracts-v5/token/ERC20/IERC20.sol"; +import { IERC20Solady } from "src/vendor/interfaces/IERC20Solady.sol"; // Target contract import { SuperchainERC20 } from "src/L2/SuperchainERC20.sol"; @@ -58,7 +58,7 @@ contract SuperchainERC20Test is Test { // Look for the emit of the `Transfer` event vm.expectEmit(address(superchainERC20)); - emit IERC20.Transfer(ZERO_ADDRESS, _to, _amount); + emit IERC20Solady.Transfer(ZERO_ADDRESS, _to, _amount); // Look for the emit of the `CrosschainMinted` event vm.expectEmit(address(superchainERC20)); @@ -101,7 +101,7 @@ contract SuperchainERC20Test is Test { // Look for the emit of the `Transfer` event vm.expectEmit(address(superchainERC20)); - emit IERC20.Transfer(_from, ZERO_ADDRESS, _amount); + emit IERC20Solady.Transfer(_from, ZERO_ADDRESS, _amount); // Look for the emit of the `CrosschainBurnt` event vm.expectEmit(address(superchainERC20));