Instructions

Instructions are classified by instruction types that describe how they manipulate the operand stack and initialize locals: A type [t1]x[t2] describes the required input stack with argument values of types t1 that an instruction pops off and the provided output stack with result values of types t2 that it pushes back. Moreover, it enumerates the indices x of locals that have been set by the instruction. In most cases, this is empty.

Note

For example, the instruction i32.add has type [i32 i32][i32], consuming two i32 values and producing one. The instruction local.set x has type [t]x[], provided t is the type declared for the local x.

Typing extends to instruction sequences instr. Such a sequence has an instruction type [t1]x[t2] if the accumulative effect of executing the instructions is consuming values of types t1 off the operand stack, pushing new values of types t2, and setting all locals x.

For some instructions, the typing rules do not fully constrain the type, and therefore allow for multiple types. Such instructions are called polymorphic. Two degrees of polymorphism can be distinguished:

  • value-polymorphic: the value type t of one or several individual operands is unconstrained. That is the case for all parametric instructions like drop and select.

  • stack-polymorphic: the entire (or most of the) instruction type [t1][t2] of the instruction is unconstrained. That is the case for all control instructions that perform an unconditional control transfer, such as unreachable, br, br_table, and return.

In both cases, the unconstrained types or type sequences can be chosen arbitrarily, as long as they meet the constraints imposed for the surrounding parts of the program.

Note

For example, the select instruction is valid with type [t t i32][t], for any possible number type t. Consequently, both instruction sequences

(i32.const 1)  (i32.const 2)  (i32.const 3)  select

and

(f64.const 1.0)  (f64.const 2.0)  (i32.const 3)  select

are valid, with t in the typing of select being instantiated to i32 or f64, respectively.

The unreachable instruction is stack-polymorphic, and hence valid with type [t1][t2] for any possible sequences of value types t1 and t2. Consequently,

unreachable  i32.add

is valid by assuming type [][i32] for the unreachable instruction. In contrast,

unreachable  (i64.const 0)  i32.add

is invalid, because there is no possible type to pick for the unreachable instruction that would make the sequence well-typed.

The Appendix describes a type checking algorithm that efficiently implements validation of instruction sequences as prescribed by the rules given here.

Numeric Instructions

t.const c

  • The instruction is valid with type [][t].

Ct.const c:[][t]

t.unop

  • The instruction is valid with type [t][t].

Ct.unop:[t][t]

t.binop

  • The instruction is valid with type [t t][t].

Ct.binop:[t t][t]

t.testop

  • The instruction is valid with type [t][i32].

Ct.testop:[t][i32]

t.relop

  • The instruction is valid with type [t t][i32].

Ct.relop:[t t][i32]

t2.cvtop_t1_sx?

  • The instruction is valid with type [t1][t2].

Ct2.cvtop_t1_sx?:[t1][t2]

Reference Instructions

ref.null ht

  • The heap type ht must be valid.

  • Then the instruction is valid with type [][(ref null ht)].

ChtokCref.null ht:[][(ref null ht)]

ref.func x

  • The function C.funcs[x] must be defined in the context.

  • Let dt be the defined type C.funcs[x].

  • The function index x must be contained in C.refs.

  • The instruction is valid with type [][(ref dt)].

C.funcs[x]=dtxC.refsCref.func x:[][(ref dt)]

ref.is_null

  • The instruction is valid with type [(ref null ht)][i32], for any valid heap type ht.

ChtokCref.is_null:[(ref null ht)][i32]

ref.as_non_null

  • The instruction is valid with type [(ref null ht)][(ref ht)], for any valid heap type ht.

ChtokCref.as_non_null:[(ref null ht)][(ref ht)]

ref.eq

  • The instruction is valid with type [(ref null eq)(ref null eq)][i32].

Cref.eq:[(ref null eq) (ref null eq)][i32]

ref.test rt

CrtokCrtokCrtrtCref.test rt:[rt][i32]

Note

The liberty to pick a supertype rt allows typing the instruction with the least precise super type of rt as input, that is, the top type in the corresponding heap subtyping hierarchy.

ref.cast rt

CrtokCrtokCrtrtCref.cast rt:[rt][rt]

Note

The liberty to pick a supertype rt allows typing the instruction with the least precise super type of rt as input, that is, the top type in the corresponding heap subtyping hierarchy.

