LLVM 22.0.0git
EHPersonalities.cpp
Go to the documentation of this file.
1//===- EHPersonalities.cpp - Compute EH-related information ---------------===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8
11#include "llvm/IR/CFG.h"
12#include "llvm/IR/Constants.h"
13#include "llvm/IR/Function.h"
15#include "llvm/IR/Module.h"
16#include "llvm/Support/Debug.h"
19using namespace llvm;
20
21/// See if the given exception handling personality function is one that we
22/// understand. If so, return a description of it; otherwise return Unknown.
24 const GlobalValue *F =
25 Pers ? dyn_cast<GlobalValue>(Pers->stripPointerCasts()) : nullptr;
26 if (!F || !F->getValueType() || !F->getValueType()->isFunctionTy())
28
29 StringRef Name = F->getName();
30 if (F->getParent()->getTargetTriple().isWindowsArm64EC()) {
31 // ARM64EC function symbols are mangled by prefixing them with "#".
32 // Demangle them by skipping this prefix.
33 Name.consume_front("#");
34 }
35
37 .Case("__gnat_eh_personality", EHPersonality::GNU_Ada)
38 .Case("__gxx_personality_v0", EHPersonality::GNU_CXX)
39 .Case("__gxx_personality_seh0", EHPersonality::GNU_CXX)
40 .Case("__gxx_personality_sj0", EHPersonality::GNU_CXX_SjLj)
41 .Case("__gcc_personality_v0", EHPersonality::GNU_C)
42 .Case("__gcc_personality_seh0", EHPersonality::GNU_C)
43 .Case("__gcc_personality_sj0", EHPersonality::GNU_C_SjLj)
44 .Case("__objc_personality_v0", EHPersonality::GNU_ObjC)
45 .Case("_except_handler3", EHPersonality::MSVC_X86SEH)
46 .Case("_except_handler4", EHPersonality::MSVC_X86SEH)
47 .Case("__C_specific_handler", EHPersonality::MSVC_TableSEH)
48 .Case("__CxxFrameHandler3", EHPersonality::MSVC_CXX)
49 .Case("ProcessCLRException", EHPersonality::CoreCLR)
50 // Rust mangles its personality function, so we can't test exact equality.
51 .EndsWith("rust_eh_personality", EHPersonality::Rust)
52 .Case("__gxx_wasm_personality_v0", EHPersonality::Wasm_CXX)
53 .Case("__xlcxx_personality_v1", EHPersonality::XL_CXX)
54 .Case("__zos_cxx_personality_v2", EHPersonality::ZOS_CXX)
56}
57
59 switch (Pers) {
61 return "__gnat_eh_personality";
63 return "__gxx_personality_v0";
65 return "__gxx_personality_sj0";
67 return "__gcc_personality_v0";
69 return "__gcc_personality_sj0";
71 return "__objc_personality_v0";
73 return "_except_handler3";
75 return "__C_specific_handler";
77 return "__CxxFrameHandler3";
79 return "ProcessCLRException";
82 "Cannot get personality name of Rust personality, since it is mangled");
84 return "__gxx_wasm_personality_v0";
86 return "__xlcxx_personality_v1";
88 return "__zos_cxx_personality_v2";
90 llvm_unreachable("Unknown EHPersonality!");
91 }
92
93 llvm_unreachable("Invalid EHPersonality!");
94}
95
97 if (T.isPS5())
99 else
101}
102
104 EHPersonality Personality = classifyEHPersonality(F->getPersonalityFn());
105 // We can't simplify any invokes to nounwind functions if the personality
106 // function wants to catch asynch exceptions. The nounwind attribute only
107 // implies that the function does not throw synchronous exceptions.
108
109 // Cannot simplify CXX Personality under AsynchEH
110 const llvm::Module *M = (const llvm::Module *)F->getParent();
111 bool EHa = M->getModuleFlag("eh-asynch");
112 return !EHa && !isAsynchronousEHPersonality(Personality);
113}
114
117 BasicBlock *EntryBlock = &F.getEntryBlock();
119
120 // Build up the color map, which maps each block to its set of 'colors'.
121 // For any block B the "colors" of B are the set of funclets F (possibly
122 // including a root "funclet" representing the main function) such that
123 // F will need to directly contain B or a copy of B (where the term "directly
124 // contain" is used to distinguish from being "transitively contained" in
125 // a nested funclet).
126 //
127 // Note: Despite not being a funclet in the truest sense, a catchswitch is
128 // considered to belong to its own funclet for the purposes of coloring.
129
130 DEBUG_WITH_TYPE("win-eh-prepare-coloring",
131 dbgs() << "\nColoring funclets for " << F.getName() << "\n");
132
133 Worklist.push_back({EntryBlock, EntryBlock});
134
135 while (!Worklist.empty()) {
136 BasicBlock *Visiting;
137 BasicBlock *Color;
138 std::tie(Visiting, Color) = Worklist.pop_back_val();
139 DEBUG_WITH_TYPE("win-eh-prepare-coloring",
140 dbgs() << "Visiting " << Visiting->getName() << ", "
141 << Color->getName() << "\n");
142 BasicBlock::iterator VisitingHead = Visiting->getFirstNonPHIIt();
143 if (VisitingHead->isEHPad()) {
144 // Mark this funclet head as a member of itself.
145 Color = Visiting;
146 }
147 // Note that this is a member of the given color.
148 ColorVector &Colors = BlockColors[Visiting];
149 if (!is_contained(Colors, Color))
150 Colors.push_back(Color);
151 else
152 continue;
153
154 DEBUG_WITH_TYPE("win-eh-prepare-coloring",
155 dbgs() << " Assigned color \'" << Color->getName()
156 << "\' to block \'" << Visiting->getName()
157 << "\'.\n");
158
159 BasicBlock *SuccColor = Color;
160 Instruction *Terminator = Visiting->getTerminator();
161 if (auto *CatchRet = dyn_cast<CatchReturnInst>(Terminator)) {
162 Value *ParentPad = CatchRet->getCatchSwitchParentPad();
163 if (isa<ConstantTokenNone>(ParentPad))
164 SuccColor = EntryBlock;
165 else
166 SuccColor = cast<Instruction>(ParentPad)->getParent();
167 }
168
169 for (BasicBlock *Succ : successors(Visiting))
170 Worklist.push_back({Succ, SuccColor});
171 }
172 return BlockColors;
173}
This file contains the declarations for the subclasses of Constant, which represent the different fla...
This file provides various utilities for inspecting and working with the control flow graph in LLVM I...
Module.h This file contains the declarations for the Module class.
#define F(x, y, z)
Definition MD5.cpp:54
#define T
This file implements the StringSwitch template, which mimics a switch() statement whose cases are str...
#define DEBUG_WITH_TYPE(TYPE,...)
DEBUG_WITH_TYPE macro - This macro should be used by passes to emit debug information.
Definition Debug.h:72
LLVM Basic Block Representation.
Definition BasicBlock.h:62
const Function * getParent() const
Return the enclosing method, or null if none.
Definition BasicBlock.h:213
LLVM_ABI InstListType::const_iterator getFirstNonPHIIt() const
Returns an iterator to the first instruction in this block that is not a PHINode instruction.
InstListType::iterator iterator
Instruction iterators...
Definition BasicBlock.h:170
const Instruction * getTerminator() const LLVM_READONLY
Returns the terminator instruction if the block is well formed or null if the block is not well forme...
Definition BasicBlock.h:233
A Module instance is used to store all the information related to an LLVM module.
Definition Module.h:67
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
StringRef - Represent a constant reference to a string, i.e.
Definition StringRef.h:55
A switch()-like statement whose cases are string literals.
StringSwitch & Case(StringLiteral S, T Value)
StringSwitch & EndsWith(StringLiteral S, T Value)
void push_back(EltTy NewVal)
Triple - Helper class for working with autoconf configuration names.
Definition Triple.h:47
LLVM Value Representation.
Definition Value.h:75
LLVM_ABI const Value * stripPointerCasts() const
Strip off pointer casts, all-zero GEPs and address space casts.
Definition Value.cpp:701
LLVM_ABI StringRef getName() const
Return a constant reference to the value's name.
Definition Value.cpp:322
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
This is an optimization pass for GlobalISel generic memory operations.
LLVM_ABI StringRef getEHPersonalityName(EHPersonality Pers)
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
Definition Casting.h:643
auto successors(const MachineBasicBlock *BB)
LLVM_ABI DenseMap< BasicBlock *, ColorVector > colorEHFunclets(Function &F)
If an EH funclet personality is in use (see isFuncletEHPersonality), this will recompute which blocks...
LLVM_ABI bool canSimplifyInvokeNoUnwind(const Function *F)
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
Definition Debug.cpp:207
LLVM_ABI EHPersonality classifyEHPersonality(const Value *Pers)
See if the given exception handling personality function is one that we understand.
bool isa(const From &Val)
isa<X> - Return true if the parameter to the template is an instance of one of the template type argu...
Definition Casting.h:547
bool isAsynchronousEHPersonality(EHPersonality Pers)
Returns true if this personality function catches asynchronous exceptions.
TinyPtrVector< BasicBlock * > ColorVector
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
Definition Casting.h:559
bool is_contained(R &&Range, const E &Element)
Returns true if Element is found in Range.
Definition STLExtras.h:1897
LLVM_ABI EHPersonality getDefaultEHPersonality(const Triple &T)