Flutter Engine
The Flutter Engine
Exceptions Implementation

This page describes how exceptions throwing and catching is implemented in the VM.

Intermediate Language

Dart VM's IL does not explicitly represent exceptional control flow in its flow graph, there are no explicit exceptional edges connecting potentially throwing instructions (e.g. calls) with corresponding catch blocks. Instead this connection is defined at the block level: all exceptions that occur in any block with the given try_index will be caught by CatchBlockEntry with the equal catch_try_index.

Catch Block Entry

For optimized code this means that data flow associated with exceptional control flow is also represented implicitly: due to the absence of explicit exceptional edges the data flow can't be represented using explicit phi-functions. Instead in optimized code each CatchBlockEntry is treated almost as if it was an independent entry into the function: for each variable v CatchBlockEntry will contain a Parameter(...) instruction restoring variable state at catch entry from a fixed location on the stack. When an exception is thrown runtime system takes care of populating these stack slots with right values - current state of corresponding local variables. It's easy to see a parallel between these Parameter(...) instructions and Phi(...) instructions that would be used if exception control flow would be explicit.

Catch Block Entry

How does runtime system populate stack slots corresponding to these Parameter(...) instructions? During compilation necessary information is available in deoptimization environment attached to the instruction. This environment encodes the state of local variables in terms of SSA values i.e. if we need to reconstruct unoptimized frame which SSA value should be stored into the given local variable (see Optimized IL for an overview). However the way we use these information for exception handling is slightly different in JIT and AOT modes.

AOT mode

AOT mode does not support deoptimization and thus AOT compiler does not associate any deoptimization metadata with generated code. Instead deoptimization environments associated with instructions that can throw are converted into CatchEntryMoves metadata during code generation and resulting metadata is stored RawCode::catch_entry_moves_maps_ in a compressed form.

CatchEntryMoves is essentially a sequence of moves which runtime needs to perform to create the state that catch entry expects. There are three types of moves:

  • *(FP + Dst) <- ObjectPool[PoolIndex] - a move of a constant from an object pool;
  • *(FP + Dst) <- *(FP + Src) - a move of a tagged value;
  • *(FP + Dst) <- Box<Rep>(*(FP + Src)) - a boxing operation for an untagged value;

When an exception is caught runtime decompresses the metadata associated with the call site which has thrown an exception and uses it to prepare the state of the stack for the catch block entry. See ExceptionHandlerFinder::{ReadCompressedCatchEntryMoves, ExecuteCatchEntryMoves}.

NOTE: See this design/motivation document for CatchEntryMoves metadata

JIT mode

JIT mode heavily relies on deoptimization and all call instructions have (lazy) deoptimization environments associated with them. These environments are converted to deoptimization instructions during code generation and stored on the Code object.

When an exception is caught the runtime system converts deoptimization environment associated with the call site that threw an exception into CatchEntryMoves and then uses it to prepare the state of the stack for the catch block entry. See ExceptionHandlerFinder::{GetCatchEntryMovesFromDeopt, ExecuteCatchEntryMoves}.

Constructing CatchEntryMoves dynamically from deoptimization instructions allows to avoid unnecessary duplication of the metadata and save memory: as deoptimization environments contain all information necessary for constructing correct stack state.