|
| 1 | +# Signals |
| 2 | + |
| 3 | +The minimal rollup stack refers to cross-chain message passing as "signalling". Signalling involves writing to a storage location so that it can later be proven on a **destination chain** using a storage proof for a trusted **source chain** state root. |
| 4 | + |
| 5 | +A storage proof is a [Merkle proof](https://en.wikipedia.org/wiki/Merkle_tree) showing that a value is stored at a specific storage slot. This proof is generated by the **source chain** and can be verified by the **destination chain** as long as the root hash of the **source chain** state is available on the **destination chain**. |
| 6 | + |
| 7 | +A signal has the following properties: |
| 8 | + |
| 9 | +- Scoped to a specific chain and account: The same signal will generate a different storage slot on different chains. This ties the account and state root to the chain. |
| 10 | +- Contains a `bytes32` value: May represent a hash of a message or any other value type that can be stored in a single EVM word. |
| 11 | +- Can't be deleted: Once a signal is stored, it remains in storage indefinitely. |
| 12 | + |
| 13 | +For the purpose of deriving the storage slot for a signal, we use the [ERC-7201](https://github.com/OpenZeppelin/openzeppelin-contracts/blob/ca7a4e39de0860bbaadf95824207886e6de9fa64/contracts/utils/SlotDerivation.sol#L4) standard formula to create a unique storage slot for each signal that does not collide with other values stored in the same contract. The id of the signal consists of `msg.sender`, `chainId` and `value`. |
| 14 | + |
| 15 | +### L1 -> L2 Same-slot signalling |
| 16 | + |
| 17 | +Nethermind's [proposal on same-slot L1 -> L2 message passing](https://ethresear.ch/t/same-slot-l1-l2-message-passing/21186) describes a process for passing messages (aka. signals) from the L1 to the L2 by using storage proofs. However, it also outlines the concept of signals injected by the L2 proposer into the L2 in the same slot, allowing to bypass the wait for the next L1 state root. The following describes the approach to implement both 'slow' and 'fast' signalling pathways using this minimal rollup stack. |
| 18 | + |
| 19 | +#### Normal ('slow') Signalling |
| 20 | + |
| 21 | +The simplest form of signalling requires waiting for the next **source chain** state root to be available on the **destination chain** via a trusted mechanism. In the case of L1 → L2 communication, the latest available L1 root is included in the L2 batch by the proposer in an anchor transaction. |
| 22 | + |
| 23 | +Using the signal requires the following steps: |
| 24 | + |
| 25 | +1. Store a signal on the L1. |
| 26 | +2. Wait for the next L2 publication to be proposed on the L1. |
| 27 | +3. Verify the signal in your L2 application using the latest L1 root to trigger some logic. |
| 28 | + |
| 29 | +However, the waiting period forbids immediate communication between the two chains since users need to wait for the next L1 state root to be available. This is where the 'fast' signalling pathway comes in. |
| 30 | + |
| 31 | +#### Same Slot L1 → L2 ('fast') Signalling |
| 32 | + |
| 33 | +In the 'fast' signalling pathway, the L2 proposer can assist an application to bypass the waiting period for the next L1 state root by injecting the signal into the L2 chain. This is done by the L2 proposer listening to the L1 signal and including it in the L2 block proposal. |
| 34 | + |
| 35 | +Using the signal requires the following steps: |
| 36 | + |
| 37 | +1. An application on the L1 stores a signal in the Signal Service (SS). |
| 38 | +2. The L2 proposer listens to any signals being stored on the L1 SS. |
| 39 | +3. After building a block and obtaining a commitment, the L2 proposer submits a transaction batch to their L1 inbox contract. This batch includes an anchor transaction where the proposer injects the L1 state root and the signals that were stored on the L1 SS. |
| 40 | +4. The inbox contract verifies that the signals exist in the L1 SS using a storage proof with the L1 state root that was just injected. |
| 41 | +5. The L2 proposer calls `receiveSignal` on the L2 SS through the anchor transaction executed at the start of L2 blocks. This fills a mapping of `_receivedSignals` to true (signal —> true). |
| 42 | +6. Any application that requires same slot signalling will just verify the signal is found in the `_receivedSignals` mapping (i.e. they don’t need to provide a storage proof). |
| 43 | + |
| 44 | + |
| 45 | + |
| 46 | +A corner case arises when an L2 proposer chooses not to include one of the signals stored in the L1 SS within the anchor transaction. This omission can cause any application relying on that signal (as in step 6) to revert. In those cases, the application can always fallback to the 'slow' signalling pathway given the signal can't be deleted and the user can always wait for the next L1 state root. Applications can build mechanisms to incentivize proposers to include their signals in the L2 anchor transaction. |
| 47 | + |
| 48 | +In the context of a deterministic L2 proposer mechanism, then the inclusion of the signal in the L2 block is guaranteed. Otherwise, the verification at step 4 will fail, reverting the whole L2 batch. For based rollups, the L2 proposer might issue preconfirmations to guarantee the inclusion of the L2 transaction that depends on the signal. |
| 49 | + |
| 50 | +Considering that [_relying on same-slot L1 messages for L2 execution creates a tight coupling between L1 and L2_](http://ethresear.ch/t/same-slot-l1-l2-message-passing/21186#p-51604-are-shared-sequencersbuilders-needed-here-7) as they would _reorg_ together, it's recommended for applications developers to build mechanisms that are resilient to reorgs. However, the ability to create atomic L1 <> L2 execution remains, as they can always write applications that enforce atomicity (e.g. [fast (and slow) L2 -> L1 withdrawals](https://ethresear.ch/t/fast-and-slow-l2-l1-withdrawals/21161)). |
| 51 | + |
| 52 | +--- |
| 53 | + |
| 54 | +## Application Examples |
| 55 | + |
| 56 | +### State root fast-tracking |
| 57 | + |
| 58 | +An application that requires immediate access to the latest L1 state root can use the fast signalling to fast-track it into the L2 in a permissioned way. The mechanisms to deal with an invalid or untrusted state roots are left to the application. |
| 59 | + |
| 60 | +For example, enabling the anchor transaction can be implemented as an application of a signal itself. |
| 61 | + |
| 62 | +### Asset Bridging |
| 63 | + |
| 64 | +Bridging assets over to L2 requires an application that signals the value and asset to be bridged. On the source chain, some mechanism will lock the asset and produce a signal. On the L2, the user can claim the assets by verifying the signal and the L1 state root. |
| 65 | + |
| 66 | +#### ETH Bridge |
| 67 | + |
| 68 | +The L2 ETH bridge is presumed to have unlimited ETH, functioning as the canonical bridge for transferring assets to Taiko rollups. The ETH bridge makes use of both 'fast' and 'slow' signalling pathways as described above, allowing users to choose between immediate transfers with potential higher costs and slower transfers with lower fees, depending on their specific needs. |
| 69 | + |
| 70 | +If isolated, the bridge can act as a cross-chain call application, allowing users to pass arbitrary calldata along with the value. This can be used to interact with other applications on the L2 or L1. This approach require applications to consider that the bridge can't hold permissions or tokens. |
| 71 | + |
| 72 | +**Slow ETH deposits** |
| 73 | + |
| 74 | +Below is a diagram showing the 'slow' ETH bridging pathway. The L2 ETH bridge requires the latest L1 state root (containing the L1 signal) to verify the L1 storage proof. |
| 75 | + |
| 76 | + |
| 77 | + |
| 78 | +**Fast ETH deposits** |
| 79 | + |
| 80 | +In contrast here is diagram showing the 'fast' signalling pathway. In this scenario the L2 ETH bridge can just verify that the signal is present in the L2 `receivedSignal` mapping instead of relying on the L1 state root. This mapping would have already been filled when the block was proposed. |
| 81 | + |
| 82 | + |
| 83 | + |
| 84 | +In either case however, the message being stored consist of the `destinationChainId`, `nonce` (to prevent storage collision), `from` address, `to` address and `value` (amount of ETH). These values form a `ValueTicket`. The `ValueTicket` is hashed and stored as the `value` of the signal. |
| 85 | + |
| 86 | +### Token Bridges |
| 87 | + |
| 88 | +The current repository does not include token-specific bridges (as these are more application-specific and not considered 'core' to the protocol). However, these would presumably use the same mechanisms as the ETH bridge, leveraging both fast and slow signaling as appropriate for different token transfer scenarios. |
| 89 | + |
| 90 | +Popular token bridges (e.g. erc20, erc721) will perhaps be added at a later date. |
| 91 | + |
| 92 | +## Future Considerations |
| 93 | + |
| 94 | +- **Merging fast and slow pathways in the SS**: Creating a unified signal service that deals with both the fast and slow signals. |
| 95 | +- **L2 → L1 fast withdrawals using solver [network](https://ethresear.ch/t/fast-and-slow-l2-l1-withdrawals/21161)**: Implementing a network of solvers which could dramatically improve withdrawal times from L2 to L1. |
| 96 | +- **Signal storage optimisation**s: Currently, signals cannot be cancelled and are stored indefinitely. We could explore allowing cancellable signals or reusing storage to improve efficiency. |
| 97 | +- **Cross-rollup communication**: Extending the signalling mechanism to enable direct communication between different L2 solutions |
| 98 | +- **Intent protocols**: As part of the stack, application presets can be included so that rollup developers can implement intent protocols to incentivize the inclusion of signals in the L2 block. |
| 99 | +- **[ERC-7786](https://eips.ethereum.org/EIPS/eip-7786) Gateway**: An ERC-7786 gateway can orchastrate other applications based on signals. This could be used to create a more complex application that requires multiple signals to be included in the L2 block. |
0 commit comments