Execution

Runtime Structure

Stack

Exception Handlers

Legacy exception handlers are installed by try instructions. Instead of branch labels, their catch clauses have instruction blocks associated with them. Furthermore, a delegate handler is associated with a label index to implicitly rewthrow to:

catch::=|catch tagidx instr|catch_all tagidx instr|delegate labelidx

Administrative Instructions

Administrative instructions are extended with the caught instruction that models exceptions caught by legacy exception handlers.

instr::=|caughtn{exnaddr} instr end

Block Contexts

Block contexts are extended to include caught instructions:

Bk::=|caughtn {exnaddr} Bk end

Throw Contexts

Throw contexts are also extended to include caught instructions:

T::=|caughtn{exnaddr} T end

Instructions

Control Instructions

try blocktype instr1 (catch x instr2) (catch_all instr3)? end

  1. Assert: due to validation, expandF(blocktype) is defined.

  2. Let [t1m][t2n] be the function type expandF(blocktype).

  3. Let L be the label whose arity is n and whose continuation is the end of the try instruction.

  4. Assert: due to validation, there are at least m values on the top of the stack.

  5. Pop the values valm from the stack.

  6. Let F be the current frame.

  7. For each catch clause (catch xi instr2i) do:

    1. Assert: due to validation, F.module.tagaddrs[xi] exists.

    2. Let ai be the tag address F.module.tagaddrs[xi].

    3. Let catchi be the catch clause (catch ai instr2i).

  8. If there is a catch-all clause (catch_all instr3), then:

    1. Let catch? be the handler (catch_all instr3).

  9. Else:

    1. Let catch? be empty.

  10. Let catch be the concatenation of catchi and catch?.

  11. Enter the block valm instr1 with label L and exception handler handlern{catch}.

 F;valm (try bt instr1 (catch x instr2) (catch_all instr3)? endF;labeln{ϵ} (handlern{(catch ax instr2) (catch_all instr3)?} valm instr1 end) end(ifexpandF(bt)=[t1m][t2n](F.module.tagaddrs[x]=ax))

try blocktype instr delegate l

  1. Assert: due to validation, expandF(blocktype) is defined.

  2. Let [t1m][t2n] be the function type expandF(blocktype).

  3. Let L be the label whose arity is n and whose continuation is the end of the try instruction.

  4. Let H be the exception handler l, targeting the l-th surrounding block.

  5. Assert: due to validation, there are at least m values on the top of the stack.

  6. Pop the values valm from the stack.

  7. Enter the block valm instr with label L and exception handler HANDLER_n{DELEGATE~l}.

 F;valm (try bt instr delegate l)F;labeln{ϵ} (handlern{delegate l} valm instr end) end(ifexpandF(bt)=[t1m][t2n])

throw_ref

  1. Let F be the current frame.

  2. Assert: due to validation, a reference is on the top of the stack.

  3. Pop the reference ref from the stack.

  4. If ref is ref.null ht, then:

    1. Trap.

  5. Assert: due to validation, ref is an exception reference.

  6. Let ref.exn ea be ref.

  7. Assert: due to validation, S.exns[ea] exists.

  8. Let exn be the exception instance S.exns[ea].

  9. Let a be the tag address exn.tag.

  10. While the stack is not empty and the top of the stack is not an exception handler, do:

  1. Pop the top element from the stack.

  1. Assert: the stack is now either empty, or there is an exception handler on the top of the stack.

  2. If the stack is empty, then:

  1. Return the exception (ref.exn a) as a result.

  1. Assert: there is an exception handler on the top of the stack.

  2. Pop the exception handler handlern{catch} from the stack.

  3. If catch is empty, then:

    1. Push the exception reference ref.exn ea back to the stack.

    2. Execute the instruction throw_ref again.

  4. Else:

    1. Let catch1 be the first catch clause in catch and catch the remaining clauses.

    2. If catch1 is of the form catch x l and the exception address a equals F.module.tagaddrs[x], then:

      1. Push the values exn.fields to the stack.

      2. Execute the instruction br l.

    3. Else if catch1 is of the form catch_ref x l and the exception address a equals F.module.tagaddrs[x], then:

      1. Push the values exn.fields to the stack.

      2. Push the exception reference ref.exn ea to the stack.

      3. Execute the instruction br l.

    4. Else if catch1 is of the form catch_all l, then:

      1. Execute the instruction br l.

    5. Else if catch1 is of the form catch_all_ref l, then:

      1. Push the exception reference ref.exn ea to the stack.

      2. Execute the instruction br l.

    6. Else if catch1 is of the form catch x instr and the exception address a equals F.module.tagaddrs[x], then:

      1. Push the caught exception caughtn{ea} to the stack.

      2. Push the values exn.fields to the stack.

      3. Enter the catch block instr.

    7. Else if catch1 is of the form catch_all instr, then:

      1. Push the caught exception caughtn{ea} to the stack.

      2. Enter the catch block instr.

    8. Else if catch1 is of the form delegate l, then:

      1. Assert: due to validation, the stack contains at least l labels.

      2. Repeat l times:

        • While the top of the stack is not a label, do:

          • Pop the top element from the stack.

      3. Assert: due to validation, the top of the stack now is a label.

      4. Pop the label from the stack.

      5. Push the exception reference ref.exn ea back to the stack.

      6. Execute the instruction throw_ref again.

    9. Else:

      1. Push the modified handler handlern{catch} back to the stack.

      2. Push the exception reference ref.exn ea back to the stack.

      3. Execute the instruction throw_ref again.

 handlern{(catch x instr) catch} T[(ref.exn a) throw_ref] endcaughtn{a} exn.fields instr end(ifexn=S.exns[a]exn.tag=F.module.tagaddrs[x])handlern{(catch_all instr) catch} T[(ref.exn a) throw_ref] endcaughtn{a} instr endBl[handlern{(delegate l) catch} T[(ref.exn a) throw_ref] end](ref.exn a) throw_ref

rethrow l

  1. Assert: due to validation, the stack contains at least l+1 labels.

  2. Let L be the l-th label appearing on the stack, starting from the top and counting from zero.

  3. Assert: due to validation, L is a catch label, i.e., a label of the form (catch [t]), which is a label followed by a caught exception in an active catch clause.

  4. Let a be the caught exception address.

  5. Push the value ref.exn a onto the stack.

  6. Execute the instruction throw_ref.

 caughtn{a} Bl[rethrow l] endcaughtn{a} Bl[(ref.exn a) throw_ref] end

Entering a catch block

  1. Jump to the start of the instruction sequence instr.

Exiting a catch block

When the end of a catch block is reached without a jump, thrown exception, or trap, then the following steps are performed.

  1. Let valm be the values on the top of the stack.

  2. Pop the values valm from the stack.

  3. Assert: due to validation, a caught exception is now on the top of the stack.

  4. Pop the caught exception from the stack.

  5. Push valm back to the stack.

  6. Jump to the position after the end of the administrative instruction associated with the caught exception.

caughtn{a} valm endvalm

Note

A caught exception can only be rethrown from the scope of the administrative instruction associated with it, i.e., from the scope of the catch or catch_all block of a legacy try instruction. Upon exit from that block, the caught exception is discarded.