Aggregate Reference Instructions

struct.new x

  • The defined type C.types[x] must exist.

  • The expansion of C.types[x] must be a structure type struct fieldtype.

  • For each field type fieldtypei in fieldtype:

    • Let fieldtypei be mut storagetypei.

    • Let ti be the value type unpack(storagetypei).

  • Let t be the concatenation of all ti.

  • Then the instruction is valid with type [t][(ref x)].

expand(C.types[x])=struct (mut st)Cstruct.new x:[(unpack(st))][(ref x)]

struct.new_default x

  • The defined type C.types[x] must exist.

  • The expansion of C.types[x] must be a structure type struct fieldtype.

  • For each field type fieldtypei in fieldtype:

    • Let fieldtypei be mut storagetypei.

    • Let ti be the value type unpack(storagetypei).

    • The type ti must be defaultable.

  • Let t be the concatenation of all ti.

  • Then the instruction is valid with type [][(ref x)].

expand(C.types[x])=struct (mut st)(Cunpack(st)defaultable)Cstruct.new_default x:[][(ref x)]

struct.get_sx? x y

  • The defined type C.types[x] must exist.

  • The expansion of C.types[x] must be a structure type struct fieldtype.

  • Let the field type mut storagetype be fieldtype[y].

  • Let t be the value type unpack(storagetype).

  • The extension sx must be present if and only if storagetype is a packed type.

  • Then the instruction is valid with type [(ref null x)][t].

expand(C.types[x])=struct ftft[y]=mut stsx?=ϵst=unpack(st)Cstruct.get_sx? x y:[(ref null x)][unpack(st)]

struct.set x y

  • The defined type C.types[x] must exist.

  • The expansion of C.types[x] must be a structure type struct fieldtype.

  • Let the field type mut storagetype be fieldtype[y].

  • The prefix mut must be var.

  • Let t be the value type unpack(storagetype).

  • Then the instruction is valid with type [(ref null x) t][].

expand(C.types[x])=struct ftft[y]=var stCstruct.set x y:[(ref null x) unpack(st)][]

array.new x

  • The defined type C.types[x] must exist.

  • The expansion of C.types[x] must be an array type array fieldtype.

  • Let fieldtype be mut storagetype.

  • Let t be the value type unpack(storagetype).

  • Then the instruction is valid with type [t i32][(ref x)].

expand(C.types[x])=array (mut st)Carray.new x:[unpack(st) i32][(ref x)]

array.new_default x

  • The defined type C.types[x] must exist.

  • The expansion of C.types[x] must be an array type array fieldtype.

  • Let fieldtype be mut storagetype.

  • Let t be the value type unpack(storagetype).

  • The type t must be defaultable.

  • Then the instruction is valid with type [i32][(ref x)].

expand(C.types[x])=array (mut st)Cunpack(st)defaultableCarray.new x:[i32][(ref x)]

array.new_fixed x n

  • The defined type C.types[x] must exist.

  • The expansion of C.types[x] must be an array type array fieldtype.

  • Let fieldtype be mut storagetype.

  • Let t be the value type unpack(storagetype).

  • Then the instruction is valid with type [tn][(ref x)].

expand(C.types[x])=array (mut st)Carray.new_fixed x n:[unpack(st)n][(ref x)]

array.new_elem x y

expand(C.types[x])=array (mut rt)CC.elems[y]rtCarray.new_elem x y:[i32 i32][(ref x)]

array.new_data x y

expand(C.types[x])=array (mut st)unpack(st)=numtypeunpack(st)=vectypeC.datas[y]=okCarray.new_data x y:[i32 i32][(ref x)]

array.get_sx? x

  • The defined type C.types[x] must exist.

  • The expansion of C.types[x] must be an array type array fieldtype.

  • Let the field type mut storagetype be fieldtype.

  • Let t be the value type unpack(storagetype).

  • The extension sx must be present if and only if storagetype is a packed type.

  • Then the instruction is valid with type [(ref null x) i32][t].

expand(C.types[x])=array (mut st)sx?=ϵst=unpack(st)Carray.get_sx? x:[(ref null x) i32][unpack(st)]

