Icon HelpCircleForumIcon Link

⌘K

Icon HelpCircleForumIcon Link
Nightly /
Fuel Vm

Icon LinkFuel VM Specification

Icon LinkIntroduction

This document provides the specification for the Fuel Virtual Machine (FuelVM). The specification covers the types, instruction set, and execution semantics.

Icon LinkParameters

nametypevaluenote
CONTRACT_MAX_SIZEuint64Maximum contract size, in bytes.
VM_MAX_RAMuint642**2664 MiB.
MESSAGE_MAX_DATA_SIZEuint64Maximum size of message data, in bytes.

Icon LinkSemantics

FuelVM instructions are exactly 32 bits (4 bytes) wide and comprise of a combination of:

  • Opcode: 8 bits
  • Register/special register (see below) identifier: 6 bits
  • Immediate value: 12, 18, or 24 bits, depending on operation

Of the 64 registers (6-bit register address space), the first 16 are reserved:

valueregisternamedescription
0x00$zerozeroContains zero (0), for convenience.
0x01$oneoneContains one (1), for convenience.
0x02$ofoverflowContains overflow/underflow of addition, subtraction, and multiplication.
0x03$pcprogram counterThe program counter. Memory address of the current instruction.
0x04$sspstack start pointerMemory address of bottom of current writable stack area.
0x05$spstack pointerMemory address on top of current writable stack area (points to free memory).
0x06$fpframe pointerMemory address of beginning of current call frame.
0x07$hpheap pointerMemory address below the current bottom of the heap (points to used/OOB memory).
0x08$errerrorError codes for particular operations.
0x09$ggasglobal gasRemaining gas globally.
0x0A$cgascontext gasRemaining gas in the context.
0x0B$balbalanceReceived balance for this context.
0x0C$isinstructions startPointer to the start of the currently-executing code.
0x0D$retreturn valueReturn value or pointer.
0x0E$retlreturn lengthReturn value length in bytes.
0x0F$flagflagsFlags register.

Integers are represented in big-endian Icon Link format, and all operations are unsigned. Boolean false is 0 and Boolean true is 1.

Registers are 64 bits (8 bytes) wide. Words are the same width as registers.

Persistent state (i.e. storage) is a key-value store with 32-byte keys and 32-byte values. Each contract has its own persistent state that is independent of other contracts. This is committed to in a Sparse Binary Merkle Tree.

Icon LinkFlags

valuenamedescription
0x01F_UNSAFEMATHIf set, undefined arithmetic zeroes target and sets $err without panic.
0x02F_WRAPPINGIf set, overflowing arithmetic wraps around and sets $of without panic.

All other flags are reserved, any must be set to zero.

Icon LinkInstruction Set

A complete instruction set of the Fuel VM is documented in the following page .

Icon LinkVM Initialization

Every time the VM runs, a single monolithic memory of size VM_MAX_RAM bytes is allocated, indexed by individual byte. A stack and heap memory model is used, allowing for dynamic memory allocation in higher-level languages. The stack begins at 0 and grows upward. The heap begins at VM_MAX_RAM and grows downward.

To initialize the VM, the following is pushed on the stack sequentially:

  1. Transaction hash (byte[32], word-aligned), computed as defined here .
  2. Base asset ID (byte[32], word-aligned)
  3. MAX_INPUTS pairs of (asset_id: byte[32], balance: uint64), of:
    1. For predicate estimation and verification , zeroes.
    2. For script execution , the free balance for each asset ID seen in the transaction's inputs, ordered in ascending order. If there are fewer than MAX_INPUTS asset IDs, the pair has a value of zero.
  4. Transaction length, in bytes (uint64, word-aligned).
  5. The transaction, serialized .

Then the following registers are initialized (without explicit initialization, all registers are initialized to zero):

  1. $ssp = 32 + 32 + MAX_INPUTS*(32+8) + size(tx)): the writable stack area starts immediately after the serialized transaction in memory (see above).
  2. $sp = $ssp: writable stack area is empty to start.
  3. $hp = VM_MAX_RAM: the heap area begins at the top and is empty to start.

Icon LinkContexts

There are 4 contexts in the FuelVM: predicate estimation and verification , scripts , and calls . A context is an isolated execution environment with defined memory ownership and can be external or internal:

  • External: predicate and script. $fp will be zero.
  • Internal: call. $fp will be non-zero.

Returning from a context behaves differently depending on whether the context is external or internal.

