Skip to content

Commit 799837d

Browse files
godzillabagzeoneth
andauthored
use create2 in BridgeCreator (#265)
* docs: add note regarding reorg * wip: use create2 * edit comment * start over * use create2 in bridge creator * require rollup exists * fix tests * fix lint * Revert "docs: add note regarding reorg" This reverts commit d451b72. * remove unnecessary args from salt * fmt * put the salt args back * refactor: move hardcoded value to config * refactor: minimumAssertionPeriod and validatorAfkBlocks in bold upgrade setting * perf: remove onchain verification * chore: update sigs * refactor: remove hardcode in rollup initialize * chore: update example * test: new config param * fix: format * fix: BridgeTemplates * fix: memory * fix: format test * feat: use msg.data hash as salt * test: fix * test: fix config * fix: slither * refactor: bridge creator create2Salt * chore: remove unintended change * chore: slither --------- Co-authored-by: gzeon <[email protected]> Co-authored-by: gzeon <[email protected]>
1 parent 2b13047 commit 799837d

File tree

3 files changed

+49
-6
lines changed

3 files changed

+49
-6
lines changed

slither.db.json

+1-1
Large diffs are not rendered by default.

src/rollup/BridgeCreator.sol

+23-5
Original file line numberDiff line numberDiff line change
@@ -65,17 +65,22 @@ contract BridgeCreator is Ownable {
6565
}
6666

6767
function _createBridge(
68+
bytes32 create2Salt,
6869
address adminProxy,
6970
BridgeTemplates memory templates,
7071
bool isDelayBufferable
7172
) internal returns (BridgeContracts memory) {
7273
BridgeContracts memory frame;
7374
frame.bridge = IBridge(
74-
address(new TransparentUpgradeableProxy(address(templates.bridge), adminProxy, ""))
75+
address(
76+
new TransparentUpgradeableProxy{salt: create2Salt}(
77+
address(templates.bridge), adminProxy, ""
78+
)
79+
)
7580
);
7681
frame.sequencerInbox = ISequencerInbox(
7782
address(
78-
new TransparentUpgradeableProxy(
83+
new TransparentUpgradeableProxy{salt: create2Salt}(
7984
address(
8085
isDelayBufferable
8186
? templates.delayBufferableSequencerInbox
@@ -87,15 +92,25 @@ contract BridgeCreator is Ownable {
8792
)
8893
);
8994
frame.inbox = IInboxBase(
90-
address(new TransparentUpgradeableProxy(address(templates.inbox), adminProxy, ""))
95+
address(
96+
new TransparentUpgradeableProxy{salt: create2Salt}(
97+
address(templates.inbox), adminProxy, ""
98+
)
99+
)
91100
);
92101
frame.rollupEventInbox = IRollupEventInbox(
93102
address(
94-
new TransparentUpgradeableProxy(address(templates.rollupEventInbox), adminProxy, "")
103+
new TransparentUpgradeableProxy{salt: create2Salt}(
104+
address(templates.rollupEventInbox), adminProxy, ""
105+
)
95106
)
96107
);
97108
frame.outbox = IOutbox(
98-
address(new TransparentUpgradeableProxy(address(templates.outbox), adminProxy, ""))
109+
address(
110+
new TransparentUpgradeableProxy{salt: create2Salt}(
111+
address(templates.outbox), adminProxy, ""
112+
)
113+
)
99114
);
100115
return frame;
101116
}
@@ -107,11 +122,14 @@ contract BridgeCreator is Ownable {
107122
ISequencerInbox.MaxTimeVariation calldata maxTimeVariation,
108123
BufferConfig calldata bufferConfig
109124
) external returns (BridgeContracts memory) {
125+
// use create2 salt to ensure deterministic addresses
126+
bytes32 create2Salt = keccak256(abi.encode(msg.data, msg.sender));
110127
// create delay bufferable sequencer inbox if threshold is non-zero
111128
bool isDelayBufferable = bufferConfig.threshold != 0;
112129

113130
// create ETH-based bridge if address zero is provided for native token, otherwise create ERC20-based bridge
114131
BridgeContracts memory frame = _createBridge(
132+
create2Salt,
115133
adminProxy,
116134
nativeToken == address(0) ? ethBasedTemplates : erc20BasedTemplates,
117135
isDelayBufferable

test/foundry/BridgeCreator.t.sol

+25
Original file line numberDiff line numberDiff line change
@@ -134,6 +134,7 @@ contract BridgeCreatorTest is Test {
134134
max: type(uint64).max,
135135
replenishRateInBasis: 0
136136
});
137+
137138
BridgeCreator.BridgeContracts memory contracts =
138139
creator.createBridge(proxyAdmin, rollup, nativeToken, timeVars, bufferConfig);
139140
(
@@ -245,4 +246,28 @@ contract BridgeCreatorTest is Test {
245246
assertEq(address(outbox.bridge()), address(bridge), "Invalid bridge ref");
246247
assertEq(address(outbox.rollup()), rollup, "Invalid rollup ref");
247248
}
249+
250+
function test_canOnlyDeployOnce() external {
251+
address proxyAdmin = address(300);
252+
address rollup = address(301);
253+
address nativeToken =
254+
address(new ERC20PresetFixedSupply("Appchain Token", "App", 1_000_000, address(this)));
255+
ISequencerInbox.MaxTimeVariation memory timeVars =
256+
ISequencerInbox.MaxTimeVariation(10, 20, 30, 40);
257+
BufferConfig memory bufferConfig = BufferConfig({
258+
threshold: type(uint64).max,
259+
max: type(uint64).max,
260+
replenishRateInBasis: 0
261+
});
262+
263+
creator.createBridge(proxyAdmin, rollup, nativeToken, timeVars, bufferConfig);
264+
265+
// can only deploy once from the same address and config
266+
vm.expectRevert();
267+
creator.createBridge(proxyAdmin, rollup, nativeToken, timeVars, bufferConfig);
268+
269+
// can deploy from a different address
270+
vm.prank(address(101));
271+
creator.createBridge(proxyAdmin, rollup, nativeToken, timeVars, bufferConfig);
272+
}
248273
}

0 commit comments

Comments
 (0)