array.set x

  • The defined type C.types[x] must exist.

  • The expansion of C.types[x] must be an array type array fieldtype.

  • Let the field type mut storagetype be fieldtype.

  • The prefix mut must be var.

  • Let t be the value type unpack(storagetype).

  • Then the instruction is valid with type [(ref null x) i32 t][].

expand(C.types[x])=array (var st)Carray.set x:[(ref null x) i32 unpack(st)][]

array.len

  • The the instruction is valid with type [(ref null array)][i32].

Carray.len:[(ref null array)][i32]

array.fill x

  • The defined type C.types[x] must exist.

  • The expansion of C.types[x] must be an array type array fieldtype.

  • Let the field type mut storagetype be fieldtype.

  • The prefix mut must be var.

  • Let t be the value type unpack(storagetype).

  • Then the instruction is valid with type [(ref null x) i32 t i32][].

expand(C.types[x])=array (var st)Carray.fill x:[(ref null x) i32 unpack(st) i32][]

array.copy x y

  • The defined type C.types[x] must exist.

  • The expansion of C.types[x] must be an array type array fieldtype1.

  • Let the field type mut1 storagetype1 be fieldtype1.

  • The prefix mut1 must be var.

  • The defined type C.types[y] must exist.

  • The expansion of C.types[y] must be an array type array fieldtype2.

  • Let the field type mut2 storagetype2 be fieldtype2.

  • The storage type storagetype2 must match storagetype1.

  • Then the instruction is valid with type [(ref null x) i32 (ref null y) i32 i32][].

expand(C.types[x])=array (var st1)expand(C.types[y])=array (mut st2)Cst2st1Carray.copy x y:[(ref null x) i32 (ref null y) i32 i32][]

array.init_data x y

expand(C.types[x])=array (var st)unpack(st)=numtypeunpack(st)=vectypeC.datas[y]=okCarray.init_data x y:[(ref null x) i32 i32 i32][]

array.init_elem x y

expand(C.types[x])=array (var rt)CC.elems[y]rtCarray.init_elem x y:[(ref null x) i32 i32 i32][]

Scalar Reference Instructions

ref.i31

  • The instruction is valid with type [i32][(ref i31)].

Cref.i31:[i32][(ref i31)]

i31.get_sx

  • The instruction is valid with type [(ref null i31)][i32].

Ci31.get_sx:[(ref null i31)][i32]

External Reference Instructions

any.convert_extern

  • The instruction is valid with type [(ref null1? extern)][(ref null2? any)] for any null1? that equals null2?.

null1?=null2?Cany.convert_extern:[(ref null1? extern)][(ref null2? any)]

extern.convert_any

  • The instruction is valid with type [(ref null1? any)][(ref null2? extern)] for any null1? that equals null2?.

null1?=null2?Cextern.convert_any:[(ref null1? any)][(ref null2? extern)]

Vector Instructions

Vector instructions can have a prefix to describe the shape of the operand. Packed numeric types, i8 and i16, are not value types. An auxiliary function maps such packed type shapes to value types:

unpack(txN)=unpack(t)

The following auxiliary function denotes the number of lanes in a vector shape, i.e., its dimension:

dim(txN)=N

v128.const c

  • The instruction is valid with type [][v128].

Cv128.const c:[][v128]

v128.vvunop

  • The instruction is valid with type [v128][v128].

Cv128.vvunop:[v128][v128]

v128.vvbinop

  • The instruction is valid with type [v128 v128][v128].

Cv128.vvbinop:[v128 v128][v128]

v128.vvternop

  • The instruction is valid with type [v128 v128 v128][v128].

Cv128.vvternop:[v128 v128 v128][v128]

v128.vvtestop

  • The instruction is valid with type [v128][i32].

Cv128.vvtestop:[v128][i32]

i8x16.swizzle

  • The instruction is valid with type [v128 v128][v128].

Ci8x16.swizzle:[v128 v128][v128]

i8x16.shuffle laneidx16

  • For all laneidxi, in laneidx16, laneidxi must be smaller than 32.

  • The instruction is valid with type [v128 v128][v128].

(laneidx<32)16Ci8x16.shuffle laneidx16:[v128 v128][v128]

shape.splat

  • Let t be unpack(shape).

  • The instruction is valid with type [t][v128].

Cshape.splat:[unpack(shape)][v128]

