Skip to content

Commit

Permalink
docs(protocol): clean up protocol docs (taikoxyz#16176)
Browse files Browse the repository at this point in the history
  • Loading branch information
dionysuzx authored Feb 29, 2024
1 parent b94cca8 commit 3800f04
Show file tree
Hide file tree
Showing 13 changed files with 16 additions and 14 deletions.
File renamed without changes
File renamed without changes
28 changes: 15 additions & 13 deletions packages/protocol/docs/multihop_bridging_deployment.md
Original file line number Diff line number Diff line change
Expand Up @@ -3,9 +3,10 @@
This document explains how multi-hop cross-chain bridging works in Taiko.

## L1<->L2 data synchronization

We'll use this diagram to illustrate a blockchain's state. The large triangle represents the world state, while the smaller triangle represents the storage tree of a special contract named the "Signal Service," deployed on both L1 and L2.

![State Diagram](./multihop/state.png)
![State Diagram](./images/state.png)

When a signal is sent by the Signal Service, a unique slot in its storage is updated with a value of `1`, as shown in the Solidity code below:

Expand All @@ -23,23 +24,24 @@ function getSignalSlot(uint64 chainId, address app, bytes32 signal) public pure
}
```

Merkle proofs can verify signals sent by specific senders when the signal service's state root is known on another chain. A full merkle proof comprises an *account proof* and a *storage proof*. However, if the signal service's storage root (or the *signal root*) is known on another chain, only a storage proof is necessary to verify the signal's source.
Merkle proofs can verify signals sent by specific senders when the signal service's state root is known on another chain. A full merkle proof comprises an _account proof_ and a _storage proof_. However, if the signal service's storage root (or the _signal root_) is known on another chain, only a storage proof is necessary to verify the signal's source.

![Merkle Proof](./multihop/merkle_proof.png)
![Merkle Proof](./images/merkle_proof.png)

Taiko's core protocol code (TaikoL1.sol and TaikoL2.sol) automatically synchronizes or relays the state roots between L1 and L2.

When chainA's state root is synced to chainB, a special signal is sent in chainB's signal service. This signal is calculated incorporating chainA's block ID. These special signals are always sent by the target chain's signal service.

![L1-L2 Sync](./multihop/l1_l2_sync.png)
![L1-L2 Sync](./images/l1_l2_sync.png)

If you deploy more chains using Taiko protocol, you can create a chain of synced state roots between them.

![Three Chains](./multihop/three_chains.png)
![Three Chains](./images/three_chains.png)

## Verifying bridged messages

### One-hop bridging

Consider the 1-hop example below.

To verify that "some app" has sent a custom message, we verify if the corresponding signal (associated with the message sender, "some app") has been set by the signal service (0x1000A) on L1. After L1's state root is synced to L2, we need the following info on L2 to verify the message on L1:
Expand All @@ -49,25 +51,25 @@ To verify that "some app" has sent a custom message, we verify if the correspond
3. L1 signal service's address associated with the merkle proof.
4. L2 signal service's address to verify that L1's state root has been synced to L2 already.

![1-Hop Bridging](./multihop/bridge_1hop.png)
![1-Hop Bridging](./images/bridge_1hop.png)

### Multi-hop bridging

In the 2-hop example below, two merkle proofs are needed, and the signal service addresses for L1 and L2 need verification. L3's signal service address does not need verification as the bridging verification occurs in L3's signal service contract, with L3's signal service address being `address(this)`.

![2-Hop Bridging](./multihop/bridge_2hop.png)
![2-Hop Bridging](./images/bridge_2hop.png)

## Caching

Caching is optional and is activated per hop when the transaction intends to reuse some state root or signal root for future bridging verification.


In the diagram below with 2 hops, L1's state root and L2's signal root can be cached to L3 if specified.

![Cache Example 1](./multihop/cache_1.png)
![Cache Example 1](./images/cache_1.png)

If both are cached, two more signals will be sent in L3's signal service.

![Cache Example 1 Done](./multihop/cache_1_done.png)
![Cache Example 1 Done](./images/cache_1_done.png)

Depending on the type of data (state root or signal root), the signal is generated differently.

Expand All @@ -79,12 +81,12 @@ function signalForChainData(uint64 chainId, bytes32 kind, bytes32 data) public p

Once cached on L3, one full merkle proof is sufficient to verify everything that happened on L1 before or when L1's state root becomes 0x1111. This allows skipping the middle-hop.

![Cache Use 1](./multihop/cache_1_use_1.png)
![Cache Use 1](./images/cache_1_use_1.png)

If L1's state root is not cached on L3 but only L2's signal root is, then one full merkle proof for L1 and a storage proof for L2 are used to verify a bridged message.

![Cache Use 2](./multihop/cache_1_use_2.png)
![Cache Use 2](./images/cache_1_use_2.png)

Note that the last hop (L2)'s state root has already been auto-relayed to L3, so it cannot be recached. Therefore, only the last hop's signal root can be cached.

For all other non-last hops, if a full proof is used, the state root can be cached; if a storage proof is used, the signal root can be cached. But by default, caching is all disabled.
For all other non-last hops, if a full proof is used, the state root can be cached; if a storage proof is used, the signal root can be cached. But by default, caching is all disabled.
2 changes: 1 addition & 1 deletion packages/protocol/docs/native_token_support.md
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
# Taiko native vs. wrapped token bridging

![Wrapped_vs_Native](./native_bridge/native_support.png "Wrapped vs. Native bridging")
![Wrapped_vs_Native](./images/native_support.png "Wrapped vs. Native bridging")

Taiko's briding concept is a lock-and-mint type. It simply means (the red path above) on the canonical chain we take custody of the assets and on the destination chain we mint the wrapped counterpart. When someone wants to bridge back (from destination to canonical) it will first burn the tokens, then release the funds on the canonical chain.

Expand Down

0 comments on commit 3800f04

Please sign in to comment.