Icon LinkPredicate Estimation and Verification

There are two ways to run predicates on the VM:

  1. Estimation: runs the predicate and updates the amount of gas used
  2. Verification: runs the predicate and verifies the amount of gas used matches the input

For any input of type InputType.Coin or InputType.Message , a non-zero predicateLength field means the UTXO being spent is a P2SH Icon Link rather than a P2PKH Icon Link output.

For each such input in the transaction, the VM is initialized , then:

  1. $pc and $is are set to the start of the input's predicate field.
  2. $ggas and $cgas are set to MAX_GAS_PER_PREDICATE for estimation, and predicateGasUsed for verification.

Predicate execution will fail if gas is exhausted during execution.

During predicate mode, hitting any contract instruction (except ldc with non-contract target) causes predicate verification to halt, returning Boolean false.

A predicate that halts without returning Boolean true would not pass verification, making the entire transaction invalid. Note that predicate return value is monotonic with respect to time (i.e. if a predicate evaluates to true then it will always evaluate to true in the future).

After successful execution, the run mode is determines the final step:

  1. Estimation: predicateGasUsed is set to MAX_GAS_PER_PREDICATE - $ggas.
  2. Verification: if $ggas is non-zero, predicate verification fails.

Icon LinkScript Execution

If script bytecode is present, transaction validation requires execution.

The VM is initialized , then:

  1. $pc and $is are set to the start of the transaction's script bytecode.
  2. $ggas and $cgas are set to tx.scriptGasLimit.

Following initialization, execution begins.

For each instruction, its gas cost gc is first computed. If gc > $cgas, deduct $cgas from $ggas and $cgas (i.e. spend all of $cgas and no more), then revert immediately without actually executing the instruction. Otherwise, deduct gc from $ggas and $cgas.

After the script has been executed, tx.receiptsRoot is updated to contain the Merkle root of the receipts, as described in the TransactionScript spec .

Icon LinkCall

Call context is entered via CALL instruction . It's also called internal context, or contract context. Call context is used to access state of a contract.

Icon LinkCall Frames

Cross-contract calls push a call frame onto the stack, similar to a stack frame used in regular languages for function calls (which may be used by a high-level language that targets the FuelVM). The distinction is as follows:

  1. Stack frames: store metadata across trusted internal (i.e. intra-contract) function calls. Not supported natively by the FuelVM, but may be used as an abstraction at a higher layer.
  2. Call frames: store metadata across untrusted external (i.e. inter-contract) calls. Supported natively by the FuelVM.

Call frames are needed to ensure that the called contract cannot mutate the running state of the current executing contract. They segment access rights for memory: the currently-executing contracts may only write to their own call frame and their own heap.

A call frame consists of the following, word-aligned:

bytestypevaluedescription
Unwritable area begins.
32byte[32]toContract ID for this call.
32byte[32]asset_idasset ID of forwarded coins.
8*64byte[8][64]regsSaved registers from previous context.
8uint64codesizeCode size in bytes, padded to the next word boundary.
8byte[8]param1First parameter.
8byte[8]param2Second parameter.
1*byte[]codeZero-padded to 8-byte alignment, but individual instructions are not aligned.
Unwritable area ends.
*Call frame's stack.

Icon LinkAccess rights

Only memory that has been allocated is accessible. In other words, memory between highest-ever $sp value and current $hp is inaccessible. Attempting to read or write memory that has not been allocated will result in VM panic. Similarly reads or writes that cross from the stack to the heap will panic. Note that stack remains readable even after stack frame has been shrunk. However, if the heap is afterwards expanded to cover that area, the crossing read prohibition still remains, while all memory is accessible.

Icon LinkOwnership

Whenever memory is written to (i.e. with SB or SW ), or write access is granted (i.e. with CALL ), ownership must be checked.

If the context is external, the owned memory range is:

  1. [$ssp, $sp): the writable stack area.
  2. [$hp, VM_MAX_RAM): the heap area allocated by this script or predicate.

If the context is internal, the owned memory range for a call frame is:

  1. [$ssp, $sp): the writable stack area of the call frame.
  2. [$hp, $fp->$hp): the heap area allocated by this call frame.

Icon LinkExecutability

Memory is only executable in range [$is, $ssp). Attempting to execute instructions outside these boundaries will cause a panic. This area never overlaps with writable memory, essentially providing W^X Icon Link protection.