shape.extract_lane_sx? laneidx

  • The lane index laneidx must be smaller than dim(shape).

  • The instruction is valid with type [v128][unpack(shape)].

laneidx<dim(shape)Cshape.extract_lane_sx? laneidx:[v128][unpack(shape)]

shape.replace_lane laneidx

  • The lane index laneidx must be smaller than dim(shape).

  • Let t be unpack(shape).

  • The instruction is valid with type [v128 t][v128].

laneidx<dim(shape)Cshape.replace_lane laneidx:[v128 unpack(shape)][v128]

shape.vunop

  • The instruction is valid with type [v128][v128].

Cshape.vunop:[v128][v128]

shape.vbinop

  • The instruction is valid with type [v128 v128][v128].

Cshape.vbinop:[v128 v128][v128]

shape.vrelop

  • The instruction is valid with type [v128 v128][v128].

Cshape.vrelop:[v128 v128][v128]

ishape.vishiftop

  • The instruction is valid with type [v128 i32][v128].

Cishape.vishiftop:[v128 i32][v128]

shape.vtestop

  • The instruction is valid with type [v128][i32].

Cshape.vtestop:[v128][i32]

shape.vcvtop_half?_shape_sx?_zero?

  • The instruction is valid with type [v128][v128].

Cshape.vcvtop_half?_shape_sx?_zero?:[v128][v128]

ishape1.narrow_ishape2_sx

  • The instruction is valid with type [v128 v128][v128].

Cishape1.narrow_ishape2_sx:[v128 v128][v128]

ishape.bitmask

  • The instruction is valid with type [v128][i32].

Cishape.bitmask:[v128][i32]

ishape1.dot_ishape2_s

  • The instruction is valid with type [v128 v128][v128].

Cishape1.dot_ishape2_s:[v128 v128][v128]

ishape1.extmul_half_ishape2_sx

  • The instruction is valid with type [v128 v128][v128].

Cishape1.extmul_half_ishape2_sx:[v128 v128][v128]

ishape1.extadd_pairwise_ishape2_sx

  • The instruction is valid with type [v128][v128].

Cishape1.extadd_pairwise_ishape2_sx:[v128][v128]

Parametric Instructions

drop

CtokCdrop:[t][]

Note

Both drop and select without annotation are value-polymorphic instructions.

select (t)?

C[t]okCselect t:[t t i32][t]C[t]okC[t][numtype]Cselect:[t t i32][t]tvectypeCselect:[t t i32][t]

Note

In future versions of WebAssembly, select may allow more than one value per choice.

Variable Instructions

local.get x

  • The local C.locals[x] must be defined in the context.

  • Let init t be the local type C.locals[x].

  • The initialization status init must be set.

  • Then the instruction is valid with type [][t].

C.locals[x]=set tClocal.get x:[][t]

local.set x

  • The local C.locals[x] must be defined in the context.

  • Let init t be the local type C.locals[x].

  • Then the instruction is valid with type [t]x[].

C.locals[x]=init tClocal.set x:[t]x[]

local.tee x

  • The local C.locals[x] must be defined in the context.

  • Let init t be the local type C.locals[x].

  • Then the instruction is valid with type [t]x[t].

C.locals[x]=init tClocal.tee x:[t]x[t]

global.get x

  • The global C.globals[x] must be defined in the context.

  • Let mut t be the global type C.globals[x].

  • Then the instruction is valid with type [][t].

C.globals[x]=mut tCglobal.get x:[][t]

global.set x

  • The global C.globals[x] must be defined in the context.

  • Let mut t be the global type C.globals[x].

  • The mutability mut must be var.

  • Then the instruction is valid with type [t][].

C.globals[x]=var tCglobal.set x:[t][]

Table Instructions

table.get x

  • The table C.tables[x] must be defined in the context.

  • Let limits t be the table type C.tables[x].

  • Then the instruction is valid with type [i32][t].

C.tables[x]=limits tCtable.get x:[i32][t]

table.set x

  • The table C.tables[x] must be defined in the context.

  • Let limits t be the table type C.tables[x].

  • Then the instruction is valid with type [i32 t][].

C.tables[x]=limits tCtable.set x:[i32 t][]

table.size x

  • The table C.tables[x] must be defined in the context.

  • Then the instruction is valid with type [][i32].

C.tables[x]=tabletypeCtable.size x:[][i32]

