-
Notifications
You must be signed in to change notification settings - Fork 3.4k
/
Copy pathL2Genesis.s.sol
429 lines (364 loc) · 20.1 KB
/
L2Genesis.s.sol
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
// SPDX-License-Identifier: MIT
pragma solidity 0.8.15;
import { Script } from "forge-std/Script.sol";
import { console2 as console } from "forge-std/console2.sol";
import { Artifacts } from "scripts/Artifacts.s.sol";
import { DeployConfig } from "scripts/DeployConfig.s.sol";
import { Predeploys } from "src/libraries/Predeploys.sol";
import { L1StandardBridge } from "src/L1/L1StandardBridge.sol";
import { L1CrossDomainMessenger } from "src/L1/L1CrossDomainMessenger.sol";
import { L2StandardBridge } from "src/L2/L2StandardBridge.sol";
import { L2CrossDomainMessenger } from "src/L2/L2CrossDomainMessenger.sol";
import { SequencerFeeVault } from "src/L2/SequencerFeeVault.sol";
import { FeeVault } from "src/universal/FeeVault.sol";
import { OptimismMintableERC20Factory } from "src/universal/OptimismMintableERC20Factory.sol";
import { L1Block } from "src/L2/L1Block.sol";
import { GovernanceToken } from "src/governance/GovernanceToken.sol";
import { EIP1967Helper } from "test/mocks/EIP1967Helper.sol";
interface IInitializable {
function initialize(address _addr) external;
}
/// @dev The general flow of adding a predeploy is:
/// 1. _setPredeployProxies uses vm.etch to set the Proxy.sol deployed bytecode for proxy address `0x420...000` to
/// `0x420...000 + PROXY_COUNT - 1`.
/// Additionally, the PROXY_ADMIN_ADDRESS and PROXY_IMPLEMENTATION_ADDRESS storage slots are set for the proxy
/// address.
/// 2. `vm.etch` sets the deployed bytecode for each predeploy at the implementation address (i.e. `0xc0d3`
/// namespace).
/// 3. The `initialize` method is called at the implementation address with zero/dummy vaules if the method exists.
/// 4. The `initialize` method is called at the proxy address with actual vaules if the method exists.
/// 5. A `require` check to verify the expected implementation address is set for the proxy.
/// @notice The following safety invariants are used when setting state:
/// 1. `vm.getDeployedBytecode` can only be used with `vm.etch` when there are no side
/// effects in the constructor and no immutables in the bytecode.
/// 2. A contract must be deployed using the `new` syntax if there are immutables in the code.
/// Any other side effects from the init code besides setting the immutables must be cleaned up afterwards.
/// 3. A contract is deployed using the `new` syntax, however it's not proxied and is still expected to exist at
/// a
/// specific implementation address (i.e. `0xc0d3` namespace). In this case we deploy an instance of the
/// contract
/// using `new` syntax, use `contract.code` to retrieve it's deployed bytecode, `vm.etch` the bytecode at the
/// expected implementation address, and `vm.store` to set any storage slots that are
/// expected to be set after a new deployment. Lastly, we reset the account code and storage slots the contract
/// was initially deployed to so it's not included in the `vm.dumpState`.
contract L2Genesis is Script, Artifacts {
uint256 constant PROXY_COUNT = 2048;
uint256 constant PRECOMPILE_COUNT = 256;
DeployConfig public constant cfg =
DeployConfig(address(uint160(uint256(keccak256(abi.encode("optimism.deployconfig"))))));
/// @notice The storage slot that holds the address of a proxy implementation.
/// @dev `bytes32(uint256(keccak256('eip1967.proxy.implementation')) - 1)`
bytes32 internal constant PROXY_IMPLEMENTATION_ADDRESS =
0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc;
/// @notice The storage slot that holds the address of the owner.
/// @dev `bytes32(uint256(keccak256('eip1967.proxy.admin')) - 1)`
bytes32 internal constant PROXY_ADMIN_ADDRESS = 0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103;
uint80 internal constant DEV_ACCOUNT_FUND_AMT = 10_000 ether;
/// @notice Default Anvil dev accounts. Only funded if `cfg.fundDevAccounts == true`.
address[10] internal devAccounts = [
0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266,
0x70997970C51812dc3A010C7d01b50e0d17dc79C8,
0x3C44CdDdB6a900fa2b585dd299e03d12FA4293BC,
0x90F79bf6EB2c4f870365E785982E1f101E93b906,
0x15d34AAf54267DB7D7c367839AAf71A00a2C6A65,
0x9965507D1a55bcC2695C58ba16FB37d819B0A4dc,
0x976EA74026E726554dB657fA54763abd0C3a0aa9,
0x14dC79964da2C08b23698B3D3cc7Ca32193d9955,
0x23618e81E3f5cdF7f54C3d65f7FBc0aBf5B21E8f,
0xa0Ee7A142d267C1f36714E4a8F75612F20a79720
];
string internal outfile;
/// @dev Reads the deploy config, sets `outfile` which is where the `vm.dumpState` will be saved to, and
/// loads in the addresses for the L1 contract deployments.
function setUp() public override {
Artifacts.setUp();
string memory path = string.concat(vm.projectRoot(), "/deploy-config/", deploymentContext, ".json");
vm.etch(address(cfg), vm.getDeployedCode("DeployConfig.s.sol:DeployConfig"));
vm.label(address(cfg), "DeployConfig");
vm.allowCheatcodes(address(cfg));
cfg.read(path);
outfile = string.concat(vm.projectRoot(), "/deployments/", deploymentContext, "/genesis-l2.json");
_loadAddresses(string.concat(vm.projectRoot(), "/deployments/", deploymentContext, "/.deploy"));
}
/// @dev Sets the precompiles, proxies, and the implementation accounts to be `vm.dumpState`
/// to generate a L2 genesis alloc.
/// @notice The alloc object is sorted numerically by address.
function run() public {
_dealEthToPrecompiles();
_setPredeployProxies();
_setPredeployImplementations();
if (cfg.fundDevAccounts()) {
_fundDevAccounts();
}
/// Reset so its not included state dump
vm.etch(address(cfg), "");
vm.dumpState(outfile);
_sortJsonByKeys(outfile);
}
/// @notice Give all of the precompiles 1 wei so that they are
/// not considered empty accounts.
function _dealEthToPrecompiles() internal {
for (uint256 i; i < PRECOMPILE_COUNT; i++) {
vm.deal(address(uint160(i)), 1);
}
}
/// @dev Set up the accounts that correspond to the predeploys.
/// The Proxy bytecode should be set. All proxied predeploys should have
/// the 1967 admin slot set to the ProxyAdmin predeploy. All defined predeploys
/// should have their implementations set.
function _setPredeployProxies() internal {
bytes memory code = vm.getDeployedCode("Proxy.sol:Proxy");
uint160 prefix = uint160(0x420) << 148;
console.log(
"Setting proxy deployed bytecode for addresses in range %s through %s",
address(prefix | uint160(0)),
address(prefix | uint160(PROXY_COUNT - 1))
);
for (uint256 i = 0; i < PROXY_COUNT; i++) {
address addr = address(prefix | uint160(i));
if (_notProxied(addr)) {
continue;
}
vm.etch(addr, code);
vm.store(addr, PROXY_ADMIN_ADDRESS, bytes32(uint256(uint160(Predeploys.PROXY_ADMIN))));
if (_isDefinedPredeploy(addr)) {
address implementation = _predeployToCodeNamespace(addr);
console.log("Setting proxy %s implementation: %s", addr, implementation);
vm.store(addr, PROXY_IMPLEMENTATION_ADDRESS, bytes32(uint256(uint160(implementation))));
}
}
}
/// @notice LEGACY_ERC20_ETH is not being predeployed since it's been deprecated.
/// @dev Sets all the implementations for the predeploy proxies. For contracts without proxies,
/// sets the deployed bytecode at their expected predeploy address.
function _setPredeployImplementations() internal {
_setLegacyMessagePasser();
_setDeployerWhitelist();
_setWETH9();
_setL2StandardBridge();
_setL2CrossDomainMessenger();
_setSequencerFeeVault();
_setOptimismMintableERC20Factory();
_setL1BlockNumber();
_setGasPriceOracle();
_setGovernanceToken();
_setL1Block();
}
/// @notice This predeploy is following the saftey invariant #1.
function _setLegacyMessagePasser() internal {
_setImplementationCode(Predeploys.LEGACY_MESSAGE_PASSER, "LegacyMessagePasser");
}
/// @notice This predeploy is following the saftey invariant #1.
function _setDeployerWhitelist() internal {
_setImplementationCode(Predeploys.DEPLOYER_WHITELIST, "DeployerWhitelist");
}
/// @notice This predeploy is following the saftey invariant #1.
/// Contract metadata hash appended to deployed bytecode will differ
/// from previous L2 genesis output.
/// This contract is NOT proxied.
/// @dev We're manually setting storage slots because we need to deployment to be at
/// the address `Predeploys.WETH9`, so we can't just deploy a new instance of `WETH9`.
function _setWETH9() internal {
console.log("Setting %s implementation at: %s", "WETH9", Predeploys.WETH9);
vm.etch(Predeploys.WETH9, vm.getDeployedCode("WETH9.sol:WETH9"));
vm.store(
Predeploys.WETH9,
/// string public name
hex"0000000000000000000000000000000000000000000000000000000000000000",
/// "Wrapped Ether"
hex"577261707065642045746865720000000000000000000000000000000000001a"
);
vm.store(
Predeploys.WETH9,
/// string public symbol
hex"0000000000000000000000000000000000000000000000000000000000000001",
/// "WETH"
hex"5745544800000000000000000000000000000000000000000000000000000008"
);
vm.store(
Predeploys.WETH9,
// uint8 public decimals
hex"0000000000000000000000000000000000000000000000000000000000000002",
/// 18
hex"0000000000000000000000000000000000000000000000000000000000000012"
);
}
/// @notice This predeploy is following the saftey invariant #1.
/// We're initializing the implementation with `address(0)` so
/// it's not left uninitialized. After `initialize` is called on the
/// proxy to set the storage slot with the expected value.
function _setL2StandardBridge() internal {
address impl = _setImplementationCode(Predeploys.L2_STANDARD_BRIDGE, "L2StandardBridge");
L2StandardBridge(payable(impl)).initialize(L1StandardBridge(payable(address(0))));
L2StandardBridge(payable(Predeploys.L2_STANDARD_BRIDGE)).initialize(
L1StandardBridge(mustGetAddress("L1StandardBridgeProxy"))
);
_checkL2StandardBridge(impl);
}
/// @notice This predeploy is following the saftey invariant #1.
/// We're initializing the implementation with `address(0)` so
/// it's not left uninitialized. After `initialize` is called on the
/// proxy to set the storage slot with the expected value.
function _setL2CrossDomainMessenger() internal {
address impl = _setImplementationCode(Predeploys.L2_CROSS_DOMAIN_MESSENGER, "L2CrossDomainMessenger");
L2CrossDomainMessenger(impl).initialize(L1CrossDomainMessenger(address(0)));
L2CrossDomainMessenger(Predeploys.L2_CROSS_DOMAIN_MESSENGER).initialize(
L1CrossDomainMessenger(mustGetAddress("L1CrossDomainMessengerProxy"))
);
_checkL2CrossDomainMessenger(impl);
}
/// @notice This predeploy is following the saftey invariant #2,
/// because the constructor args are non-static L1 contract
/// addresses that are being read from the deploy config
/// that are set as immutables.
/// @dev Because the constructor args are stored as immutables,
/// we don't have to worry about setting storage slots.
function _setSequencerFeeVault() internal {
SequencerFeeVault vault = new SequencerFeeVault({
_recipient: cfg.sequencerFeeVaultRecipient(),
_minWithdrawalAmount: cfg.sequencerFeeVaultMinimumWithdrawalAmount(),
_withdrawalNetwork: FeeVault.WithdrawalNetwork(cfg.sequencerFeeVaultWithdrawalNetwork())
});
address impl = _predeployToCodeNamespace(Predeploys.SEQUENCER_FEE_WALLET);
console.log("Setting %s implementation at: %s", "SequencerFeeVault", impl);
vm.etch(impl, address(vault).code);
/// Reset so its not included state dump
vm.etch(address(vault), "");
vm.resetNonce(address(vault));
_checkSequencerFeeVault(impl);
}
/// @notice This predeploy is following the saftey invariant #1.
/// We're initializing the implementation with `address(0)` so
/// it's not left uninitialized. After `initialize` is called on the
/// proxy to set the storage slot with the expected value.
function _setOptimismMintableERC20Factory() internal {
address impl =
_setImplementationCode(Predeploys.OPTIMISM_MINTABLE_ERC20_FACTORY, "OptimismMintableERC20Factory");
OptimismMintableERC20Factory(impl).initialize(address(0));
OptimismMintableERC20Factory(Predeploys.OPTIMISM_MINTABLE_ERC20_FACTORY).initialize(
Predeploys.L2_STANDARD_BRIDGE
);
_checkOptimismMintableERC20Factory(impl);
}
/// @notice This predeploy is following the saftey invariant #1.
/// This contract has no initializer.
function _setL1BlockNumber() internal {
_setImplementationCode(Predeploys.L1_BLOCK_NUMBER, "L1BlockNumber");
}
/// @notice This predeploy is following the saftey invariant #1.
/// This contract has no initializer.
function _setGasPriceOracle() internal {
_setImplementationCode(Predeploys.GAS_PRICE_ORACLE, "GasPriceOracle");
}
/// @notice This predeploy is following the saftey invariant #3.
function _setGovernanceToken() internal {
if (!cfg.enableGovernance()) {
console.log("Governance not enabled, skipping setting governanace token");
return;
}
GovernanceToken token = new GovernanceToken();
console.log("Setting %s implementation at: %s", "GovernanceToken", Predeploys.GOVERNANCE_TOKEN);
vm.etch(Predeploys.GOVERNANCE_TOKEN, address(token).code);
bytes32 _nameSlot = hex"0000000000000000000000000000000000000000000000000000000000000003";
bytes32 _symbolSlot = hex"0000000000000000000000000000000000000000000000000000000000000004";
bytes32 _ownerSlot = hex"000000000000000000000000000000000000000000000000000000000000000a";
vm.store(Predeploys.GOVERNANCE_TOKEN, _nameSlot, vm.load(address(token), _nameSlot));
vm.store(Predeploys.GOVERNANCE_TOKEN, _symbolSlot, vm.load(address(token), _symbolSlot));
vm.store(Predeploys.GOVERNANCE_TOKEN, _ownerSlot, bytes32(uint256(uint160(cfg.governanceTokenOwner()))));
/// Reset so its not included state dump
vm.etch(address(token), "");
vm.resetNonce(address(token));
}
/// @notice This predeploy is following the saftey invariant #1.
/// This contract has no initializer.
/// @dev Previously the initial L1 attributes was set at genesis, to simplify,
/// they no longer are so the resulting storage slots are no longer set.
function _setL1Block() internal {
_setImplementationCode(Predeploys.L1_BLOCK_ATTRIBUTES, "L1Block");
}
/// @dev Returns true if the address is not proxied.
function _notProxied(address _addr) internal pure returns (bool) {
return _addr == Predeploys.GOVERNANCE_TOKEN || _addr == Predeploys.WETH9;
}
/// @dev Returns true if the address is a predeploy.
function _isDefinedPredeploy(address _addr) internal pure returns (bool) {
return _addr == Predeploys.L2_TO_L1_MESSAGE_PASSER || _addr == Predeploys.L2_CROSS_DOMAIN_MESSENGER
|| _addr == Predeploys.L2_STANDARD_BRIDGE || _addr == Predeploys.L2_ERC721_BRIDGE
|| _addr == Predeploys.SEQUENCER_FEE_WALLET || _addr == Predeploys.OPTIMISM_MINTABLE_ERC20_FACTORY
|| _addr == Predeploys.OPTIMISM_MINTABLE_ERC721_FACTORY || _addr == Predeploys.L1_BLOCK_ATTRIBUTES
|| _addr == Predeploys.GAS_PRICE_ORACLE || _addr == Predeploys.DEPLOYER_WHITELIST || _addr == Predeploys.WETH9
|| _addr == Predeploys.L1_BLOCK_NUMBER || _addr == Predeploys.LEGACY_MESSAGE_PASSER
|| _addr == Predeploys.PROXY_ADMIN || _addr == Predeploys.BASE_FEE_VAULT || _addr == Predeploys.L1_FEE_VAULT
|| _addr == Predeploys.GOVERNANCE_TOKEN || _addr == Predeploys.SCHEMA_REGISTRY || _addr == Predeploys.EAS;
}
/// @dev Function to compute the expected address of the predeploy implementation
/// in the genesis state.
function _predeployToCodeNamespace(address _addr) internal pure returns (address) {
return address(
uint160(uint256(uint160(_addr)) & 0xffff | uint256(uint160(0xc0D3C0d3C0d3C0D3c0d3C0d3c0D3C0d3c0d30000)))
);
}
function _setImplementationCode(address _addr, string memory _name) internal returns (address) {
address impl = _predeployToCodeNamespace(_addr);
console.log("Setting %s implementation at: %s", _name, impl);
vm.etch(impl, vm.getDeployedCode(string.concat(_name, ".sol:", _name)));
_verifyProxyImplementationAddress(_addr, impl);
return impl;
}
/// @dev Function to verify the expected implementation address is set for the respective proxy.
function _verifyProxyImplementationAddress(address _proxy, address _impl) internal view {
require(
EIP1967Helper.getImplementation(_proxy) == _impl,
"Expected different address at Proxys PROXY_IMPLEMENTATION_ADDRESS storage slot"
);
}
/// @dev Function to verify that a contract was initialized, and can't be reinitialized.
/// @notice There isn't a good way to know if the resulting revering is due to abi mismatch
/// or because it's already been initialized
function _verifyCantReinitialize(address _contract, address _arg) internal {
vm.expectRevert("Initializable: contract is already initialized");
IInitializable(_contract).initialize(_arg);
}
/// @dev Helper function to sort the genesis alloc numerically by address.
function _sortJsonByKeys(string memory _path) internal {
string[] memory commands = new string[](3);
commands[0] = "bash";
commands[1] = "-c";
commands[2] = string.concat("cat <<< $(jq -S '.' ", _path, ") > ", _path);
vm.ffi(commands);
}
function _fundDevAccounts() internal {
for (uint256 i; i < devAccounts.length; i++) {
console.log("Funding dev account %s with %s ETH", devAccounts[i], DEV_ACCOUNT_FUND_AMT / 1e18);
vm.deal(devAccounts[i], DEV_ACCOUNT_FUND_AMT);
}
_checkDevAccountsFunded();
}
//////////////////////////////////////////////////////
/// Post Checks
//////////////////////////////////////////////////////
function _checkL2StandardBridge(address _impl) internal {
_verifyCantReinitialize(_impl, address(0));
_verifyCantReinitialize(Predeploys.L2_STANDARD_BRIDGE, mustGetAddress("L1StandardBridgeProxy"));
}
function _checkL2CrossDomainMessenger(address _impl) internal {
_verifyCantReinitialize(_impl, address(0));
_verifyCantReinitialize(Predeploys.L2_CROSS_DOMAIN_MESSENGER, mustGetAddress("L1CrossDomainMessengerProxy"));
}
function _checkSequencerFeeVault(address _impl) internal view {
_verifyProxyImplementationAddress(Predeploys.SEQUENCER_FEE_WALLET, _impl);
}
function _checkOptimismMintableERC20Factory(address _impl) internal {
_verifyCantReinitialize(_impl, address(0));
_verifyCantReinitialize(Predeploys.OPTIMISM_MINTABLE_ERC20_FACTORY, Predeploys.L2_STANDARD_BRIDGE);
}
function _checkDevAccountsFunded() internal view {
for (uint256 i; i < devAccounts.length; i++) {
if (devAccounts[i].balance != DEV_ACCOUNT_FUND_AMT) {
revert(
string.concat("Dev account not funded with expected amount of ETH: ", vm.toString(devAccounts[i]))
);
}
}
}
}