diff --git a/.clabot b/.clabot new file mode 100644 index 00000000..b4ac3516 --- /dev/null +++ b/.clabot @@ -0,0 +1,5 @@ +{ + "contributors": "https://api.github.com/repos/OffchainLabs/clabot-config/contents/nitro-contributors.json", + "message": "We require contributors to sign our Contributor License Agreement. In order for us to review and merge your code, please sign the linked documents below to get yourself added. https://na3.docusign.net/Member/PowerFormSigning.aspx?PowerFormId=b15c81cc-b5ea-42a6-9107-3992526f2898&env=na3&acct=6e152afc-6284-44af-a4c1-d8ef291db402&v=2", + "label": "s" +} \ No newline at end of file diff --git a/README.md b/README.md index fdfeb383..8fc3d98c 100644 --- a/README.md +++ b/README.md @@ -12,8 +12,8 @@ For the token bridge contracts see https://github.com/OffchainLabs/token-bridge- Compile these contracts locally by running ```bash -git clone https://github.com/offchainlabs/rollup-contracts -cd rollup-contracts +git clone https://github.com/offchainlabs/nitro-contracts +cd nitro-contracts yarn install yarn build ``` diff --git a/src/bridge/Bridge.sol b/src/bridge/Bridge.sol index 722a5e28..e8a6764e 100644 --- a/src/bridge/Bridge.sol +++ b/src/bridge/Bridge.sol @@ -230,7 +230,7 @@ contract Bridge is Initializable, DelegateCallAware, IBridge { InOutInfo storage info = allowedDelayedInboxesMap[inbox]; bool alreadyEnabled = info.allowed; emit InboxToggle(inbox, enabled); - if ((alreadyEnabled && enabled) || (!alreadyEnabled && !enabled)) { + if (alreadyEnabled == enabled) { return; } if (enabled) { @@ -252,7 +252,7 @@ contract Bridge is Initializable, DelegateCallAware, IBridge { InOutInfo storage info = allowedOutboxesMap[outbox]; bool alreadyEnabled = info.allowed; emit OutboxToggle(outbox, enabled); - if ((alreadyEnabled && enabled) || (!alreadyEnabled && !enabled)) { + if (alreadyEnabled == enabled) { return; } if (enabled) { diff --git a/src/bridge/ISequencerInbox.sol b/src/bridge/ISequencerInbox.sol index d801dca3..55454a88 100644 --- a/src/bridge/ISequencerInbox.sol +++ b/src/bridge/ISequencerInbox.sol @@ -69,14 +69,24 @@ interface ISequencerInbox is IDelayedMessageProvider { function isBatchPoster(address) external view returns (bool); + function isSequencer(address) external view returns (bool); + struct DasKeySetInfo { bool isValidKeyset; uint64 creationBlock; } - // https://github.com/ethereum/solidity/issues/11826 - // function maxTimeVariation() external view returns (MaxTimeVariation calldata); - // function dasKeySetInfo(bytes32) external view returns (DasKeySetInfo calldata); + function maxTimeVariation() + external + view + returns ( + uint256, + uint256, + uint256, + uint256 + ); + + function dasKeySetInfo(bytes32) external view returns (bool, uint64); /// @notice Remove force inclusion delay after a L1 chainId fork function removeDelayAfterFork() external; @@ -154,6 +164,14 @@ interface ISequencerInbox is IDelayedMessageProvider { */ function invalidateKeysetHash(bytes32 ksHash) external; + /** + * @notice Updates whether an address is authorized to be a sequencer. + * @dev The IsSequencer information is used only off-chain by the nitro node to validate sequencer feed signer. + * @param addr the address + * @param isSequencer_ if the specified address should be authorized as a sequencer + */ + function setIsSequencer(address addr, bool isSequencer_) external; + // ---------- initializer ---------- function initialize(IBridge bridge_, MaxTimeVariation calldata maxTimeVariation_) external; diff --git a/src/bridge/SequencerInbox.sol b/src/bridge/SequencerInbox.sol index 5072359b..b0c88a32 100644 --- a/src/bridge/SequencerInbox.sol +++ b/src/bridge/SequencerInbox.sol @@ -27,10 +27,12 @@ import "./IInbox.sol"; import "./ISequencerInbox.sol"; import "../rollup/IRollupLogic.sol"; import "./Messages.sol"; +import "../precompiles/ArbGasInfo.sol"; import {L1MessageType_batchPostingReport} from "../libraries/MessageTypes.sol"; import {GasRefundEnabled, IGasRefunder} from "../libraries/IGasRefunder.sol"; import "../libraries/DelegateCallAware.sol"; +import "../libraries/ArbitrumChecker.sol"; import {MAX_DATA_SIZE} from "../libraries/Constants.sol"; /** @@ -64,6 +66,11 @@ contract SequencerInbox is DelegateCallAware, GasRefundEnabled, ISequencerInbox uint256 internal immutable deployTimeChainId = block.chainid; + mapping(address => bool) public isSequencer; + + // If the chain this SequencerInbox is deployed on is an Arbitrum chain. + bool internal immutable hostChainIsArbitrum = ArbitrumChecker.runningOnArbitrum(); + function _chainIdChanged() internal view returns (bool) { return deployTimeChainId != block.chainid; } @@ -385,12 +392,17 @@ contract SequencerInbox is DelegateCallAware, GasRefundEnabled, ISequencerInbox // this msg isn't included in the current sequencer batch, but instead added to // the delayed messages queue that is yet to be included address batchPoster = msg.sender; + uint256 dataCost = block.basefee; + if (hostChainIsArbitrum) { + // Include extra cost for the host chain's L1 gas charging + dataCost += ArbGasInfo(address(0x6c)).getL1BaseFeeEstimate(); + } bytes memory spendingReportMsg = abi.encodePacked( block.timestamp, batchPoster, dataHash, seqMessageIndex, - block.basefee + dataCost ); uint256 msgNum = bridge.submitBatchSpendingReport( batchPoster, @@ -450,6 +462,12 @@ contract SequencerInbox is DelegateCallAware, GasRefundEnabled, ISequencerInbox emit OwnerFunctionCalled(3); } + /// @inheritdoc ISequencerInbox + function setIsSequencer(address addr, bool isSequencer_) external onlyRollupOwner { + isSequencer[addr] = isSequencer_; + emit OwnerFunctionCalled(4); + } + function isValidKeysetHash(bytes32 ksHash) external view returns (bool) { return dasKeySetInfo[ksHash].isValidKeyset; } diff --git a/src/libraries/AdminFallbackProxy.sol b/src/libraries/AdminFallbackProxy.sol index 78334cdb..52dc9bff 100644 --- a/src/libraries/AdminFallbackProxy.sol +++ b/src/libraries/AdminFallbackProxy.sol @@ -107,13 +107,13 @@ contract AdminFallbackProxy is Proxy, DoubleLogicERC1967Upgrade { * Only the `adminAddr` is able to use the `adminLogic` functions * All other addresses can interact with the `userLogic` functions */ - constructor( + function _initialize( address adminLogic, bytes memory adminData, address userLogic, bytes memory userData, address adminAddr - ) payable { + ) internal { assert(_ADMIN_SLOT == bytes32(uint256(keccak256("eip1967.proxy.admin")) - 1)); assert( _IMPLEMENTATION_SLOT == bytes32(uint256(keccak256("eip1967.proxy.implementation")) - 1) diff --git a/src/libraries/ArbitrumChecker.sol b/src/libraries/ArbitrumChecker.sol new file mode 100644 index 00000000..714c4e75 --- /dev/null +++ b/src/libraries/ArbitrumChecker.sol @@ -0,0 +1,16 @@ +// Copyright 2021-2022, Offchain Labs, Inc. +// For license information, see https://github.com/nitro/blob/master/LICENSE +// SPDX-License-Identifier: BUSL-1.1 + +pragma solidity ^0.8.0; + +import "../precompiles/ArbSys.sol"; + +library ArbitrumChecker { + function runningOnArbitrum() internal view returns (bool) { + (bool ok, bytes memory data) = address(100).staticcall( + abi.encodeWithSelector(ArbSys.arbOSVersion.selector) + ); + return ok && data.length == 32; + } +} diff --git a/src/libraries/IGasRefunder.sol b/src/libraries/IGasRefunder.sol index 3e915c3c..99e48bf5 100644 --- a/src/libraries/IGasRefunder.sol +++ b/src/libraries/IGasRefunder.sol @@ -21,10 +21,7 @@ abstract contract GasRefundEnabled { uint256 startGasLeft = gasleft(); _; if (address(gasRefunder) != address(0)) { - uint256 calldataSize; - assembly { - calldataSize := calldatasize() - } + uint256 calldataSize = msg.data.length; uint256 calldataWords = (calldataSize + 31) / 32; // account for the CALLDATACOPY cost of the proxy contract, including the memory expansion cost startGasLeft += calldataWords * 6 + (calldataWords**2) / 512; diff --git a/src/mocks/BridgeStub.sol b/src/mocks/BridgeStub.sol index 08dca5c4..261cfabf 100644 --- a/src/mocks/BridgeStub.sol +++ b/src/mocks/BridgeStub.sol @@ -139,7 +139,7 @@ contract BridgeStub is IBridge { InOutInfo storage info = allowedDelayedInboxesMap[inbox]; bool alreadyEnabled = info.allowed; emit InboxToggle(inbox, enabled); - if ((alreadyEnabled && enabled) || (!alreadyEnabled && !enabled)) { + if (alreadyEnabled == enabled) { return; } if (enabled) { diff --git a/src/mocks/Simple.sol b/src/mocks/Simple.sol index 1563fdb4..25d4b67b 100644 --- a/src/mocks/Simple.sol +++ b/src/mocks/Simple.sol @@ -43,6 +43,10 @@ contract Simple { return block.number; } + function getBlockDifficulty() external view returns (uint256) { + return block.difficulty; + } + function noop() external pure {} function pleaseRevert() external pure { diff --git a/src/osp/IOneStepProver.sol b/src/osp/IOneStepProver.sol index e26ff604..202d5859 100644 --- a/src/osp/IOneStepProver.sol +++ b/src/osp/IOneStepProver.sol @@ -7,6 +7,7 @@ pragma solidity ^0.8.0; import "../state/Machine.sol"; import "../state/Module.sol"; import "../state/Instructions.sol"; +import "../state/GlobalState.sol"; import "../bridge/ISequencerInbox.sol"; import "../bridge/IBridge.sol"; diff --git a/src/osp/OneStepProverHostIo.sol b/src/osp/OneStepProverHostIo.sol index f4c04c77..f10c7ec1 100644 --- a/src/osp/OneStepProverHostIo.sol +++ b/src/osp/OneStepProverHostIo.sol @@ -8,6 +8,7 @@ import "../state/Value.sol"; import "../state/Machine.sol"; import "../state/MerkleProof.sol"; import "../state/Deserialize.sol"; +import "../state/ModuleMemory.sol"; import "./IOneStepProver.sol"; import "../bridge/Messages.sol"; import "../bridge/IBridge.sol"; diff --git a/src/osp/OneStepProverMemory.sol b/src/osp/OneStepProverMemory.sol index 2031149b..7d396ad3 100644 --- a/src/osp/OneStepProverMemory.sol +++ b/src/osp/OneStepProverMemory.sol @@ -7,6 +7,7 @@ pragma solidity ^0.8.0; import "../state/Value.sol"; import "../state/Machine.sol"; import "../state/Deserialize.sol"; +import "../state/ModuleMemory.sol"; import "./IOneStepProver.sol"; contract OneStepProverMemory is IOneStepProver { diff --git a/src/rollup/Config.sol b/src/rollup/Config.sol new file mode 100644 index 00000000..41950a9b --- /dev/null +++ b/src/rollup/Config.sol @@ -0,0 +1,43 @@ +// Copyright 2021-2022, Offchain Labs, Inc. +// For license information, see https://github.com/nitro/blob/master/LICENSE +// SPDX-License-Identifier: BUSL-1.1 + +pragma solidity ^0.8.0; + +import "../state/GlobalState.sol"; +import "../state/Machine.sol"; +import "../bridge/ISequencerInbox.sol"; +import "../bridge/IBridge.sol"; +import "../bridge/IOutbox.sol"; +import "../bridge/IInbox.sol"; +import "./IRollupEventInbox.sol"; +import "./IRollupLogic.sol"; +import "../challenge/IChallengeManager.sol"; + +struct Config { + uint64 confirmPeriodBlocks; + uint64 extraChallengeTimeBlocks; + address stakeToken; + uint256 baseStake; + bytes32 wasmModuleRoot; + address owner; + address loserStakeEscrow; + uint256 chainId; + string chainConfig; + uint64 genesisBlockNum; + ISequencerInbox.MaxTimeVariation sequencerInboxMaxTimeVariation; +} + +struct ContractDependencies { + IBridge bridge; + ISequencerInbox sequencerInbox; + IInbox inbox; + IOutbox outbox; + IRollupEventInbox rollupEventInbox; + IChallengeManager challengeManager; + address rollupAdminLogic; + IRollupUser rollupUserLogic; + // misc contracts that are useful when interacting with the rollup + address validatorUtils; + address validatorWalletCreator; +} diff --git a/src/rollup/IRollupAdmin.sol b/src/rollup/IRollupAdmin.sol new file mode 100644 index 00000000..8420c29c --- /dev/null +++ b/src/rollup/IRollupAdmin.sol @@ -0,0 +1,138 @@ +// Copyright 2021-2022, Offchain Labs, Inc. +// For license information, see https://github.com/nitro/blob/master/LICENSE +// SPDX-License-Identifier: BUSL-1.1 + +pragma solidity ^0.8.0; + +import "./IRollupCore.sol"; +import "../bridge/ISequencerInbox.sol"; +import "../bridge/IOutbox.sol"; +import "../bridge/IOwnable.sol"; +import "./Config.sol"; + +interface IRollupAdmin { + event OwnerFunctionCalled(uint256 indexed id); + + function initialize(Config calldata config, ContractDependencies calldata connectedContracts) + external; + + /** + * @notice Add a contract authorized to put messages into this rollup's inbox + * @param _outbox Outbox contract to add + */ + function setOutbox(IOutbox _outbox) external; + + /** + * @notice Disable an old outbox from interacting with the bridge + * @param _outbox Outbox contract to remove + */ + function removeOldOutbox(address _outbox) external; + + /** + * @notice Enable or disable an inbox contract + * @param _inbox Inbox contract to add or remove + * @param _enabled New status of inbox + */ + function setDelayedInbox(address _inbox, bool _enabled) external; + + /** + * @notice Pause interaction with the rollup contract + */ + function pause() external; + + /** + * @notice Resume interaction with the rollup contract + */ + function resume() external; + + /** + * @notice Set the addresses of the validator whitelist + * @dev It is expected that both arrays are same length, and validator at + * position i corresponds to the value at position i + * @param _validator addresses to set in the whitelist + * @param _val value to set in the whitelist for corresponding address + */ + function setValidator(address[] memory _validator, bool[] memory _val) external; + + /** + * @notice Set a new owner address for the rollup proxy + * @param newOwner address of new rollup owner + */ + function setOwner(address newOwner) external; + + /** + * @notice Set minimum assertion period for the rollup + * @param newPeriod new minimum period for assertions + */ + function setMinimumAssertionPeriod(uint256 newPeriod) external; + + /** + * @notice Set number of blocks until a node is considered confirmed + * @param newConfirmPeriod new number of blocks until a node is confirmed + */ + function setConfirmPeriodBlocks(uint64 newConfirmPeriod) external; + + /** + * @notice Set number of extra blocks after a challenge + * @param newExtraTimeBlocks new number of blocks + */ + function setExtraChallengeTimeBlocks(uint64 newExtraTimeBlocks) external; + + /** + * @notice Set base stake required for an assertion + * @param newBaseStake maximum avmgas to be used per block + */ + function setBaseStake(uint256 newBaseStake) external; + + /** + * @notice Set the token used for stake, where address(0) == eth + * @dev Before changing the base stake token, you might need to change the + * implementation of the Rollup User logic! + * @param newStakeToken address of token used for staking + */ + function setStakeToken(address newStakeToken) external; + + /** + * @notice Upgrades the implementation of a beacon controlled by the rollup + * @param beacon address of beacon to be upgraded + * @param newImplementation new address of implementation + */ + function upgradeBeacon(address beacon, address newImplementation) external; + + function forceResolveChallenge(address[] memory stackerA, address[] memory stackerB) external; + + function forceRefundStaker(address[] memory stacker) external; + + function forceCreateNode( + uint64 prevNode, + uint256 prevNodeInboxMaxCount, + Assertion memory assertion, + bytes32 expectedNodeHash + ) external; + + function forceConfirmNode( + uint64 nodeNum, + bytes32 blockHash, + bytes32 sendRoot + ) external; + + function setLoserStakeEscrow(address newLoserStakerEscrow) external; + + /** + * @notice Set the proving WASM module root + * @param newWasmModuleRoot new module root + */ + function setWasmModuleRoot(bytes32 newWasmModuleRoot) external; + + /** + * @notice set a new sequencer inbox contract + * @param _sequencerInbox new address of sequencer inbox + */ + function setSequencerInbox(address _sequencerInbox) external; + + /** + * @notice set the validatorWhitelistDisabled flag + * @param _validatorWhitelistDisabled new value of validatorWhitelistDisabled, i.e. true = disabled + */ + function setValidatorWhitelistDisabled(bool _validatorWhitelistDisabled) external; +} diff --git a/src/rollup/IRollupCore.sol b/src/rollup/IRollupCore.sol index f66958a8..e17c27af 100644 --- a/src/rollup/IRollupCore.sol +++ b/src/rollup/IRollupCore.sol @@ -5,7 +5,11 @@ pragma solidity ^0.8.0; import "./Node.sol"; -import "./RollupLib.sol"; +import "../bridge/IBridge.sol"; +import "../bridge/IOutbox.sol"; +import "../bridge/IInbox.sol"; +import "./IRollupEventInbox.sol"; +import "../challenge/IChallengeManager.sol"; interface IRollupCore { struct Staker { @@ -24,7 +28,7 @@ interface IRollupCore { bytes32 indexed parentNodeHash, bytes32 indexed nodeHash, bytes32 executionHash, - RollupLib.Assertion assertion, + Assertion assertion, bytes32 afterInboxBatchAcc, bytes32 wasmModuleRoot, uint256 inboxMaxCount diff --git a/src/rollup/IRollupLogic.sol b/src/rollup/IRollupLogic.sol index 33cfdc51..d413b277 100644 --- a/src/rollup/IRollupLogic.sol +++ b/src/rollup/IRollupLogic.sol @@ -4,7 +4,6 @@ pragma solidity ^0.8.0; -import "./RollupLib.sol"; import "./IRollupCore.sol"; import "../bridge/ISequencerInbox.sol"; import "../bridge/IOutbox.sol"; @@ -28,7 +27,7 @@ interface IRollupUserAbs is IRollupCore, IOwnable { function stakeOnExistingNode(uint64 nodeNum, bytes32 nodeHash) external; function stakeOnNewNode( - RollupLib.Assertion memory assertion, + Assertion memory assertion, bytes32 expectedNodeHash, uint256 prevNodeInboxMaxCount ) external; @@ -75,7 +74,7 @@ interface IRollupUser is IRollupUserAbs { function newStakeOnExistingNode(uint64 nodeNum, bytes32 nodeHash) external payable; function newStakeOnNewNode( - RollupLib.Assertion calldata assertion, + Assertion calldata assertion, bytes32 expectedNodeHash, uint256 prevNodeInboxMaxCount ) external payable; @@ -92,137 +91,10 @@ interface IRollupUserERC20 is IRollupUserAbs { function newStakeOnNewNode( uint256 tokenAmount, - RollupLib.Assertion calldata assertion, + Assertion calldata assertion, bytes32 expectedNodeHash, uint256 prevNodeInboxMaxCount ) external; function addToDeposit(address stakerAddress, uint256 tokenAmount) external; } - -interface IRollupAdmin { - event OwnerFunctionCalled(uint256 indexed id); - - function initialize(Config calldata config, ContractDependencies calldata connectedContracts) - external; - - /** - * @notice Add a contract authorized to put messages into this rollup's inbox - * @param _outbox Outbox contract to add - */ - function setOutbox(IOutbox _outbox) external; - - /** - * @notice Disable an old outbox from interacting with the bridge - * @param _outbox Outbox contract to remove - */ - function removeOldOutbox(address _outbox) external; - - /** - * @notice Enable or disable an inbox contract - * @param _inbox Inbox contract to add or remove - * @param _enabled New status of inbox - */ - function setDelayedInbox(address _inbox, bool _enabled) external; - - /** - * @notice Pause interaction with the rollup contract - */ - function pause() external; - - /** - * @notice Resume interaction with the rollup contract - */ - function resume() external; - - /** - * @notice Set the addresses of the validator whitelist - * @dev It is expected that both arrays are same length, and validator at - * position i corresponds to the value at position i - * @param _validator addresses to set in the whitelist - * @param _val value to set in the whitelist for corresponding address - */ - function setValidator(address[] memory _validator, bool[] memory _val) external; - - /** - * @notice Set a new owner address for the rollup proxy - * @param newOwner address of new rollup owner - */ - function setOwner(address newOwner) external; - - /** - * @notice Set minimum assertion period for the rollup - * @param newPeriod new minimum period for assertions - */ - function setMinimumAssertionPeriod(uint256 newPeriod) external; - - /** - * @notice Set number of blocks until a node is considered confirmed - * @param newConfirmPeriod new number of blocks until a node is confirmed - */ - function setConfirmPeriodBlocks(uint64 newConfirmPeriod) external; - - /** - * @notice Set number of extra blocks after a challenge - * @param newExtraTimeBlocks new number of blocks - */ - function setExtraChallengeTimeBlocks(uint64 newExtraTimeBlocks) external; - - /** - * @notice Set base stake required for an assertion - * @param newBaseStake maximum avmgas to be used per block - */ - function setBaseStake(uint256 newBaseStake) external; - - /** - * @notice Set the token used for stake, where address(0) == eth - * @dev Before changing the base stake token, you might need to change the - * implementation of the Rollup User logic! - * @param newStakeToken address of token used for staking - */ - function setStakeToken(address newStakeToken) external; - - /** - * @notice Upgrades the implementation of a beacon controlled by the rollup - * @param beacon address of beacon to be upgraded - * @param newImplementation new address of implementation - */ - function upgradeBeacon(address beacon, address newImplementation) external; - - function forceResolveChallenge(address[] memory stackerA, address[] memory stackerB) external; - - function forceRefundStaker(address[] memory stacker) external; - - function forceCreateNode( - uint64 prevNode, - uint256 prevNodeInboxMaxCount, - RollupLib.Assertion memory assertion, - bytes32 expectedNodeHash - ) external; - - function forceConfirmNode( - uint64 nodeNum, - bytes32 blockHash, - bytes32 sendRoot - ) external; - - function setLoserStakeEscrow(address newLoserStakerEscrow) external; - - /** - * @notice Set the proving WASM module root - * @param newWasmModuleRoot new module root - */ - function setWasmModuleRoot(bytes32 newWasmModuleRoot) external; - - /** - * @notice set a new sequencer inbox contract - * @param _sequencerInbox new address of sequencer inbox - */ - function setSequencerInbox(address _sequencerInbox) external; - - /** - * @notice set the validatorWhitelistDisabled flag - * @param _validatorWhitelistDisabled new value of validatorWhitelistDisabled, i.e. true = disabled - */ - function setValidatorWhitelistDisabled(bool _validatorWhitelistDisabled) external; -} diff --git a/src/rollup/Node.sol b/src/rollup/Node.sol index 78bb19a1..a5a50270 100644 --- a/src/rollup/Node.sol +++ b/src/rollup/Node.sol @@ -4,6 +4,20 @@ pragma solidity ^0.8.0; +import "../state/GlobalState.sol"; +import "../state/Machine.sol"; + +struct ExecutionState { + GlobalState globalState; + MachineStatus machineStatus; +} + +struct Assertion { + ExecutionState beforeState; + ExecutionState afterState; + uint64 numBlocks; +} + struct Node { // Hash of the state of the chain as of this node bytes32 stateHash; diff --git a/src/rollup/RollupAdminLogic.sol b/src/rollup/RollupAdminLogic.sol index dd813fd5..44a9989c 100644 --- a/src/rollup/RollupAdminLogic.sol +++ b/src/rollup/RollupAdminLogic.sol @@ -4,7 +4,8 @@ pragma solidity ^0.8.0; -import {IRollupAdmin, IRollupUser} from "./IRollupLogic.sol"; +import "./IRollupAdmin.sol"; +import "./IRollupLogic.sol"; import "./RollupCore.sol"; import "../bridge/IOutbox.sol"; import "../bridge/ISequencerInbox.sol"; @@ -76,7 +77,7 @@ contract RollupAdminLogic is RollupCore, IRollupAdmin, DoubleLogicUUPSUpgradeabl function createInitialNode() private view returns (Node memory) { GlobalState memory emptyGlobalState; bytes32 state = RollupLib.stateHashMem( - RollupLib.ExecutionState(emptyGlobalState, MachineStatus.FINISHED), + ExecutionState(emptyGlobalState, MachineStatus.FINISHED), 1 // inboxMaxCount - force the first assertion to read a message ); return @@ -285,7 +286,7 @@ contract RollupAdminLogic is RollupCore, IRollupAdmin, DoubleLogicUUPSUpgradeabl function forceCreateNode( uint64 prevNode, uint256 prevNodeInboxMaxCount, - RollupLib.Assertion calldata assertion, + Assertion calldata assertion, bytes32 expectedNodeHash ) external override whenPaused { require(prevNode == latestConfirmed(), "ONLY_LATEST_CONFIRMED"); @@ -340,10 +341,7 @@ contract RollupAdminLogic is RollupCore, IRollupAdmin, DoubleLogicUUPSUpgradeabl emit OwnerFunctionCalled(28); } - function createNitroMigrationGenesis(RollupLib.Assertion calldata assertion) - external - whenPaused - { + function createNitroMigrationGenesis(Assertion calldata assertion) external whenPaused { bytes32 expectedSendRoot = bytes32(0); uint64 expectedInboxCount = 1; diff --git a/src/rollup/RollupCore.sol b/src/rollup/RollupCore.sol index f96f609b..d12fb149 100644 --- a/src/rollup/RollupCore.sol +++ b/src/rollup/RollupCore.sol @@ -7,7 +7,6 @@ pragma solidity ^0.8.0; import "@openzeppelin/contracts-upgradeable/security/PausableUpgradeable.sol"; import "./Node.sol"; -import "./IRollupCore.sol"; import "./RollupLib.sol"; import "./IRollupEventInbox.sol"; import "./IRollupCore.sol"; @@ -20,6 +19,7 @@ import "../bridge/IOutbox.sol"; import "../precompiles/ArbSys.sol"; +import "../libraries/ArbitrumChecker.sol"; import {NO_CHAL_INDEX} from "../libraries/Constants.sol"; abstract contract RollupCore is IRollupCore, PausableUpgradeable { @@ -79,17 +79,10 @@ abstract contract RollupCore is IRollupCore, PausableUpgradeable { bool public validatorWhitelistDisabled; // If the chain this RollupCore is deployed on is an Arbitrum chain. - bool internal immutable _hostChainIsArbitrum; + bool internal immutable _hostChainIsArbitrum = ArbitrumChecker.runningOnArbitrum(); // If the chain RollupCore is deployed on, this will contain the ArbSys.blockNumber() at each node's creation. mapping(uint64 => uint256) internal _nodeCreatedAtArbSysBlock; - constructor() { - (bool ok, bytes memory data) = address(100).staticcall( - abi.encodeWithSelector(ArbSys.arbOSVersion.selector) - ); - _hostChainIsArbitrum = ok && data.length == 32; - } - /** * @notice Get a storage reference to the Node for the given node index * @param nodeNum Index of the node @@ -568,7 +561,7 @@ abstract contract RollupCore is IRollupCore, PausableUpgradeable { } function createNewNode( - RollupLib.Assertion calldata assertion, + Assertion calldata assertion, uint64 prevNodeNum, uint256 prevNodeInboxMaxCount, bytes32 expectedNodeHash diff --git a/src/rollup/RollupCreator.sol b/src/rollup/RollupCreator.sol index ed086560..58ef0909 100644 --- a/src/rollup/RollupCreator.sol +++ b/src/rollup/RollupCreator.sol @@ -52,82 +52,70 @@ contract RollupCreator is Ownable { emit TemplatesUpdated(); } - struct CreateRollupFrame { - ProxyAdmin admin; - IBridge bridge; - ISequencerInbox sequencerInbox; - IInbox inbox; - IRollupEventInbox rollupEventInbox; - IOutbox outbox; - RollupProxy rollup; - } - // After this setup: // Rollup should be the owner of bridge // RollupOwner should be the owner of Rollup's ProxyAdmin // RollupOwner should be the owner of Rollup // Bridge should have a single inbox and outbox - function createRollup(Config memory config, address expectedRollupAddr) - external - returns (address) - { - CreateRollupFrame memory frame; - frame.admin = new ProxyAdmin(); + function createRollup(Config memory config) external returns (address) { + ProxyAdmin proxyAdmin = new ProxyAdmin(); + + // Create the rollup proxy to figure out the address and initialize it later + RollupProxy rollup = new RollupProxy{salt: keccak256(abi.encode(config))}(); ( - frame.bridge, - frame.sequencerInbox, - frame.inbox, - frame.rollupEventInbox, - frame.outbox + IBridge bridge, + ISequencerInbox sequencerInbox, + IInbox inbox, + IRollupEventInbox rollupEventInbox, + IOutbox outbox ) = bridgeCreator.createBridge( - address(frame.admin), - expectedRollupAddr, - config.sequencerInboxMaxTimeVariation - ); + address(proxyAdmin), + address(rollup), + config.sequencerInboxMaxTimeVariation + ); - frame.admin.transferOwnership(config.owner); + proxyAdmin.transferOwnership(config.owner); IChallengeManager challengeManager = IChallengeManager( address( new TransparentUpgradeableProxy( address(challengeManagerTemplate), - address(frame.admin), + address(proxyAdmin), "" ) ) ); challengeManager.initialize( - IChallengeResultReceiver(expectedRollupAddr), - frame.sequencerInbox, - frame.bridge, + IChallengeResultReceiver(address(rollup)), + sequencerInbox, + bridge, osp ); - frame.rollup = new RollupProxy( + rollup.initializeProxy( config, ContractDependencies({ - bridge: frame.bridge, - sequencerInbox: frame.sequencerInbox, - inbox: frame.inbox, - outbox: frame.outbox, - rollupEventInbox: frame.rollupEventInbox, + bridge: bridge, + sequencerInbox: sequencerInbox, + inbox: inbox, + outbox: outbox, + rollupEventInbox: rollupEventInbox, challengeManager: challengeManager, - rollupAdminLogic: rollupAdminLogic, + rollupAdminLogic: address(rollupAdminLogic), rollupUserLogic: rollupUserLogic, validatorUtils: validatorUtils, validatorWalletCreator: validatorWalletCreator }) ); - require(address(frame.rollup) == expectedRollupAddr, "WRONG_ROLLUP_ADDR"); emit RollupCreated( - address(frame.rollup), - address(frame.inbox), - address(frame.admin), - address(frame.sequencerInbox), - address(frame.bridge) + address(rollup), + address(inbox), + address(proxyAdmin), + address(sequencerInbox), + address(bridge) ); - return address(frame.rollup); + return address(rollup); } } diff --git a/src/rollup/RollupEventInbox.sol b/src/rollup/RollupEventInbox.sol index 9d7353fc..3a56288d 100644 --- a/src/rollup/RollupEventInbox.sol +++ b/src/rollup/RollupEventInbox.sol @@ -7,7 +7,9 @@ pragma solidity ^0.8.0; import "./IRollupEventInbox.sol"; import "../bridge/IBridge.sol"; import "../bridge/IDelayedMessageProvider.sol"; +import "../precompiles/ArbGasInfo.sol"; import "../libraries/DelegateCallAware.sol"; +import "../libraries/ArbitrumChecker.sol"; import {INITIALIZATION_MSG_TYPE} from "../libraries/MessageTypes.sol"; import {AlreadyInit, HadZeroInit} from "../libraries/Error.sol"; @@ -36,7 +38,17 @@ contract RollupEventInbox is IRollupEventInbox, IDelayedMessageProvider, Delegat onlyRollup { require(bytes(chainConfig).length > 0, "EMPTY_CHAIN_CONFIG"); - bytes memory initMsg = abi.encodePacked(chainId, uint8(0), chainConfig); + uint8 initMsgVersion = 1; + uint256 currentDataCost = block.basefee; + if (ArbitrumChecker.runningOnArbitrum()) { + currentDataCost += ArbGasInfo(address(0x6c)).getL1BaseFeeEstimate(); + } + bytes memory initMsg = abi.encodePacked( + chainId, + initMsgVersion, + currentDataCost, + chainConfig + ); uint256 num = bridge.enqueueDelayedMessage( INITIALIZATION_MSG_TYPE, address(0), diff --git a/src/rollup/RollupLib.sol b/src/rollup/RollupLib.sol index a46ec42c..8120fe83 100644 --- a/src/rollup/RollupLib.sol +++ b/src/rollup/RollupLib.sol @@ -12,45 +12,12 @@ import "../bridge/ISequencerInbox.sol"; import "../bridge/IBridge.sol"; import "../bridge/IOutbox.sol"; import "../bridge/IInbox.sol"; +import "./Node.sol"; import "./IRollupEventInbox.sol"; -import "./IRollupLogic.sol"; - -struct Config { - uint64 confirmPeriodBlocks; - uint64 extraChallengeTimeBlocks; - address stakeToken; - uint256 baseStake; - bytes32 wasmModuleRoot; - address owner; - address loserStakeEscrow; - uint256 chainId; - string chainConfig; - uint64 genesisBlockNum; - ISequencerInbox.MaxTimeVariation sequencerInboxMaxTimeVariation; -} - -struct ContractDependencies { - IBridge bridge; - ISequencerInbox sequencerInbox; - IInbox inbox; - IOutbox outbox; - IRollupEventInbox rollupEventInbox; - IChallengeManager challengeManager; - IRollupAdmin rollupAdminLogic; - IRollupUser rollupUserLogic; - // misc contracts that are useful when interacting with the rollup - address validatorUtils; - address validatorWalletCreator; -} library RollupLib { using GlobalStateLib for GlobalState; - struct ExecutionState { - GlobalState globalState; - MachineStatus machineStatus; - } - function stateHash(ExecutionState calldata execState, uint256 inboxMaxCount) internal pure @@ -82,12 +49,6 @@ library RollupLib { ); } - struct Assertion { - ExecutionState beforeState; - ExecutionState afterState; - uint64 numBlocks; - } - function executionHash(Assertion memory assertion) internal pure returns (bytes32) { MachineStatus[2] memory statuses; statuses[0] = assertion.beforeState.machineStatus; diff --git a/src/rollup/RollupProxy.sol b/src/rollup/RollupProxy.sol index 04531be6..110dd399 100644 --- a/src/rollup/RollupProxy.sol +++ b/src/rollup/RollupProxy.sol @@ -5,16 +5,31 @@ pragma solidity ^0.8.0; import "../libraries/AdminFallbackProxy.sol"; -import "./IRollupLogic.sol"; +import "./IRollupAdmin.sol"; +import "./Config.sol"; contract RollupProxy is AdminFallbackProxy { - constructor(Config memory config, ContractDependencies memory connectedContracts) - AdminFallbackProxy( - address(connectedContracts.rollupAdminLogic), - abi.encodeWithSelector(IRollupAdmin.initialize.selector, config, connectedContracts), - address(connectedContracts.rollupUserLogic), - abi.encodeWithSelector(IRollupUserAbs.initialize.selector, config.stakeToken), - config.owner - ) - {} + function initializeProxy(Config memory config, ContractDependencies memory connectedContracts) + external + { + if ( + _getAdmin() == address(0) && + _getImplementation() == address(0) && + _getSecondaryImplementation() == address(0) + ) { + _initialize( + address(connectedContracts.rollupAdminLogic), + abi.encodeWithSelector( + IRollupAdmin.initialize.selector, + config, + connectedContracts + ), + address(connectedContracts.rollupUserLogic), + abi.encodeWithSelector(IRollupUserAbs.initialize.selector, config.stakeToken), + config.owner + ); + } else { + _fallback(); + } + } } diff --git a/src/rollup/RollupUserLogic.sol b/src/rollup/RollupUserLogic.sol index c1d307dd..00145fa2 100644 --- a/src/rollup/RollupUserLogic.sol +++ b/src/rollup/RollupUserLogic.sol @@ -9,6 +9,8 @@ import "@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol"; import {IRollupUser} from "./IRollupLogic.sol"; import "../libraries/UUPSNotUpgradeable.sol"; import "./RollupCore.sol"; +import "./IRollupLogic.sol"; + import {ETH_POS_BLOCK_TIME} from "../libraries/Constants.sol"; abstract contract AbsRollupUserLogic is @@ -189,7 +191,7 @@ abstract contract AbsRollupUserLogic is * @param expectedNodeHash The hash of the node being created (protects against reorgs) */ function stakeOnNewNode( - RollupLib.Assertion calldata assertion, + Assertion calldata assertion, bytes32 expectedNodeHash, uint256 prevNodeInboxMaxCount ) public onlyValidator whenNotPaused { @@ -632,7 +634,7 @@ contract RollupUserLogic is AbsRollupUserLogic, IRollupUser { * @param prevNodeInboxMaxCount Total of messages in the inbox as of the previous node */ function newStakeOnNewNode( - RollupLib.Assertion calldata assertion, + Assertion calldata assertion, bytes32 expectedNodeHash, uint256 prevNodeInboxMaxCount ) external payable override { @@ -701,7 +703,7 @@ contract ERC20RollupUserLogic is AbsRollupUserLogic, IRollupUserERC20 { */ function newStakeOnNewNode( uint256 tokenAmount, - RollupLib.Assertion calldata assertion, + Assertion calldata assertion, bytes32 expectedNodeHash, uint256 prevNodeInboxMaxCount ) external override { diff --git a/src/rollup/ValidatorUtils.sol b/src/rollup/ValidatorUtils.sol index 561cc1c2..a7bb7d9d 100644 --- a/src/rollup/ValidatorUtils.sol +++ b/src/rollup/ValidatorUtils.sol @@ -8,6 +8,7 @@ pragma experimental ABIEncoderV2; import "../rollup/IRollupCore.sol"; import "../challenge/IChallengeManager.sol"; +import "./IRollupLogic.sol"; import {NO_CHAL_INDEX} from "../libraries/Constants.sol"; diff --git a/src/state/Deserialize.sol b/src/state/Deserialize.sol index adcf8530..e63e4930 100644 --- a/src/state/Deserialize.sol +++ b/src/state/Deserialize.sol @@ -11,7 +11,7 @@ import "./Instructions.sol"; import "./StackFrame.sol"; import "./GuardStack.sol"; import "./MerkleProof.sol"; -import "./ModuleMemory.sol"; +import "./ModuleMemoryCompact.sol"; import "./Module.sol"; import "./GlobalState.sol"; diff --git a/src/state/Module.sol b/src/state/Module.sol index 1a515a1d..4dc4cf5c 100644 --- a/src/state/Module.sol +++ b/src/state/Module.sol @@ -4,7 +4,7 @@ pragma solidity ^0.8.0; -import "./ModuleMemory.sol"; +import "./ModuleMemoryCompact.sol"; struct Module { bytes32 globalsMerkleRoot; @@ -15,7 +15,7 @@ struct Module { } library ModuleLib { - using ModuleMemoryLib for ModuleMemory; + using ModuleMemoryCompactLib for ModuleMemory; function hash(Module memory mod) internal pure returns (bytes32) { return diff --git a/src/state/ModuleMemory.sol b/src/state/ModuleMemory.sol index f3efb501..f240d354 100644 --- a/src/state/ModuleMemory.sol +++ b/src/state/ModuleMemory.sol @@ -6,12 +6,7 @@ pragma solidity ^0.8.0; import "./MerkleProof.sol"; import "./Deserialize.sol"; - -struct ModuleMemory { - uint64 size; - uint64 maxSize; - bytes32 merkleRoot; -} +import "./ModuleMemoryCompact.sol"; library ModuleMemoryLib { using MerkleProofLib for MerkleProof; @@ -19,7 +14,7 @@ library ModuleMemoryLib { uint256 private constant LEAF_SIZE = 32; function hash(ModuleMemory memory mem) internal pure returns (bytes32) { - return keccak256(abi.encodePacked("Memory:", mem.size, mem.maxSize, mem.merkleRoot)); + return ModuleMemoryCompactLib.hash(mem); } function proveLeaf( diff --git a/src/state/ModuleMemoryCompact.sol b/src/state/ModuleMemoryCompact.sol new file mode 100644 index 00000000..935fdf38 --- /dev/null +++ b/src/state/ModuleMemoryCompact.sol @@ -0,0 +1,17 @@ +// Copyright 2021-2022, Offchain Labs, Inc. +// For license information, see https://github.com/nitro/blob/master/LICENSE +// SPDX-License-Identifier: BUSL-1.1 + +pragma solidity ^0.8.0; + +struct ModuleMemory { + uint64 size; + uint64 maxSize; + bytes32 merkleRoot; +} + +library ModuleMemoryCompactLib { + function hash(ModuleMemory memory mem) internal pure returns (bytes32) { + return keccak256(abi.encodePacked("Memory:", mem.size, mem.maxSize, mem.merkleRoot)); + } +} diff --git a/test/contract/arbRollup.spec.ts b/test/contract/arbRollup.spec.ts index f9b35379..ad71ad1e 100644 --- a/test/contract/arbRollup.spec.ts +++ b/test/contract/arbRollup.spec.ts @@ -50,9 +50,8 @@ import { forceCreateNode, assertionEquals, } from './common/rolluplib' -import { RollupLib } from '../../build/types/src/rollup/RollupUserLogic.sol/RollupUserLogic' -type AssertionStruct = RollupLib.AssertionStruct -type ExecutionStateStruct = RollupLib.ExecutionStateStruct +import { AssertionStruct } from '../../build/types/src/rollup/RollupCore' +import { ExecutionStateStruct } from '../../build/types/src/rollup/RollupCore' import { keccak256 } from 'ethers/lib/utils' import { ConfigStruct, @@ -183,18 +182,7 @@ const setup = async () => { ethers.constants.AddressZero ) - const nonce = await rollupCreator.signer.provider!.getTransactionCount( - rollupCreator.address - ) - const expectedRollupAddress = ethers.utils.getContractAddress({ - from: rollupCreator.address, - nonce: nonce + 2, - }) - - const response = await rollupCreator.createRollup( - await getDefaultConfig(), - expectedRollupAddress - ) + const response = await rollupCreator.createRollup(await getDefaultConfig()) const rec = await response.wait() const rollupCreatedEvent = rollupCreator.interface.parseLog( @@ -202,10 +190,10 @@ const setup = async () => { ).args as RollupCreatedEvent['args'] const rollupAdmin = rollupAdminLogicFac - .attach(expectedRollupAddress) + .attach(rollupCreatedEvent.rollupAddress) .connect(rollupCreator.signer) const rollupUser = rollupUserLogicFac - .attach(expectedRollupAddress) + .attach(rollupCreatedEvent.rollupAddress) .connect(user) await rollupAdmin.setValidator( diff --git a/test/contract/common/rolluplib.ts b/test/contract/common/rolluplib.ts index 2f19859b..4ef9c10a 100644 --- a/test/contract/common/rolluplib.ts +++ b/test/contract/common/rolluplib.ts @@ -10,12 +10,9 @@ import { RollupAdminLogic, SequencerInbox, } from '../../../build/types' -import { - RollupLib, - NodeCreatedEvent, -} from '../../../build/types/src/rollup/RollupUserLogic.sol/RollupUserLogic' -type AssertionStruct = RollupLib.AssertionStruct -type ExecutionStateStruct = RollupLib.ExecutionStateStruct +import { NodeCreatedEvent } from '../../../build/types/src/rollup/RollupUserLogic.sol/RollupUserLogic' +import { AssertionStruct } from '../../../build/types/src/rollup/RollupCore' +import { ExecutionStateStruct } from '../../../build/types/src/rollup/RollupCore' import { blockStateHash, hashChallengeState } from './challengeLib' import * as globalStateLib from './globalStateLib' import { constants } from 'ethers' diff --git a/test/storage/SequencerInbox.dot b/test/storage/SequencerInbox.dot index 81ca079d..f9ea05ab 100644 --- a/test/storage/SequencerInbox.dot +++ b/test/storage/SequencerInbox.dot @@ -4,7 +4,7 @@ rankdir=LR color=black arrowhead=open node [shape=record, style=filled, fillcolor=gray95 fontname="Courier New"] -3 [label="SequencerInbox \<\\>\n | {{ slot| 0 | 1 | 2 | 3 | 4-7 | 8 } | { type: \.variable (bytes) | { uint256: totalDelayedMessagesRead (32) } | { unallocated (12) | IBridge: bridge (20) } | { unallocated (12) | IOwnable: rollup (20) } | { mapping\(address=\>bool\): isBatchPoster (32) } | { <9> ISequencerInbox.MaxTimeVariation: maxTimeVariation (128) } | { <12> mapping\(bytes32=\>DasKeySetInfo\): dasKeySetInfo (32) }}}"] +3 [label="SequencerInbox \<\\>\n | {{ slot| 0 | 1 | 2 | 3 | 4-7 | 8 | 9 } | { type: \.variable (bytes) | { uint256: totalDelayedMessagesRead (32) } | { unallocated (12) | IBridge: bridge (20) } | { unallocated (12) | IOwnable: rollup (20) } | { mapping\(address=\>bool\): isBatchPoster (32) } | { <9> ISequencerInbox.MaxTimeVariation: maxTimeVariation (128) } | { <12> mapping\(bytes32=\>DasKeySetInfo\): dasKeySetInfo (32) } | { mapping\(address=\>bool\): isSequencer (32) }}}"] 1 [label="ISequencerInbox.MaxTimeVariation \<\\>\n | {{ slot| 4 | 5 | 6 | 7 } | { type: variable (bytes) | { uint256: MaxTimeVariation.delayBlocks (32) } | { uint256: MaxTimeVariation.futureBlocks (32) } | { uint256: MaxTimeVariation.delaySeconds (32) } | { uint256: MaxTimeVariation.futureSeconds (32) }}}"]