table.grow x

  • The table C.tables[x] must be defined in the context.

  • Let limits t be the table type C.tables[x].

  • Then the instruction is valid with type [t i32][i32].

C.tables[x]=limits tCtable.grow x:[t i32][i32]

table.fill x

  • The table C.tables[x] must be defined in the context.

  • Let limits t be the table type C.tables[x].

  • Then the instruction is valid with type [i32 t i32][].

C.tables[x]=limits tCtable.fill x:[i32 t i32][]

table.copy x y

  • The table C.tables[x] must be defined in the context.

  • Let limits1 t1 be the table type C.tables[x].

  • The table C.tables[y] must be defined in the context.

  • Let limits2 t2 be the table type C.tables[y].

  • The reference type t2 must match t1.

  • Then the instruction is valid with type [i32 i32 i32][].

C.tables[x]=limits1 t1C.tables[y]=limits2 t2Ct2t1Ctable.copy x y:[i32 i32 i32][]

table.init x y

  • The table C.tables[x] must be defined in the context.

  • Let limits t1 be the table type C.tables[x].

  • The element segment C.elems[y] must be defined in the context.

  • Let t2 be the reference type C.elems[y].

  • The reference type t2 must match t1.

  • Then the instruction is valid with type [i32 i32 i32][].

C.tables[x]=limits t1C.elems[y]=t2Ct2t1Ctable.init x y:[i32 i32 i32][]

elem.drop x

  • The element segment C.elems[x] must be defined in the context.

  • Then the instruction is valid with type [][].

C.elems[x]=tCelem.drop x:[][]

Memory Instructions

t.load memarg

  • The memory C.mems[0] must be defined in the context.

  • The alignment 2memarg.align must not be larger than the bit width of t divided by 8.

  • Then the instruction is valid with type [i32][t].

C.mems[0]=memtype2memarg.align|t|/8Ct.load memarg:[i32][t]

t.loadN_sx memarg

  • The memory C.mems[0] must be defined in the context.

  • The alignment 2memarg.align must not be larger than N/8.

  • Then the instruction is valid with type [i32][t].

C.mems[0]=memtype2memarg.alignN/8Ct.loadN_sx memarg:[i32][t]

t.store memarg

  • The memory C.mems[0] must be defined in the context.

  • The alignment 2memarg.align must not be larger than the bit width of t divided by 8.

  • Then the instruction is valid with type [i32 t][].

C.mems[0]=memtype2memarg.align|t|/8Ct.store memarg:[i32 t][]

t.storeN memarg

  • The memory C.mems[0] must be defined in the context.

  • The alignment 2memarg.align must not be larger than N/8.

  • Then the instruction is valid with type [i32 t][].

C.mems[0]=memtype2memarg.alignN/8Ct.storeN memarg:[i32 t][]

v128.loadNxM_sx memarg

  • The memory C.mems[0] must be defined in the context.

  • The alignment 2memarg.align must not be larger than N/8M.

  • Then the instruction is valid with type [i32][v128].

C.mems[0]=memtype2memarg.alignN/8MCv128.loadNxM_sx memarg:[i32][v128]

v128.loadN_splat memarg

  • The memory C.mems[0] must be defined in the context.

  • The alignment 2memarg.align must not be larger than N/8.

  • Then the instruction is valid with type [i32][v128].

C.mems[0]=memtype2memarg.alignN/8Cv128.loadN_splat memarg:[i32][v128]

v128.loadN_zero memarg

  • The memory C.mems[0] must be defined in the context.

  • The alignment 2memarg.align must not be larger than N/8.

  • Then the instruction is valid with type [i32][v128].

C.mems[0]=memtype2memarg.alignN/8Cv128.loadN_zero memarg:[i32][v128]

v128.loadN_lane memarg laneidx

  • The lane index laneidx must be smaller than 128/N.

  • The memory C.mems[0] must be defined in the context.

  • The alignment 2memarg.align must not be larger than N/8.

  • Then the instruction is valid with type [i32 v128][v128].

laneidx<128/NC.mems[0]=memtype2memarg.align<N/8Cv128.loadN_lane memarg laneidx:[i32 v128][v128]

