UCS03 - ZKGM
ucs03-zkgm-0
is the most advanced and recommended protocol to use for
- message passing
- transfers (assets and NFTs)
- intents
- storage proofs
It’s the most gas-efficient version and suitable for almost all use cases.
Protocol
Open Filling
A groundbreaking protocol improvement on IBC and trust-minimized bridging in general, is that ucs03-zkgm-0
allows arbitrary filling of orders by any party.
For packet submissions and transfers, the protocol allows a different filler from the Union IBC contracts. In the case of an alternative filler, the assets are not minted but transferred from the filler’s account. This allows a transfer to be filled before chains have been finalized or client updates processed. Instead, fillers can rely on preconfirmations to reduce the risk of reorgs.
Theoretically, a filler can submit the transfer to the destination side before the transaction is included on the source, given that they protect themselves against double-spend attacks.
The Acknowledgement
, which may contain arbitrary payloads, is used to encode information on the filler and repay the filler for the service by unlocking assets from the vault.
Open filling is opt-in for any protocol, allowing for the same optimizations that ucs03-zkgm-0
leverages to increase transfer speeds.
Definition
The zkgm protocol abstracts away multiple facets of IBC assets transfer protocol (ics20). We employ versioning in this protocol to ensure backward compatibility with future upgrades (not relying on the IBC channel upgrade feature). Each instruction has a version and opcode to allow for protocol evolution. Its features include:
- batching
- forward/callback envelopes
- channel multiplexing
- fungible assets transfer
- non-fungible assets transfer
Packet
The zkgm protocol uses two main structures for packet handling:
ZkgmPacket
struct ZkgmPacket { bytes32 salt; // Unique packet identifier uint256 path; // Channel routing information Instruction instruction; // The instruction to execute}
Fields:
-
salt
: A uniquebytes32
identifier used for deterministic packet hashing- For regular packets:
keccak256(abi.encodePacked(sender, user_provided_salt))
- For forwarded packets: Uses a tinting mechanism to track packet chain
- Magic value:
0xC0DE00000000000000000000000000000000000000000000000000BABE
- Tinting applied as:
salt | magic_value
(bitwise OR) - previous_salt is the salt of the packet being forwarded
- Next hop salt derived as:
keccak256(previous_salt) | magic_value
- This creates a verifiable chain of salts across hops while preventing salt collisions
- Magic value:
- For batch instructions: Each sub-instruction uses
keccak256(index, batch_salt)
- Where index is the instruction’s position in the batch (0-based)
- And batch_salt is the parent packet’s salt
- For regular packets:
-
path
: Auint256
that tracks packet routing and asset origins- Composed of compressed
uint32
channel IDs in 32-bit segments - Format:
prevDstChannel0 | nextSrcChannel0 << 32 | prevDstChannel1 << 64 ...
- Supports up to 3 hops (256/32/2 - 1), one hop is a placeholder for the final channel ID.
- Updated during:
- Packet forwarding (appending channel pairs)
- Asset origin tracking
- Return path validation
- Composed of compressed
-
instruction
: The Instruction to execute
Instruction
struct Instruction { uint8 version; // Protocol version uint8 opcode; // Instruction type bytes operand; // Instruction-specific data}
Fields:
-
version
: Protocol version number asuint8
- Required for backward compatibility
- Allows dispatching between different versions
- Each instruction type specifies required version
-
opcode
: Instruction type identifier asuint8
- 0x00: Forward
- 0x01: Multiplex
- 0x02: Batch
- 0x03: FungibleAssetOrder
-
operand
: Instruction-specific data asbytes
- Forward: Path, timeouts and instruction to forward
- Multiplex: Sender, callback mode and contract call data
- Batch: Array of instructions to execute atomically
- FungibleAssetOrder: Transfer details like tokens, amounts and parties
Instructions
0x00 - Forward
The forward instruction uses opcode 0x00
and requires version INSTR_VERSION_0
.
struct Forward { uint256 path; // Channel sequence as (prevDst,nextSrc) pairs uint64 timeoutHeight; // Block height timeout uint64 timeoutTimestamp; // Unix timestamp timeout Instruction instruction; // Instruction to forward}
Fields:
-
path
: Auint256
that encodes the forwarding route- Composed of (prevDst,nextSrc) channel ID pairs
- Each pair uses 64 bits (32 bits per channel ID)
- Must match valid channel connections
- Used to verify packet routing path
-
timeoutHeight
: Block height timeout asuint64
- After this height, packet cannot be executed
- Set to 0 to disable height timeout
- Must be greater than current height when executed
-
timeoutTimestamp
: Unix timestamp timeout asuint64
- After this time, packet cannot be executed
- Set to 0 to disable timestamp timeout
- Must be greater than current time when executed
-
instruction
: The Instruction to forward- Can be Multiplex, FungibleAssetOrder, or Batch
- Will be executed on final destination chain
- Cannot be another Forward instruction
0x01 - Multiplex
The multiplex instruction uses opcode 0x01
and requires version INSTR_VERSION_0
.
struct Multiplex { bytes sender; // Source chain sender address (must match msg.sender) bool eureka; // Whether to use IBC-style callbacks bytes contractAddress; // Target contract address on destination bytes contractCalldata; // Call data for the target contract}
Fields:
-
sender
: Source chain sender address asbytes
- Must match transaction sender (msg.sender)
- Prevents address impersonation
- Used for callback routing in eureka mode
-
eureka
: Callback mode flag asbool
- false: Standard fire-and-forget mode
- true: IBC-style callback mode
- Determines target contract interface
- Controls acknowledgement handling
-
contractAddress
: Target contract address asbytes
- Must be valid contract on destination chain
- Must implement required interface based on eureka flag
- Where message will be delivered
-
contractCalldata
: Contract call data asbytes
- Arbitrary data passed to target contract
- Interpreted by target contract’s implementation
- Available in both standard and eureka modes
The multiplex instruction has two modes:
-
Standard Mode (eureka = false):
- Target must implement IEurekaModule interface
- Calls
onZkgm(path, sourceChannel, destChannel, sender, calldata)
on target - Returns success acknowledgement immediately
- Fire-and-forget style, no callback to sender
-
IBC Mode (eureka = true):
- Calls
onRecvPacket(packet, relayer, relayerMsg)
on target - Packet contains path, sender and calldata
- Target must return non-empty acknowledgement
- Acknowledgement forwarded back to original sender
- Calls
If the target contract is invalid or calls fail:
- Standard mode returns failure acknowledgement
- IBC mode propagates target’s error response
0x02 - Batch
The batch instruction uses opcode 0x02
and requires version INSTR_VERSION_0
.
struct Batch { Instruction[] instructions; // Array of instructions to execute}
Fields:
instructions
: Array of Instructions to execute atomically- Only Multiplex and FungibleAssetOrder allowed
- Executed in sequential order
- All must succeed or entire batch reverts
- Individual acknowledgements collected in array
- Minimum 2 instructions required
This allows atomic composition of transfers and contract calls in a single transaction.
0x03 - Fungible Asset Order
The fungible asset order instruction uses opcode 0x03
and requires version INSTR_VERSION_1
.
struct FungibleAssetOrder { bytes sender; // Source chain sender address bytes receiver; // Destination chain receiver address bytes baseToken; // Token being sent uint256 baseAmount; // Amount being sent string baseTokenSymbol; // Token symbol for wrapped asset string baseTokenName; // Token name for wrapped asset uint8 baseTokenDecimals; // Token decimals for wrapped asset uint256 baseTokenPath; // Origin path for unwrapping bytes quoteToken; // Token requested in return uint256 quoteAmount; // Minimum amount requested}
Fields:
-
sender
: Source chain sender address asbytes
- Must be valid address on source chain
- Used for refunds on failure/timeout
-
receiver
: Destination chain receiver address asbytes
- Must be valid address on destination chain
- Where quote tokens will be sent on success
- Must be specified by sender
-
baseToken
: Token being sent asbytes
- Token identifier on source chain
- Used to identify/create wrapped token
- Must exist on source chain
-
baseAmount
: Amount being sent asuint256
- Must be available from sender
- Maximum amount to exchange
-
baseTokenSymbol
: Token symbol asstring
- Used when creating wrapped token
-
baseTokenName
: Token name asstring
- Used when creating wrapped token
-
baseTokenDecimals
: Token decimals asuint8
- Used when creating wrapped token
-
baseTokenPath
: Origin path asuint256
- Used for unwrapping return transfers
- Must match original wrapping path
-
quoteToken
: Requested token asbytes
- Token identifier on destination chain
- What sender wants in exchange
- Must exist on destination chain
-
quoteAmount
: Minimum amount requested asuint256
- Minimum acceptable exchange amount
- Difference (if less than
baseAmount
) is taken as fee by the relayer
The order can be filled in two ways:
-
Protocol Fill - If the quote token matches the wrapped version of the base token and base amount >= quote amount:
- For new assets: Deploy wrapped token contract and mint quote amount to receiver
- For returning assets: Unwrap base token and transfer quote amount to receiver
- Any difference between baseAmount and quoteAmount is minted/transferred to the relayer as a fee
-
Market Maker Fill - Any party can fill the order by providing the quote token:
- Market maker is specified in acknowledgement
- Base token is transferred/minted to market maker
- Market maker must handle quote token transfer on behalf of the protocol
The acknowledgement includes:
- Fill type (Protocol =
0xB0CAD0
or MarketMaker =0xD1CEC45E
) - Market maker address for MM fills (empty for protocol fills)
If the order fails or times out:
- For new assets: Base token is minted back to sender
- For returning assets: Base token is transferred back to sender
- Outstanding balances are decreased
If any of the order in the orders
list is failing on execution, the whole packet is reverted and a failure acknowledgement will be yield.