eip | title | description | author | discussions-to | status | type | category | created | requires |
---|---|---|---|---|---|---|---|---|---|
7511 |
Minimal Proxy Contract with PUSH0 |
Optimizes the previous Minimal Proxy Contract with the PUSH0 opcode |
0xAA (@AmazingAng), vectorized (@Vectorized), 0age (@0age) |
Draft |
Standards Track |
ERC |
2023-09-04 |
7, 211, 1167, 3855 |
With the PUSH0
opcode (EIP-3855) introduced with the Shanghai upgrade, the previous Minimal Proxy Contract (ERC-1167) cen be further optimized. We provides two starndard optimized Minimal Proxy Contract with PUSH0
: Clone044
optimized for deployment cost, saving 200 gas at deployment and 5 gas at runtime; Clone045
optimized for runtime cost, saving 8 gas at runtime.
- Reduce the contract bytecode size.
- Reduce the runtime gas by replacing
DUP
(cost3
gas each) withPUSH0
(cost2
gas each). - Increase the readability of the proxy contract by redesigning it from first principles with
PUSH0
.
The exact runtime code for the minimal proxy contract with PUSH0
is:
365f5f375f5f365f73bebebebebebebebebebebebebebebebebebebebe5af43d5f5f3e5f3d91602a57fd5bf3
where the bytes at indices 9 - 28 (inclusive) are replaced with the 20-byte address of the master implementation contract. The length of the runtime code is 44
bytes.
The disassembly of the new minimal proxy contract code is:
pc | op | opcode | stack |
---|---|---|---|
[00] | 36 | CALLDATASIZE | cds |
[01] | 5f | PUSH0 | 0 cds |
[02] | 5f | PUSH0 | 0 0 cds |
[03] | 37 | CALLDATACOPY | |
[04] | 5f | PUSH0 | 0 |
[05] | 5f | PUSH0 | 0 0 |
[06] | 36 | CALLDATASIZE | cds 0 0 |
[07] | 5f | PUSH0 | 0 cds 0 0 |
[08] | 73bebe. | PUSH20 0xbebe. | 0xbebe. 0 cds 0 0 |
[1d] | 5a | GAS | gas 0xbebe. 0 cds 0 0 |
[1e] | f4 | DELEGATECALL | suc |
[1f] | 3d | RETURNDATASIZE | rds suc |
[20] | 5f | PUSH0 | 0 rds suc |
[21] | 5f | PUSH0 | 0 0 rds suc |
[22] | 3e | RETURNDATACOPY | suc |
[23] | 5f | PUSH0 | 0 suc |
[24] | 3d | RETURNDATASIZE | rds 0 suc |
[25] | 91 | SWAP2 | suc 0 rds |
[26] | 602a | PUSH1 0x2a | 0x2a suc 0 rds |
[27] | 57 | JUMPI | 0 rds |
[29] | fd | REVERT | |
[2a] | 5b | JUMPDEST | 0 rds |
[2b] | f3 | RETURN |
The minimal creation code of the minimal proxy contract is:
602c8060095f395ff3365f5f375f5f365f73bebebebebebebebebebebebebebebebebebebebe5af43d5f5f3e5f3d91602a57fd5bf3
where the first 9 bytes are the initcode:
602c8060095f395ff3
And the rest are runtime/contract code of the proxy. The length of the creation code is 53
bytes.
The minimal proxy contract can be deployed with Solidity using the following contract:
// SPDX-License-Identifier: CC0-1.0
pragma solidity ^0.8.20;
// Note: this contract requires `PUSH0`, which is available in solidity > 0.8.20 and EVM version > Shanghai
contract Clone0Factory {
error FailedCreateClone();
receive() external payable {}
/**
* @dev Deploys and returns the address of a clone0 (Minimal Proxy Contract with `PUSH0`) that mimics the behaviour of `implementation`.
*
* This function uses the create opcode, which should never revert.
*/
function clone0(address impl) public payable returns (address addr) {
// first 18 bytes of the creation code
bytes memory data1 = hex"602c8060095f395ff3365f5f375f5f365f73";
// last 15 bytes of the creation code
bytes memory data2 = hex"5af43d5f5f3e5f3d91602a57fd5bf3";
// complete the creation code of Clone0
bytes memory _code = abi.encodePacked(data1, impl, data2);
// deploy with create op
assembly {
// create(v, p, n)
addr := create(callvalue(), add(_code, 0x20), mload(_code))
}
if (addr == address(0)) {
revert FailedCreateClone();
}
}
}
The optimized contract is constructed with essential components of the proxy contract and incorporates the recently added PUSH0
opcode. The core elements of the minimal proxy include:
- Copy the calldata with
CALLDATACOPY
. - Forward the calldata to the implementation contract using
DELEGATECALL
. - Copy the returned data from the
DELEGATECALL
. - Return the results or revert the transaction based on whether the
DELEGATECALL
is successful.
To copy the calldata, we need to provide the arguments for the CALLDATACOPY
opcodes, which are [0, 0, cds]
, where cds
represents calldata size.
pc | op | opcode | stack |
---|---|---|---|
[00] | 36 | CALLDATASIZE | cds |
[01] | 5f | PUSH0 | 0 cds |
[02] | 5f | PUSH0 | 0 0 cds |
[03] | 37 | CALLDATACOPY |
To forward the calldata to the delegate call, we need to prepare arguments for the DELEGATECALL
opcodes, which are [gas 0xbebe. 0 cds 0 0]
, where gas
represents the remaining gas, 0xbebe.
represents the address of the implementation contract, and suc
represents whether the delegatecall is successful.
pc | op | opcode | stack |
---|---|---|---|
[04] | 5f | PUSH0 | 0 |
[05] | 5f | PUSH0 | 0 0 |
[06] | 36 | CALLDATASIZE | cds 0 0 |
[07] | 5f | PUSH0 | 0 cds 0 0 |
[08] | 73bebe. | PUSH20 0xbebe. | 0xbebe. 0 cds 0 0 |
[1d] | 5a | GAS | gas 0xbebe. 0 cds 0 0 |
[1e] | f4 | DELEGATECALL | suc |
To copy the returndata, we need to provide the arguments for the RETURNDATACOPY
opcodes, which are [0, 0, rds]
, where rds
represents size of returndata from the DELEGATECALL
.
pc | op | opcode | stack |
---|---|---|---|
[1f] | 3d | RETURNDATASIZE | rds suc |
[20] | 5f | PUSH0 | 0 rds suc |
[21] | 5f | PUSH0 | 0 0 rds suc |
[22] | 3e | RETURNDATACOPY | suc |
Lastly, we need to return the data or revert the transaction based on whether the DELEGATECALL
is successful. There is no if/else
in opcodes, so we need to use JUMPI
and JUMPDEST
instead. The arguments for JUMPI
is [0x2a, suc]
, where 0x2a
is the destination of the conditional jump.
We also need to prepare the argument [0, rds]
for REVERT
and RETURN
opcodes before the JUMPI
, otherwise we have to prepare them twice. We cannot avoid the SWAP
operation, because we can only get rds
after the DELEGATECALL
.
pc | op | opcode | stack |
---|---|---|---|
[23] | 5f | PUSH0 | 0 suc |
[24] | 3d | RETURNDATASIZE | rds 0 suc |
[25] | 91 | SWAP2 | suc 0 rds |
[26] | 602a | PUSH1 0x2a | 0x2a suc 0 rds |
[27] | 57 | JUMPI | 0 rds |
[29] | fd | REVERT | |
[2a] | 5b | JUMPDEST | 0 rds |
[2b] | f3 | RETURN |
In the end, we arrived at the runtime code for Minimal Proxy Contract with PUSH0
:
365f5f375f5f365f73bebebebebebebebebebebebebebebebebebebebe5af43d5f5f3e5f3d91602a57fd5bf3
The length of the runtime code is 44
bytes, which reduced 1
byte from the previous Minimal Proxy Contract. Moreover, it replaced the RETURNDATASIZE
and DUP
operations with PUSH0
, which saves gas and increases the readability of the code. In summary, the new Minimal Proxy Contract reduces 200
gas at deployment and 5
gas at runtime, while remaining the same functionalities as the old one.
Because the new minimal proxy contract uses PUSH0
opcode, it can only be deployed after the Shanghai Upgrade. It behaves the same as the previous Minimal Proxy Contract.
The new proxy contract standard is identical to the previous one (ERC-1167). Here are the security considerations when using minimal proxy contracts:
-
Non-Upgradability: Minimal Proxy Contracts delegate their logic to another contract (often termed the "implementation" or "logic" contract). This delegation is fixed upon deployment, meaning you can't change which implementation contract the proxy delegates to after its creation.
-
Initialization Concerns: Proxy contracts lack constructors, so you need to use an initialization function after deployment. Skipping this step could leave the contract unsafe.
-
Safety of Logic Contract: Vulnerabilities in the logic contract affect all associated proxy contracts.
-
Transparency Issues: Because of its complexity, users might see the proxy as an empty contract, making it challenging to trace back to the actual logic contract.
Copyright and related rights waived via CC0.