v128.storeN_lane memarg laneidx

  • The lane index laneidx must be smaller than 128/N.

  • The memory C.mems[0] must be defined in the context.

  • The alignment 2memarg.align must not be larger than N/8.

  • Then the instruction is valid with type [i32 v128][v128].

laneidx<128/NC.mems[0]=memtype2memarg.align<N/8Cv128.storeN_lane memarg laneidx:[i32 v128][]

memory.size

  • The memory C.mems[0] must be defined in the context.

  • Then the instruction is valid with type [][i32].

C.mems[0]=memtypeCmemory.size:[][i32]

memory.grow

  • The memory C.mems[0] must be defined in the context.

  • Then the instruction is valid with type [i32][i32].

C.mems[0]=memtypeCmemory.grow:[i32][i32]

memory.fill

  • The memory C.mems[0] must be defined in the context.

  • Then the instruction is valid with type [i32 i32 i32][].

C.mems[0]=memtypeCmemory.fill:[i32 i32 i32][]

memory.copy

  • The memory C.mems[0] must be defined in the context.

  • Then the instruction is valid with type [i32 i32 i32][].

C.mems[0]=memtypeCmemory.copy:[i32 i32 i32][]

memory.init x

  • The memory C.mems[0] must be defined in the context.

  • The data segment C.datas[x] must be defined in the context.

  • Then the instruction is valid with type [i32 i32 i32][].

C.mems[0]=memtypeC.datas[x]=okCmemory.init x:[i32 i32 i32][]

data.drop x

  • The data segment C.datas[x] must be defined in the context.

  • Then the instruction is valid with type [][].

C.datas[x]=okCdata.drop x:[][]

Control Instructions

nop

  • The instruction is valid with type [][].

Cnop:[][]

unreachable

  • The instruction is valid with any valid type of the form [t1][t2].

C[t1][t2]okCunreachable:[t1][t2]

Note

The unreachable instruction is stack-polymorphic.

block blocktype instr end

  • The block type must be valid as some instruction type [t1][t2].

  • Let C be the same context as C, but with the result type [t2] prepended to the labels vector.

  • Under context C, the instruction sequence instr must be valid with type [t1][t2].

  • Then the compound instruction is valid with type [t1][t2].

Cblocktype:[t1][t2]C,labels[t2]instr:[t1][t2]Cblock blocktype instr end:[t1][t2]

Note

The notation C,labels[t] inserts the new label type at index 0, shifting all others.

loop blocktype instr end

  • The block type must be valid as some instruction type [t1]x[t2].

  • Let C be the same context as C, but with the result type [t1] prepended to the labels vector.

  • Under context C, the instruction sequence instr must be valid with type [t1][t2].

  • Then the compound instruction is valid with type [t1][t2].

Cblocktype:[t1][t2]C,labels[t1]instr:[t1][t2]Cloop blocktype instr end:[t1][t2]

Note

The notation C,labels[t] inserts the new label type at index 0, shifting all others.

if blocktype instr1 else instr2 end

  • The block type must be valid as some instruction type [t1][t2].

  • Let C be the same context as C, but with the result type [t2] prepended to the labels vector.

  • Under context C, the instruction sequence instr1 must be valid with type [t1][t2].

  • Under context C, the instruction sequence instr2 must be valid with type [t1][t2].

  • Then the compound instruction is valid with type [t1 i32][t2].

Cblocktype:[t1][t2]C,labels[t2]instr1:[t1][t2]C,labels[t2]instr2:[t1][t2]Cif blocktype instr1 else instr2 end:[t1 i32][t2]

Note

The notation C,labels[t] inserts the new label type at index 0, shifting all others.

br l

  • The label C.labels[l] must be defined in the context.

  • Let [t] be the result type C.labels[l].

  • Then the instruction is valid with any valid type of the form [t1 t][t2].

C.labels[l]=[t]C[t1 t][t2]okCbr l:[t1 t][t2]

Note

The label index space in the context C contains the most recent label first, so that C.labels[l] performs a relative lookup as expected.

The br instruction is stack-polymorphic.

br_if l

  • The label C.labels[l] must be defined in the context.

  • Let [t] be the result type C.labels[l].

  • Then the instruction is valid with type [t i32][t].

C.labels[l]=[t]Cbr_if l:[t i32][t]

Note

The label index space in the context C contains the most recent label first, so that C.labels[l] performs a relative lookup as expected.

