There is no commentSelect some text and then click Comment, or simply add a comment to this page from below to start a discussion.
Superchain Shared Sequencer V2
TLDR; Below is a short proposal for a simple shared-sequencer design that enables synchronous cross-chain transactions with a minimal diff to the current Optimism architecture.
Design Goals
Enable seamless atomic mint/burn of assets within the superchain to enable a unified native asset layer for the Superchain
Minimal diff to the current Optimism architecture to reduce technical complexity
Context
Let and denote Optimism and Base transactions respectively. Currently, the Optimism and Base sequencer commit to an ordering of transactions:
We want to enable synchronous cross-chain transactions between Optimism and Base. In other words, an arbitrary transaction should have the power to conditionally include a transaction in if and only if succeeded. Specifically, it should satisfy the following properties:
If executes succesfully, gets included in
If reverts, does not get included in
and should happen at the same "time"
Note that given an arbitrary transaction , it is not possible to know whether the transaction will execute successfully or that it will attempt to emit a cross-chain transaction without execution. Thus, we believe that it is not possible to enable synchronous cross-chain communication without some form of execution at the sequencer level (Espresso and Astria do not execute transactions, so they only enable atomic inclusion of transactions).
Here is a simple proposal for a shared sequencer that would enable synchronous cross-chain transactions between Optimism and Base, under a shared trust model.
Instead of routing their transactions to the rollup-specific sequencer, users will send their transactions to the shared sequencer.
Upon receiving a transaction , the shared sequencer will:
Observe the chain id and sequence to be included in or respectively
Upon receiving a shared transaction , the shared sequencer will:
Observe the chain id and request an execution trace for on the respective chain
If the execution trace denotes that the transaction sucessfully executed and requested a cross-chain transaction to be included, the sequencer will include into the requested chain.
How do we make sure the sequencer was not malicious?
To ensure that the sequencer was not malicious, assume a few smart contracts have been implemented on-chain for sending these cross-chain transactions.
Note that we implement mint-and-burn for simplicity, but it could be generalized to arbitrary messaging.
Then, for every batch posted on L1, it suffices to prove that the mint and burn Merkle trees have the same root. This can be done using a fraud proof which would require more latency, a ZK proof, or a simple Merkle proof. The design space here is open.
If the roots do not match, either the sequencer is posting fradulent state roots OR the sequencer failed to include a cross-chain transaction.
If the roots do match, but it's not supposed to, then the execution fraud proof system will catch the incorrectly executed transactions.
Note that a Base full node must now run a Optimism full node as well and vice-versa. Certain transactions on Base should only be included if a transaction successfully executed on Optimism. This is the price you pay for atomic composability.
This design is very similar to the existing Optimism sequencer/fraud proof design, but adds 1 extra condition to validate when posting state roots. The fundamental difference between this architecture and other proposals for shared/decentralized sequencers is the following:
Our model:
enables atomic composability across all chains that are being sequenced (i.e. mint/burn for native assets across all members of the superchain)
sequencer has to run execution as well as sequencing, but the execution is sharded across the different chains
superchain specific architecture
Other shared sequencing models (e.g., Espresso or Astria)
Only allows for atomic transaction inclusion, but not conditional inclusion based on whether a tx executes succesfully
sequencer doesn't have to run execution (execution is done lazily after sequencing is posted to L1)
Open questions
How to handle L1 exit
Mostly seems to work out of the box, but some edge cases need to be handled.
How to ensure they get included at the same "time"
If we're okay with "time" being at the batch level, the current design works
If we want more granularity, we may need to add a global tx index