pub enum TerminatorKind<'tcx> {
Show 15 variants
Goto {
target: BasicBlock,
},
SwitchInt {
discr: Operand<'tcx>,
targets: SwitchTargets,
},
UnwindResume,
UnwindTerminate(UnwindTerminateReason),
Return,
Unreachable,
Drop {
place: Place<'tcx>,
target: BasicBlock,
unwind: UnwindAction,
replace: bool,
},
Call {
func: Operand<'tcx>,
args: Box<[Spanned<Operand<'tcx>>]>,
destination: Place<'tcx>,
target: Option<BasicBlock>,
unwind: UnwindAction,
call_source: CallSource,
fn_span: Span,
},
TailCall {
func: Operand<'tcx>,
args: Box<[Spanned<Operand<'tcx>>]>,
fn_span: Span,
},
Assert {
cond: Operand<'tcx>,
expected: bool,
msg: Box<AssertKind<Operand<'tcx>>>,
target: BasicBlock,
unwind: UnwindAction,
},
Yield {
value: Operand<'tcx>,
resume: BasicBlock,
resume_arg: Place<'tcx>,
drop: Option<BasicBlock>,
},
CoroutineDrop,
FalseEdge {
real_target: BasicBlock,
imaginary_target: BasicBlock,
},
FalseUnwind {
real_target: BasicBlock,
unwind: UnwindAction,
},
InlineAsm {
asm_macro: InlineAsmMacro,
template: &'tcx [InlineAsmTemplatePiece],
operands: Box<[InlineAsmOperand<'tcx>]>,
options: InlineAsmOptions,
line_spans: &'tcx [Span],
targets: Box<[BasicBlock]>,
unwind: UnwindAction,
},
}
Expand description
The various kinds of terminators, representing ways of exiting from a basic block.
A note on unwinding: Panics may occur during the execution of some terminators. Depending on the
-C panic
flag, this may either cause the program to abort or the call stack to unwind. Such
terminators have a unwind: UnwindAction
field on them. If stack unwinding occurs, then
once the current function is reached, an action will be taken based on the unwind
field.
If the action is Cleanup
, then the execution continues at the given basic block. If the
action is Continue
then no cleanup is performed, and the stack continues unwinding.
The basic block pointed to by a Cleanup
unwind action must have its cleanup
flag set.
cleanup
basic blocks have a couple restrictions:
-
All
unwind
fields in them must beUnwindAction::Terminate
orUnwindAction::Unreachable
. -
Return
terminators are not allowed in them.Terminate
andResume
terminators are. -
All other basic blocks (in the current body) that are reachable from
cleanup
basic blocks must also becleanup
. This is a part of the type system and checked statically, so it is still an error to have such an edge in the CFG even if it’s known that it won’t be taken at runtime. -
The control flow between cleanup blocks must look like an upside down tree. Roughly speaking, this means that control flow that looks like a V is allowed, while control flow that looks like a W is not. This is necessary to ensure that landing pad information can be correctly codegened on MSVC. More precisely:
Begin with the standard control flow graph
G
. ModifyG
as follows: for any two cleanup verticesu
andv
such thatu
dominatesv
, contractu
andv
into a single vertex, deleting self edges and duplicate edges in the process. Now remove all vertices fromG
that are not cleanup vertices or are not reachable. The resulting graph must be an inverted tree, that is each vertex may have at most one successor and there may be no cycles.
Variants§
Goto
Block has one successor; we continue execution there.
Fields
target: BasicBlock
SwitchInt
Switches based on the computed value.
First, evaluates the discr
operand. The type of the operand must be a signed or unsigned
integer, char, or bool, and must match the given type. Then, if the list of switch targets
contains the computed value, continues execution at the associated basic block. Otherwise,
continues execution at the “otherwise” basic block.
Target values may not appear more than once.
UnwindResume
Indicates that the landing pad is finished and that the process should continue unwinding.
Like a return, this marks the end of this invocation of the function.
Only permitted in cleanup blocks. Resume
is not permitted with -C unwind=abort
after
deaggregation runs.
UnwindTerminate(UnwindTerminateReason)
Indicates that the landing pad is finished and that the process should terminate.
Used to prevent unwinding for foreign items or with -C unwind=abort
. Only permitted in
cleanup blocks.
Return
Returns from the function.
Like function calls, the exact semantics of returns in Rust are unclear. Returning very
likely at least assigns the value currently in the return place (_0
) to the place
specified in the associated Call
terminator in the calling function, as if assigned via
dest = move _0
. It might additionally do other things, like have side-effects in the
aliasing model.
If the body is a coroutine body, this has slightly different semantics; it instead causes a
CoroutineState::Returned(_0)
to be created (as if by an Aggregate
rvalue) and assigned
to the return place.
Unreachable
Indicates a terminator that can never be reached.
Executing this terminator is UB.
Drop
The behavior of this statement differs significantly before and after drop elaboration.
After drop elaboration: Drop
terminators are a complete nop for types that have no drop
glue. For other types, Drop
terminators behave exactly like a call to
core::mem::drop_in_place
with a pointer to the given place.
Drop
before drop elaboration is a conditional execution of the drop glue. Specifically,
the Drop
will be executed if…
Needs clarification: End of that sentence. This in effect should document the exact behavior of drop elaboration. The following sounds vaguely right, but I’m not quite sure:
The drop glue is executed if, among all statements executed within this
Body
, an assignment to the place or one of its “parents” occurred more recently than a move out of it. This does not consider indirect assignments.
The replace
flag indicates whether this terminator was created as part of an assignment.
This should only be used for diagnostic purposes, and does not have any operational
meaning.
Call
Roughly speaking, evaluates the func
operand and the arguments, and starts execution of
the referred to function. The operand types must match the argument types of the function.
The return place type must match the return type. The type of the func
operand must be
callable, meaning either a function pointer, a function type, or a closure type.
Needs clarification: The exact semantics of this. Current backends rely on move
operands not aliasing the return place. It is unclear how this is justified in MIR, see
#71117.
Fields
args: Box<[Spanned<Operand<'tcx>>]>
Arguments the function is called with.
These are owned by the callee, which is free to modify them.
This allows the memory occupied by “by-value” arguments to be
reused across function calls without duplicating the contents.
The span for each arg is also included
(e.g. a
and b
in x.foo(a, b)
).
target: Option<BasicBlock>
Where to go after this call returns. If none, the call necessarily diverges.
unwind: UnwindAction
Action to be taken if the call unwinds.
call_source: CallSource
Where this call came from in HIR/THIR.
TailCall
Tail call.
Roughly speaking this is a chimera of Call
and Return
, with some caveats.
Semantically tail calls consists of two actions:
- pop of the current stack frame
- a call to the
func
, with the return address of the current caller- so that a
return
insidefunc
returns to the caller of the caller of the function that is currently being executed
- so that a
Note that in difference with Call
this is missing
destination
(because it’s always the return place)target
(because it’s always taken from the current stack frame)unwind
(because it’s always taken from the current stack frame)
Fields
Assert
Evaluates the operand, which must have type bool
. If it is not equal to expected
,
initiates a panic. Initiating a panic corresponds to a Call
terminator with some
unspecified constant as the function to call, all the operands stored in the AssertMessage
as parameters, and None
for the destination. Keep in mind that the cleanup
path is not
necessarily executed even in the case of a panic, for example in -C panic=abort
. If the
assertion does not fail, execution continues at the specified basic block.
When overflow checking is disabled and this is run-time MIR (as opposed to compile-time MIR
that is used for CTFE), the following variants of this terminator behave as goto target
:
OverflowNeg(..)
,Overflow(op, ..)
if op is add, sub, mul, shl, shr, but NOT div or rem.
Yield
Marks a suspend point.
Like Return
terminators in coroutine bodies, this computes value
and then a
CoroutineState::Yielded(value)
as if by Aggregate
rvalue. That value is then assigned to
the return place of the function calling this one, and execution continues in the calling
function. When next invoked with the same first argument, execution of this function
continues at the resume
basic block, with the second argument written to the resume_arg
place. If the coroutine is dropped before then, the drop
basic block is invoked.
Not permitted in bodies that are not coroutine bodies, or after coroutine lowering.
Needs clarification: What about the evaluation order of the resume_arg
and value
?
Fields
resume: BasicBlock
Where to resume to.
drop: Option<BasicBlock>
Cleanup to be done if the coroutine is dropped at this suspend point.
CoroutineDrop
Indicates the end of dropping a coroutine.
Semantically just a return
(from the coroutines drop glue). Only permitted in the same situations
as yield
.
Needs clarification: Is that even correct? The coroutine drop code is always confusing to me, because it’s not even really in the current body.
Needs clarification: Are there type system constraints on these terminators? Should
there be a “block type” like cleanup
blocks for them?
FalseEdge
A block where control flow only ever takes one real path, but borrowck needs to be more conservative.
At runtime this is semantically just a goto.
Disallowed after drop elaboration.
Fields
real_target: BasicBlock
The target normal control flow will take.
imaginary_target: BasicBlock
A block control flow could conceptually jump to, but won’t in practice.
FalseUnwind
A terminator for blocks that only take one path in reality, but where we reserve the right to unwind in borrowck, even if it won’t happen in practice. This can arise in infinite loops with no function calls for example.
At runtime this is semantically just a goto.
Disallowed after drop elaboration.
Fields
real_target: BasicBlock
The target normal control flow will take.
unwind: UnwindAction
The imaginary cleanup block link. This particular path will never be taken
in practice, but in order to avoid fragility we want to always
consider it in borrowck. We don’t want to accept programs which
pass borrowck only when panic=abort
or some assertions are disabled
due to release vs. debug mode builds.
InlineAsm
Block ends with an inline assembly block. This is a terminator since inline assembly is allowed to diverge.
Fields
asm_macro: InlineAsmMacro
Macro used to create this inline asm: one of asm!
or naked_asm!
template: &'tcx [InlineAsmTemplatePiece]
The template for the inline assembly, with placeholders.
operands: Box<[InlineAsmOperand<'tcx>]>
The operands for the inline assembly, as Operand
s or Place
s.
options: InlineAsmOptions
Miscellaneous options for the inline assembly.
line_spans: &'tcx [Span]
Source spans for each line of the inline assembly code. These are used to map assembler errors back to the line in the source code.
targets: Box<[BasicBlock]>
Valid targets for the inline assembly. The first element is the fallthrough destination, unless asm_macro == InlineAsmMacro::NakedAsm or InlineAsmOptions::NORETURN is set.
unwind: UnwindAction
Action to be taken if the inline assembly unwinds. This is present if and only if InlineAsmOptions::MAY_UNWIND is set.