br_table l lN

  • The label C.labels[lN] must be defined in the context.

  • For each label li in l, the label C.labels[li] must be defined in the context.

  • There must be a sequence t of value types, such that:

    • The result type [t] matches C.labels[lN].

    • For all li in l, the result type [t] matches C.labels[li].

  • Then the instruction is valid with any valid type of the form [t1 t i32][t2].

(C[t]C.labels[l])C[t]C.labels[lN]C[t1 t i32][t2]okCbr_table l lN:[t1 t i32][t2]

Note

The label index space in the context C contains the most recent label first, so that C.labels[li] performs a relative lookup as expected.

The br_table instruction is stack-polymorphic.

Furthermore, the result type [t] is also chosen non-deterministically in this rule. Although it may seem necessary to compute [t] as the greatest lower bound of all label types in practice, a simple linear algorithm does not require this.

br_on_null l

  • The label C.labels[l] must be defined in the context.

  • Let [t] be the result type C.labels[l].

  • Then the instruction is valid with type [t (ref null ht)][t (ref ht)] for any valid heap type ht.

C.labels[l]=[t]ChtokCbr_on_null l:[t (ref null ht)][t (ref ht)]

br_on_non_null l

  • The label C.labels[l] must be defined in the context.

  • Let [t] be the result type C.labels[l].

  • The result type [t] must contain at least one type.

  • Let the value type tl be the last element in the sequence t, and [t] the remainder of the sequence preceding it.

  • The value type tl must be a reference type of the form ref null? ht.

  • Then the instruction is valid with type [t (ref null ht)][t].

C.labels[l]=[t (ref ht)]Cbr_on_non_null l:[t (ref null ht)][t]

br_on_cast l rt1 rt2

C.labels[l]=[t rt]Crt1okCrt2okCrt2rt1Crt2rtCbr_on_cast l rt1 rt2:[t rt1][t rt1rt2]

br_on_cast_fail l rt1 rt2

C.labels[l]=[t rt]Crt1okCrt2okCrt2rt1Crt1rt2rtCbr_on_cast_fail l rt1 rt2:[t rt1][t rt2]

return

  • The return type C.return must not be absent in the context.

  • Let [t] be the result type of C.return.

  • Then the instruction is valid with any valid type of the form [t1][t2].

C.return=[t]C[t1 t][t2]okCreturn:[t1 t][t2]

Note

The return instruction is stack-polymorphic.

C.return is absent (set to ϵ) when validating an expression that is not a function body. This differs from it being set to the empty result type ([ϵ]), which is the case for functions not returning anything.

call x

  • The function C.funcs[x] must be defined in the context.

  • The expansion of C.funcs[x] must be a function type func [t1][t2].

  • Then the instruction is valid with type [t1][t2].

expand(C.funcs[x])=func [t1][t2]Ccall x:[t1][t2]

call_ref x

  • The type C.types[x] must be defined in the context.

  • The expansion of C.funcs[x] must be a function type func [t1][t2].

  • Then the instruction is valid with type [t1 (ref null x)][t2].

expand(C.types[x])=func [t1][t2]Ccall_ref x:[t1 (ref null x)][t2]

call_indirect x y

  • The table C.tables[x] must be defined in the context.

  • Let limits t be the table type C.tables[x].

  • The reference type t must match type ref null func.

  • The type C.types[y] must be defined in the context.

  • The expansion of C.types[y] must be a function type func [t1][t2].

  • Then the instruction is valid with type [t1 i32][t2].

C.tables[x]=limits tCtref null funcexpand(C.types[y])=func [t1][t2]Ccall_indirect x y:[t1 i32][t2]

return_call x

  • The return type C.return must not be absent in the context.

  • The function C.funcs[x] must be defined in the context.

  • The expansion of C.funcs[x] must be a function type func [t1][t2].

  • The result type [t2] must match C.return.

  • Then the instruction is valid with any valid type [t3 t1][t4].

expand(C.funcs[x])=func [t1][t2]C[t2]C.returnCreturn_call x:[t3 t1][t4]

Note

The return_call instruction is stack-polymorphic.

return_call_ref x

  • The type C.types[x] must be defined in the context.

  • The expansion of C.types[x] must be a function type func [t1][t2].

  • The result type [t2] must match C.return.

  • Then the instruction is valid with any valid type [t3 t1 (ref null x)][t4].

