Fault proof VM: Cannon
Cannon is an instance of a Fault Proof Virtual Machine (FPVM) that can be used as part of the Dispute Game for any OP Stack Blockchain. The Dispute Game itself is modular, allowing for any FPVM to be used in a dispute. However, Cannon is Optimism's default Fault Proof Virtual Machine (FPVM). Cannon has two main components:
- Onchain
MIPS.sol
: EVM implementation to verify execution of a single MIPS instruction. - Offchain
mipsevm
: Go implementation to produce a proof for any MIPS instruction to verify onchain.
Note that Cannon is just one example of an FPVM that can be used to resolve disputes.
This documentation will go into detail about the subcomponents that make up the offchain Cannon implementation as a whole.
Additionally, we will explore the differences between Cannon and the onchain MIPS.sol
. For more information about the onchain
implementation, please refer to MIPS reference.
Now for simplicity, when referring to Cannon in this documentation, we are referring to the offchain implementation.
Control flow
In the above diagram, recreated from the OP Fault Proof System video (opens in a new tab) by Clabby, we can see that Cannon interacts with OP-Challenger and OP-Program. However, this diagram is a simplification of the relationship between OP-Challenger <> Cannon, and OP-Program <> Cannon. In general, Cannon will not be run until an active fault dispute reaches the execution trace portion of the bisection game. This does not occur until the participants in the active fault dispute game reach a single L2 block state transition that they disagree on.
OP-Challenger <> Cannon
Once an active fault dispute game reaches a depth below attacking / defending L2 block state transitions, OP-Challenger will run
Cannon to begin processing MIPS instructions within the FPVM. As part of processing MIPS instructions, Cannon will generate state
witness hashes, which are the commitment to the results of the MIPS instructions' computation within the FPVM. Now, in the bisection game, OP-Challenger will provide the generated hashes
until a single MIPS instruction is identified as the root disagreement between participants in the active dispute. Cannon will then
generate the witness proof, which contains all the information required to run the MIPS instruction onchain. Running this single MIPS
instruction onchain in MIPS.sol
will be used to definitively prove the correct post state, which will then be used to resolve the fault
dispute game.
OP-Program <> Cannon
Once the execution trace bisection begins and Cannon is run, an Executable and Linkable Format (ELF) binary will be loaded and run within Cannon.
Within Cannon is the mipsevm
that is built to handle the MIPS R3000, 32-bit Instruction Set Architecture (ISA). The ELF file contains MIPS instructions,
where the code that has been compiled into MIPS instructions is OP-Program.
OP-Program is golang code that will be compiled into MIPS instructions and run within the Cannon FPVM. OP-Program, whether run as standalone golang code or in Cannon, fetches all necessary data used for deriving the state of the L2. It is built such that the same inputs will produce not only the same outputs, but the same execution trace. This allows all participants in a fault dispute game to run OP-Program such that, given the same L2 output root state transition, they can generate the same execution traces. This in turn generates the same witness proof for the exact same MIPS instruction that will be run onchain.
Overview of offchain Cannon components
Now, we will go over each major component that makes up Cannon. Components are grouped by what functionality is being performed for Cannon, and may be correlated to one or more Go files. For brevity, each Go file will be explained at a high level, with the most important features / considerations highlighted.
mipsevm
state and memory
As mentioned previously, the mipsevm
is 32-bit, which means the full addressable address range is [0, 2^32-1]
. The memory layout
uses the typical monolithic memory structure, and the VM operates as though it were interacting directly with physical memory.
For the mipsevm
, how memory is stored isn't important, as it can hold the entire monolithic memory within the Go runtime.
In this way, how memory is represented is abstracted away from the VM itself. However, it is important for memory to be represented
such that only small portions are needed in order to run a MIPS instruction onchain. This is because it is infeasible to represent the
entire 32-bit memory space onchain due to cost. Therefore, memory is stored in a binary Merkle tree data structure, with the implementation
spread across memory.go
(opens in a new tab) and
page.go
(opens in a new tab).
The tree has a fixed-depth of 27 levels, with leaf values of 32 bytes each. This spans the full 32-bit address space: 2**27 * 32 = 2**32
.
Each leaf contains the memory for that part of the tree.
memory.go
defines the data structure, Memory
, which holds pointers to nodes
and pages
. A memory node
holds the calculated Merkle
root of its parent node within the memory binary Merkle tree, where the 'location' of the node is determined by its generalized index.
The index calculated for a Merkle root in the nodes mapping follows the
generalized Merkle tree index specification (opens in a new tab).
page.go
, as the name implies, defines memory pages. Each Page
is 4096 bytes, which is also specified as the minimum page allocation
size for the Go runtime. A Page
represents the lowest depths of the memory binary Merkle tree, and page.go
performs a similar role
to memory.go
, calculating Merkle roots for each level of the tree.
Nodes in this memory tree are combined as: out = keccak256(left ++ right)
, where ++
is concatenation,
and left
and right
are the 32-byte outputs of the respective subtrees or the leaf values themselves.
Individual leaf nodes are not hashed.
In both memory.go
and page.go
, there are a few optimizations designed to reduce the computationally expensive Merkle root calculations.
One such optimization is that the Merkle root of zeroed-out regions of memory are calculated for each depth. This means the tree is efficiently allocated,
since the root of fully zeroed subtrees can be computed without actually creating the full-subtree:
zero_hash[d] = hash(zero_hash[d-1], zero_hash[d-1])
, until the base-case of zero_hash[0] == bytes32(0)
.
So, pre-calculating zeroed Merkle roots initially allows unused memory regions to be cached. Additionally, non-zero Merkle roots
are cached in both memory.go
and page.go
, and used so long as the memory region the Merkle root covers has not been written to.
Otherwise, the cache is invalidated, which requires the Merkle root to be calculated over its entire subtree. Another optimization
specifically in memory.go
is caching the last two pages that have been used. Two pages are cached because the mipsevm
uses up
to two memory addresses per instruction: one address is the PC
, which contains the instruction to run, and the other address may be the
target of a load or store instruction.
Another important implementation detail is that the endianness of the mipsevm
, which is the ordering of bytes in a word, is Big-Endian.
The assumption of Cannon is that it is operating on a Little-Endian machine, and as such all reads / writes have the endianness swapped
so that the internal mipsevm
always handles Big-Endian words and the machine running Cannon always handles Little-Endian words.
Endianness is also an important factor for the onchain MIPS.sol
, where the EVM itself is Big-Endian. Therefore, MIPS.sol
does not have to
do any endianness swapping and can assume all data uses Big-Endian ordering. This reduces complexity within the smart contract itself.
The last major component is located in state.go
(opens in a new tab).
The State
struct in state.go
holds all the execution state that is required for the mipsevm
.
The information stored is largely identical to the VM execution state for MIPS.sol
. The key differences are:
- Instead of storing just the memory Merkle root, there is a
Memory
Struct pointer for the binary Merkle tree representation of the entire 32-bit memory space. - There is an optional
LastHint
bytes variable, which can be used to communicate a Pre-image hint to avoid having to load in multiple prior Pre-images.
Generating the witness proof
Cannon handles two major components in the dispute game: generating state witness hashes for OP-Challenger to post during the execution trace
bisection game, and generating the witness proof once a single MIPS instruction is reached as the root of disagreement in the fault dispute game.
The witness proof, as mentioned previously, contains all the necessary information for MIPS.sol
to be able to run the same instruction onchain,
and derive the post state that will be used to resolve the fault dispute game. The post state of the instruction run by MIPS.sol
should be
exactly the same as the post state generated by the mipsevm
.
The top-level witness.go
(opens in a new tab)
in cannon/cmd
initiates the witness proof generation. The internal
witness.go
(opens in a new tab) in cannon/mipsevm
defines the
struct that holds all the relevant information for the particular MIPS instruction. The information that is encoded for the MIPS.sol
calldata can be seen in the MIPS.sol
documentation.
Additionally, if a Pre-image is required for the MIPS
instruction, witness.go
will communicate the relevant Pre-image key and offset to OP-Challenger so that it can be posted onchain
to PreimageOracle.sol
.
An important note about generating the witness proof: it is imperative that all relevant information about the instruction to be run
onchain is generated before the mipsevm
execution state changes as a result of processing the MIPS instruction. Otherwise, if the
witness proof is generated after running the instruction offchain, the state that will be encoded will be the post state.
Loading the ELF file
Once the execution trace portion of the bisection game begins, the ELF file containing OP-Program compiled into MIPS instructions will
be run within Cannon. However, getting OP-Program into Cannon so that it can be run requires
a binary loader. The binary loader is composed of load_elf.go
(opens in a new tab)
and patch.go
(opens in a new tab). load_elf.go
parses
the top-level arguments and reads, loads, and patches the ELF binary such that it can be run by Cannon.
patch.go
is responsible for actually parsing the headers of the ELF file, which among other information specifies what programs
exist within the file and where they are located in memory. The loader uses this information to instantiate the execution state
for the mipsevm
and load each program into memory at the expected location. Additionally, as part of instantiating the execution
state, patch.go
sets the values of PC
, NextPC
, the initial Heap location of 0x20000000
, the initial stack location of 0x7fffd000
,
and also sets the arguments required for the Go runtime above the stack pointer location.
While loading the ELF file into Cannon, metadata.go
(opens in a new tab)
is used to parse all the symbols stored in the ELF file. Understanding which ELF symbols exist and at which regions of memory
they are located is important for other functionality, such as understanding if the current PC
is running within a specific function.
Another step that occurs while loading the ELF file is patching the binary of any incompatible functions. This step is crucial, as a key design decision
in both the onchain and offchain mipsevm
implementations is that neither implementation
has access to a kernel. This is primarily due to the limitations within the EVM itself, and since the offchain Cannon
implementation must match functionality exactly with its onchain counterpart, kernel access is also not available within Cannon. This means that the VMs
cannot replicate behavior that would otherwise be performed by a kernel 1:1, which primarily impacts system calls (syscalls).
So features like concurrency, memory management, I/O, etc. are either partially implemented or unimplemented. For the
unimplemented functionality, any function within the ELF file that would require this functionality is patched out.
Patching the binary of these functions involves identifying problematic functions, searching for their corresponding symbols
within the ELF file, and effectively stubbing out the function by returning immediately, and adding a nop
to the delay slot.
Instruction stepping
Once the MIPS binary is loaded into Cannon, we can then begin to run MIPS instructions one at a time.
run.go
(opens in a new tab) contains the top-level
code responsible for stepping through MIPS instructions. Additionally, before each MIPS instruction, run.go
will determine whether
separate action(s) need(s) to be performed. The action(s) to be performed are configured by the user, and can include logging information,
stopping at a certain instruction, taking a snapshot at a certain instruction, or signaling to the VM to generate the witness proof.
The action(s) to be performed are instantiated and checked by matcher.go
(opens in a new tab),
which generates a match function that triggers when the configured regular expression is true.
Within run.go
, the StepFn
is the wrapper that initiates the MIPS instruction to be run.
instrumented.go
(opens in a new tab) implements Step
as the
interface to be initiated for each MIPS instruction. Additionally, instrumented.go
handles encoding information for the witness proof
and Pre-image information (if required for the MIPS instruction).
mips.go
mips.go
(opens in a new tab) implements all the required MIPS
instructions, and also tracks additional memory access for the instruction currently being run.
This is important to make sure that the second memory proof is encoded correctly for instructions that use it, such as loads, stores, and
certain syscalls. The full list of instructions supported can be found here.
mips.go
vs. MIPS.sol
The offchain mips.go
and the onchain Cannon MIPS.sol
behave similarly when it comes to executing 32-bit, MIPS III instructions.
In fact, they must produce exactly the same results given the same instruction, memory, and register state.
Consequently, the witness data is essential to reproduce the same instruction onchain and offchain.
However, there are differences between the two:
- A single instruction will be run onchain in
MIPS.sol
, whereas the offchainmips.go
will run all MIPS instructions for all state transitions in the disputed L2 state. - The
mipsevm
contains the entire 32-bit monolithic memory space, is responsible for maintaining the memory state based on the results of MIPS instructions, and generates the memory binary Merkle tree, Merkle root, and memory Merkle proofs.MIPS.sol
is mostly stateless, and does not maintain the full memory space. Instead, it only requires the memory Merkle root, and up to two memory Merkle proofs: 1 for the instruction and 1 for potential load, store, or certain syscall instructions. - Unlike
MIPS.sol
,mips.go
is responsible for writing Pre-images to the PreimageOracle Server, and optionally writing hints to the Server.
PreimageOracle interaction
As mentioned previously, Cannon is responsible for setting up all state that may be required to run an instruction in MIPS.sol
.
Cannon is also responsible for interacting with the PreimageOracle Server, and directing OP-Challenger to provide Pre-images to
the onchain PreimageOracle.sol
if necessary for the instruction that will be run in MIPS.sol
.
The PreimageOracle Server connection is instantiated in run.go
, where the server itself is run locally with its own local Key-Value store.
mips.go
communicates with the PreimageOracle Server when reading and writing Pre-images as part of MIPS syscall instructions,
as well as hinting to the PreimageOracle Server.
The OP-stack fault-proof Pre-image Oracle specs (opens in a new tab) define the ABI for communicating pre-images.
This ABI is implemented by the VM by intercepting the read
/write
syscalls to specific file descriptors. See Cannon VM Specs (opens in a new tab) for more details.
Note that although the oracle provides up to 32 bytes of the pre-image, Cannon only supports reading at most 4 bytes at a time, to unify the memory operations with 32-bit load/stores.