XCVM
π« The XCVM
At the core of XCM lies the Cross-Consensus Virtual Machine (XCVM).
A βmessageβ in XCM is an XCVM program, which is a sequence of instructions.
The XCVM is a state machine, state is kept track in registers.
XCVM Registers
π The Origin Register
Contains an Option<Location>
of the cross-consensus origin where the message originated from.
πΈ The Holding Register
Expresses a number of assets in control of the xcm execution that have no on-chain representation.
They don't belong to any account.
It can be seen as the register holding "unspent assets".
π XCVM Instructions
XCVM Instructions might change a register, they might change the state of the consensus system or both.
Kinds of instructions
- Command
- Trusted Indication
- Information
- System Notification
Example: WithdrawAsset
An instruction used to get assets from an account and put them into the holding register.
WithdrawAsset(Assets)
Example: ReceiveTeleportedAsset
Used for teleporting assets between two systems.
ReceiveTeleportedAsset(Assets)
Example: QueryResponse
Used for reporting information back to another system.
QueryResponse {
#[codec(compact)]
query_id: QueryId,
response: Response,
max_weight: Weight,
querier: Option<Location>,
}
Basic XCVM Operation
XCVM Operation
XCVM Operation
XCVM Operation
XCVM Operation
XCVM Operation
XCVM Operation
XCVM Operation
XCVM vs. Standard State Machine
- Error handler register
- Appendix register
Reanchoring
How do different locations reference the same asset?
USDT from Asset Hub
PalletInstance(50)/GeneralIndex(1984)
USDT from Bridge Hub
../Parachain(1000)/PalletInstance(50)/GeneralIndex(1984)
Reanchoring to the rescue
π€Ή Cross-consensus transfers
1. Asset teleportation

1.1. Example: System parachains?
1.2. Example: Polkadot and Kusama?
2. Reserve asset transfers

2.1. Example: Parachain native tokens
2.2. Example: Polkadot to Kusama
π XCM by example
The WithdrawAsset
instruction
enum Instruction {
/* snip */
WithdrawAsset(Assets),
/* snip */
}
The BuyExecution
instruction
enum Instruction {
/* snip */
BuyExecution {
fees: Asset,
weight_limit: WeightLimit,
},
/* snip */
}
The DepositAsset
instruction
enum Instruction {
/* snip */
DepositAsset {
assets: AssetFilter,
beneficiary: Location,
},
/* snip */
}
Putting it all together
Xcm(vec![
WithdrawAsset((Here, amount).into()),
BuyExecution {
fees: (Here, amount).into(),
weight_limit: Limited(sanity_check_weight_limit)
},
DepositAsset { assets: All.into(), beneficiary: AccountId32 { ... }.into() },
])
Good pattern
Xcm(vec![
WithdrawAsset((Here, amount).into()),
BuyExecution {
fees: (Here, amount).into(),
weight_limit: Limited(sanity_check_weight_limit)
},
DepositAsset { assets: All.into(), beneficiary: AccountId32 { ... }.into() },
RefundSurplus,
DepositAsset { assets: All.into(), beneficiary: sender }
])
Reserve asset transfer
Xcm(vec![
WithdrawAsset(asset),
InitiateReserveWithdraw {
assets: All.into(),
reserve: reserve_location,
xcm: /* ...what to do with the funds in the reserve... */,
},
])
Message received in reserve
Xcm(vec![
WithdrawAsset(reanchored_asset),
ClearOrigin, // <- Why is this needed?
/* ...custom instructions... */
])
Custom XCM
let xcm_for_reserve = Xcm(vec![
DepositReserveAsset {
assets: All.into(),
dest: location,
xcm: Xcm(vec![
DepositAsset {
assets: All.into(),
beneficiary: AccountId32 { ... }.into(),
},
]),
},
]);
Message received in destination
Xcm(vec![
ReserveAssetDeposited(reanchored_asset),
ClearOrigin, // <- Why is this needed?
/* ...custom instructions... */
])
Summary
- XCVM
- Kinds of instructions
- Registers: Origin, Holding, Error handler, Appendix
- Reanchoring
- Cross-consensus transfers
- Teleports
- Reserve asset transfers
- Instructions
- WithdrawAsset, BuyExecution, DepositAsset
- RefundSurplus
- InitiateReserveWithdraw, ReserveAssetDeposited
Next steps
- Blog series introducing XCM: Parts 1, 2, and 3.
- XCM Format repository
- XCM Docs