expand(C.types[x])=func [t1][t2]C[t2]C.returnCcall_ref x:[t3 t1 (ref null x)][t4]

Note

The return_call_ref instruction is stack-polymorphic.

return_call_indirect x y

  • The return type C.return must not be empty in the context.

  • The table C.tables[x] must be defined in the context.

  • Let limits t be the table type C.tables[x].

  • The reference type t must match type ref null func.

  • The type C.types[y] must be defined in the context.

  • The expansion of C.types[y] must be a function type func [t1][t2].

  • The result type [t2] must match C.return.

  • Then the instruction is valid with type [t3 t1 i32][t4], for any sequences of value types t3 and t4.

C.tables[x]=limits tCtref null funcexpand(C.types[y])=func [t1][t2]C[t2]C.returnCreturn_call_indirect x y:[t3 t1 i32][t4]

Note

The return_call_indirect instruction is stack-polymorphic.

Instruction Sequences

Typing of instruction sequences is defined recursively.

Empty Instruction Sequence: ϵ

  • The empty instruction sequence is valid with type [][].

Cϵ:[][]

Non-empty Instruction Sequence: instr instr

  • The instruction instr must be valid with some type [t1]x1[t2].

  • Let C be the same context as C, but with:

  • Under the context C, the instruction sequence instr must be valid with some type [t2]x2[t3].

  • Then the combined instruction sequence is valid with type [t1]x1x2[t3].

Cinstr:[t1]x1[t2](C.locals[x1]=init t)Cinstr:[t2]x2[t3]C=C (withC.locals[x1]=set t)Cinstr instr:[t1]x1x2[t2 t3]

Subsumption for instr

  • The instruction sequence instr must be valid with some type instrtype.

  • The instruction type instrtype: must be a valid

  • The instruction type instrtype must match the type instrtype.

  • Then the instruction sequence instr is also valid with type instrtype.

Cinstr:instrtypeCinstrtypeokCinstrtypeinstrtypeCinstr:instrtype

Note

In combination with the previous rule, subsumption allows to compose instructions whose types would not directly fit otherwise. For example, consider the instruction sequence

(i32.const 1) (i32.const 1) i32.add

To type this sequence, its subsequence (i32.const 1) i32.add needs to be valid with an intermediate type. But the direct type of (i32.const 1) is [][i32], not matching the two inputs expected by i32.add. The subsumption rule allows to weaken the type of (i32.const 1) to the supertype [i32][i32 i32], such that it can be composed with i32.add and yields the intermediate type [i32][i32] for the subsequence. That can in turn be composed with the first constant.

Furthermore, subsumption allows to drop init variables x from the instruction type in a context where they are not needed, for example, at the end of the body of a block.

Expressions

Expressions expr are classified by result types of the form [t].

instr end

  • The instruction sequence instr must be valid with type [][t].

  • Then the expression is valid with result type [t].

Cinstr:[][t]Cinstr end:[t]

Constant Expressions

  • In a constant expression instr end all instructions in instr must be constant.

  • A constant instruction instr must be:

    • either of the form t.const c,

    • or of the form ref.null,

    • or of the form ref.i31,

    • or of the form ref.func x,

    • or of the form struct.new x,

    • or of the form struct.new_default x,

    • or of the form array.new x,

    • or of the form array.new_default x,

    • or of the form array.new_fixed x,

    • or of the form any.convert_extern,

    • or of the form extern.convert_any,

    • or of the form global.get x, in which case C.globals[x] must be a global type of the form const t.

(Cinstrconst)Cinstr endconst
Ct.const cconstC.globals[x]=const tCglobal.get xconst
Cref.null tconstCref.i31constCref.func xconst
Cstruct.new xconstCstruct.new_default xconst
Carray.new xconstCarray.new_default xconstCarray.new_fixed xconst
Cany.convert_externconstCextern.convert_anyconst

Note

Currently, constant expressions occurring in globals are further constrained in that contained global.get instructions are only allowed to refer to imported or previously defined globals. Constant expressions occurring in tables may only have global.get instructions that refer to imported globals. This is enforced in the validation rule for modules by constraining the context C accordingly.

The definition of constant expression may be extended in future versions of WebAssembly.