1*67e74705SXin Li //===-- CGCleanup.h - Classes for cleanups IR generation --------*- C++ -*-===//
2*67e74705SXin Li //
3*67e74705SXin Li // The LLVM Compiler Infrastructure
4*67e74705SXin Li //
5*67e74705SXin Li // This file is distributed under the University of Illinois Open Source
6*67e74705SXin Li // License. See LICENSE.TXT for details.
7*67e74705SXin Li //
8*67e74705SXin Li //===----------------------------------------------------------------------===//
9*67e74705SXin Li //
10*67e74705SXin Li // These classes support the generation of LLVM IR for cleanups.
11*67e74705SXin Li //
12*67e74705SXin Li //===----------------------------------------------------------------------===//
13*67e74705SXin Li
14*67e74705SXin Li #ifndef LLVM_CLANG_LIB_CODEGEN_CGCLEANUP_H
15*67e74705SXin Li #define LLVM_CLANG_LIB_CODEGEN_CGCLEANUP_H
16*67e74705SXin Li
17*67e74705SXin Li #include "EHScopeStack.h"
18*67e74705SXin Li
19*67e74705SXin Li #include "Address.h"
20*67e74705SXin Li #include "llvm/ADT/SmallPtrSet.h"
21*67e74705SXin Li #include "llvm/ADT/SmallVector.h"
22*67e74705SXin Li
23*67e74705SXin Li namespace llvm {
24*67e74705SXin Li class BasicBlock;
25*67e74705SXin Li class Value;
26*67e74705SXin Li class ConstantInt;
27*67e74705SXin Li class AllocaInst;
28*67e74705SXin Li }
29*67e74705SXin Li
30*67e74705SXin Li namespace clang {
31*67e74705SXin Li class FunctionDecl;
32*67e74705SXin Li namespace CodeGen {
33*67e74705SXin Li class CodeGenModule;
34*67e74705SXin Li class CodeGenFunction;
35*67e74705SXin Li
36*67e74705SXin Li /// The MS C++ ABI needs a pointer to RTTI data plus some flags to describe the
37*67e74705SXin Li /// type of a catch handler, so we use this wrapper.
38*67e74705SXin Li struct CatchTypeInfo {
39*67e74705SXin Li llvm::Constant *RTTI;
40*67e74705SXin Li unsigned Flags;
41*67e74705SXin Li };
42*67e74705SXin Li
43*67e74705SXin Li /// A protected scope for zero-cost EH handling.
44*67e74705SXin Li class EHScope {
45*67e74705SXin Li llvm::BasicBlock *CachedLandingPad;
46*67e74705SXin Li llvm::BasicBlock *CachedEHDispatchBlock;
47*67e74705SXin Li
48*67e74705SXin Li EHScopeStack::stable_iterator EnclosingEHScope;
49*67e74705SXin Li
50*67e74705SXin Li class CommonBitFields {
51*67e74705SXin Li friend class EHScope;
52*67e74705SXin Li unsigned Kind : 3;
53*67e74705SXin Li };
54*67e74705SXin Li enum { NumCommonBits = 3 };
55*67e74705SXin Li
56*67e74705SXin Li protected:
57*67e74705SXin Li class CatchBitFields {
58*67e74705SXin Li friend class EHCatchScope;
59*67e74705SXin Li unsigned : NumCommonBits;
60*67e74705SXin Li
61*67e74705SXin Li unsigned NumHandlers : 32 - NumCommonBits;
62*67e74705SXin Li };
63*67e74705SXin Li
64*67e74705SXin Li class CleanupBitFields {
65*67e74705SXin Li friend class EHCleanupScope;
66*67e74705SXin Li unsigned : NumCommonBits;
67*67e74705SXin Li
68*67e74705SXin Li /// Whether this cleanup needs to be run along normal edges.
69*67e74705SXin Li unsigned IsNormalCleanup : 1;
70*67e74705SXin Li
71*67e74705SXin Li /// Whether this cleanup needs to be run along exception edges.
72*67e74705SXin Li unsigned IsEHCleanup : 1;
73*67e74705SXin Li
74*67e74705SXin Li /// Whether this cleanup is currently active.
75*67e74705SXin Li unsigned IsActive : 1;
76*67e74705SXin Li
77*67e74705SXin Li /// Whether this cleanup is a lifetime marker
78*67e74705SXin Li unsigned IsLifetimeMarker : 1;
79*67e74705SXin Li
80*67e74705SXin Li /// Whether the normal cleanup should test the activation flag.
81*67e74705SXin Li unsigned TestFlagInNormalCleanup : 1;
82*67e74705SXin Li
83*67e74705SXin Li /// Whether the EH cleanup should test the activation flag.
84*67e74705SXin Li unsigned TestFlagInEHCleanup : 1;
85*67e74705SXin Li
86*67e74705SXin Li /// The amount of extra storage needed by the Cleanup.
87*67e74705SXin Li /// Always a multiple of the scope-stack alignment.
88*67e74705SXin Li unsigned CleanupSize : 12;
89*67e74705SXin Li };
90*67e74705SXin Li
91*67e74705SXin Li class FilterBitFields {
92*67e74705SXin Li friend class EHFilterScope;
93*67e74705SXin Li unsigned : NumCommonBits;
94*67e74705SXin Li
95*67e74705SXin Li unsigned NumFilters : 32 - NumCommonBits;
96*67e74705SXin Li };
97*67e74705SXin Li
98*67e74705SXin Li union {
99*67e74705SXin Li CommonBitFields CommonBits;
100*67e74705SXin Li CatchBitFields CatchBits;
101*67e74705SXin Li CleanupBitFields CleanupBits;
102*67e74705SXin Li FilterBitFields FilterBits;
103*67e74705SXin Li };
104*67e74705SXin Li
105*67e74705SXin Li public:
106*67e74705SXin Li enum Kind { Cleanup, Catch, Terminate, Filter, PadEnd };
107*67e74705SXin Li
EHScope(Kind kind,EHScopeStack::stable_iterator enclosingEHScope)108*67e74705SXin Li EHScope(Kind kind, EHScopeStack::stable_iterator enclosingEHScope)
109*67e74705SXin Li : CachedLandingPad(nullptr), CachedEHDispatchBlock(nullptr),
110*67e74705SXin Li EnclosingEHScope(enclosingEHScope) {
111*67e74705SXin Li CommonBits.Kind = kind;
112*67e74705SXin Li }
113*67e74705SXin Li
getKind()114*67e74705SXin Li Kind getKind() const { return static_cast<Kind>(CommonBits.Kind); }
115*67e74705SXin Li
getCachedLandingPad()116*67e74705SXin Li llvm::BasicBlock *getCachedLandingPad() const {
117*67e74705SXin Li return CachedLandingPad;
118*67e74705SXin Li }
119*67e74705SXin Li
setCachedLandingPad(llvm::BasicBlock * block)120*67e74705SXin Li void setCachedLandingPad(llvm::BasicBlock *block) {
121*67e74705SXin Li CachedLandingPad = block;
122*67e74705SXin Li }
123*67e74705SXin Li
getCachedEHDispatchBlock()124*67e74705SXin Li llvm::BasicBlock *getCachedEHDispatchBlock() const {
125*67e74705SXin Li return CachedEHDispatchBlock;
126*67e74705SXin Li }
127*67e74705SXin Li
setCachedEHDispatchBlock(llvm::BasicBlock * block)128*67e74705SXin Li void setCachedEHDispatchBlock(llvm::BasicBlock *block) {
129*67e74705SXin Li CachedEHDispatchBlock = block;
130*67e74705SXin Li }
131*67e74705SXin Li
hasEHBranches()132*67e74705SXin Li bool hasEHBranches() const {
133*67e74705SXin Li if (llvm::BasicBlock *block = getCachedEHDispatchBlock())
134*67e74705SXin Li return !block->use_empty();
135*67e74705SXin Li return false;
136*67e74705SXin Li }
137*67e74705SXin Li
getEnclosingEHScope()138*67e74705SXin Li EHScopeStack::stable_iterator getEnclosingEHScope() const {
139*67e74705SXin Li return EnclosingEHScope;
140*67e74705SXin Li }
141*67e74705SXin Li };
142*67e74705SXin Li
143*67e74705SXin Li /// A scope which attempts to handle some, possibly all, types of
144*67e74705SXin Li /// exceptions.
145*67e74705SXin Li ///
146*67e74705SXin Li /// Objective C \@finally blocks are represented using a cleanup scope
147*67e74705SXin Li /// after the catch scope.
148*67e74705SXin Li class EHCatchScope : public EHScope {
149*67e74705SXin Li // In effect, we have a flexible array member
150*67e74705SXin Li // Handler Handlers[0];
151*67e74705SXin Li // But that's only standard in C99, not C++, so we have to do
152*67e74705SXin Li // annoying pointer arithmetic instead.
153*67e74705SXin Li
154*67e74705SXin Li public:
155*67e74705SXin Li struct Handler {
156*67e74705SXin Li /// A type info value, or null (C++ null, not an LLVM null pointer)
157*67e74705SXin Li /// for a catch-all.
158*67e74705SXin Li CatchTypeInfo Type;
159*67e74705SXin Li
160*67e74705SXin Li /// The catch handler for this type.
161*67e74705SXin Li llvm::BasicBlock *Block;
162*67e74705SXin Li
isCatchAllHandler163*67e74705SXin Li bool isCatchAll() const { return Type.RTTI == nullptr; }
164*67e74705SXin Li };
165*67e74705SXin Li
166*67e74705SXin Li private:
167*67e74705SXin Li friend class EHScopeStack;
168*67e74705SXin Li
getHandlers()169*67e74705SXin Li Handler *getHandlers() {
170*67e74705SXin Li return reinterpret_cast<Handler*>(this+1);
171*67e74705SXin Li }
172*67e74705SXin Li
getHandlers()173*67e74705SXin Li const Handler *getHandlers() const {
174*67e74705SXin Li return reinterpret_cast<const Handler*>(this+1);
175*67e74705SXin Li }
176*67e74705SXin Li
177*67e74705SXin Li public:
getSizeForNumHandlers(unsigned N)178*67e74705SXin Li static size_t getSizeForNumHandlers(unsigned N) {
179*67e74705SXin Li return sizeof(EHCatchScope) + N * sizeof(Handler);
180*67e74705SXin Li }
181*67e74705SXin Li
EHCatchScope(unsigned numHandlers,EHScopeStack::stable_iterator enclosingEHScope)182*67e74705SXin Li EHCatchScope(unsigned numHandlers,
183*67e74705SXin Li EHScopeStack::stable_iterator enclosingEHScope)
184*67e74705SXin Li : EHScope(Catch, enclosingEHScope) {
185*67e74705SXin Li CatchBits.NumHandlers = numHandlers;
186*67e74705SXin Li assert(CatchBits.NumHandlers == numHandlers && "NumHandlers overflow?");
187*67e74705SXin Li }
188*67e74705SXin Li
getNumHandlers()189*67e74705SXin Li unsigned getNumHandlers() const {
190*67e74705SXin Li return CatchBits.NumHandlers;
191*67e74705SXin Li }
192*67e74705SXin Li
setCatchAllHandler(unsigned I,llvm::BasicBlock * Block)193*67e74705SXin Li void setCatchAllHandler(unsigned I, llvm::BasicBlock *Block) {
194*67e74705SXin Li setHandler(I, CatchTypeInfo{nullptr, 0}, Block);
195*67e74705SXin Li }
196*67e74705SXin Li
setHandler(unsigned I,llvm::Constant * Type,llvm::BasicBlock * Block)197*67e74705SXin Li void setHandler(unsigned I, llvm::Constant *Type, llvm::BasicBlock *Block) {
198*67e74705SXin Li assert(I < getNumHandlers());
199*67e74705SXin Li getHandlers()[I].Type = CatchTypeInfo{Type, 0};
200*67e74705SXin Li getHandlers()[I].Block = Block;
201*67e74705SXin Li }
202*67e74705SXin Li
setHandler(unsigned I,CatchTypeInfo Type,llvm::BasicBlock * Block)203*67e74705SXin Li void setHandler(unsigned I, CatchTypeInfo Type, llvm::BasicBlock *Block) {
204*67e74705SXin Li assert(I < getNumHandlers());
205*67e74705SXin Li getHandlers()[I].Type = Type;
206*67e74705SXin Li getHandlers()[I].Block = Block;
207*67e74705SXin Li }
208*67e74705SXin Li
getHandler(unsigned I)209*67e74705SXin Li const Handler &getHandler(unsigned I) const {
210*67e74705SXin Li assert(I < getNumHandlers());
211*67e74705SXin Li return getHandlers()[I];
212*67e74705SXin Li }
213*67e74705SXin Li
214*67e74705SXin Li // Clear all handler blocks.
215*67e74705SXin Li // FIXME: it's better to always call clearHandlerBlocks in DTOR and have a
216*67e74705SXin Li // 'takeHandler' or some such function which removes ownership from the
217*67e74705SXin Li // EHCatchScope object if the handlers should live longer than EHCatchScope.
clearHandlerBlocks()218*67e74705SXin Li void clearHandlerBlocks() {
219*67e74705SXin Li for (unsigned I = 0, N = getNumHandlers(); I != N; ++I)
220*67e74705SXin Li delete getHandler(I).Block;
221*67e74705SXin Li }
222*67e74705SXin Li
223*67e74705SXin Li typedef const Handler *iterator;
begin()224*67e74705SXin Li iterator begin() const { return getHandlers(); }
end()225*67e74705SXin Li iterator end() const { return getHandlers() + getNumHandlers(); }
226*67e74705SXin Li
classof(const EHScope * Scope)227*67e74705SXin Li static bool classof(const EHScope *Scope) {
228*67e74705SXin Li return Scope->getKind() == Catch;
229*67e74705SXin Li }
230*67e74705SXin Li };
231*67e74705SXin Li
232*67e74705SXin Li /// A cleanup scope which generates the cleanup blocks lazily.
233*67e74705SXin Li class LLVM_ALIGNAS(/*alignof(uint64_t)*/ 8) EHCleanupScope : public EHScope {
234*67e74705SXin Li /// The nearest normal cleanup scope enclosing this one.
235*67e74705SXin Li EHScopeStack::stable_iterator EnclosingNormal;
236*67e74705SXin Li
237*67e74705SXin Li /// The nearest EH scope enclosing this one.
238*67e74705SXin Li EHScopeStack::stable_iterator EnclosingEH;
239*67e74705SXin Li
240*67e74705SXin Li /// The dual entry/exit block along the normal edge. This is lazily
241*67e74705SXin Li /// created if needed before the cleanup is popped.
242*67e74705SXin Li llvm::BasicBlock *NormalBlock;
243*67e74705SXin Li
244*67e74705SXin Li /// An optional i1 variable indicating whether this cleanup has been
245*67e74705SXin Li /// activated yet.
246*67e74705SXin Li llvm::AllocaInst *ActiveFlag;
247*67e74705SXin Li
248*67e74705SXin Li /// Extra information required for cleanups that have resolved
249*67e74705SXin Li /// branches through them. This has to be allocated on the side
250*67e74705SXin Li /// because everything on the cleanup stack has be trivially
251*67e74705SXin Li /// movable.
252*67e74705SXin Li struct ExtInfo {
253*67e74705SXin Li /// The destinations of normal branch-afters and branch-throughs.
254*67e74705SXin Li llvm::SmallPtrSet<llvm::BasicBlock*, 4> Branches;
255*67e74705SXin Li
256*67e74705SXin Li /// Normal branch-afters.
257*67e74705SXin Li SmallVector<std::pair<llvm::BasicBlock*,llvm::ConstantInt*>, 4>
258*67e74705SXin Li BranchAfters;
259*67e74705SXin Li };
260*67e74705SXin Li mutable struct ExtInfo *ExtInfo;
261*67e74705SXin Li
262*67e74705SXin Li /// The number of fixups required by enclosing scopes (not including
263*67e74705SXin Li /// this one). If this is the top cleanup scope, all the fixups
264*67e74705SXin Li /// from this index onwards belong to this scope.
265*67e74705SXin Li unsigned FixupDepth;
266*67e74705SXin Li
getExtInfo()267*67e74705SXin Li struct ExtInfo &getExtInfo() {
268*67e74705SXin Li if (!ExtInfo) ExtInfo = new struct ExtInfo();
269*67e74705SXin Li return *ExtInfo;
270*67e74705SXin Li }
271*67e74705SXin Li
getExtInfo()272*67e74705SXin Li const struct ExtInfo &getExtInfo() const {
273*67e74705SXin Li if (!ExtInfo) ExtInfo = new struct ExtInfo();
274*67e74705SXin Li return *ExtInfo;
275*67e74705SXin Li }
276*67e74705SXin Li
277*67e74705SXin Li public:
278*67e74705SXin Li /// Gets the size required for a lazy cleanup scope with the given
279*67e74705SXin Li /// cleanup-data requirements.
getSizeForCleanupSize(size_t Size)280*67e74705SXin Li static size_t getSizeForCleanupSize(size_t Size) {
281*67e74705SXin Li return sizeof(EHCleanupScope) + Size;
282*67e74705SXin Li }
283*67e74705SXin Li
getAllocatedSize()284*67e74705SXin Li size_t getAllocatedSize() const {
285*67e74705SXin Li return sizeof(EHCleanupScope) + CleanupBits.CleanupSize;
286*67e74705SXin Li }
287*67e74705SXin Li
EHCleanupScope(bool isNormal,bool isEH,bool isActive,unsigned cleanupSize,unsigned fixupDepth,EHScopeStack::stable_iterator enclosingNormal,EHScopeStack::stable_iterator enclosingEH)288*67e74705SXin Li EHCleanupScope(bool isNormal, bool isEH, bool isActive,
289*67e74705SXin Li unsigned cleanupSize, unsigned fixupDepth,
290*67e74705SXin Li EHScopeStack::stable_iterator enclosingNormal,
291*67e74705SXin Li EHScopeStack::stable_iterator enclosingEH)
292*67e74705SXin Li : EHScope(EHScope::Cleanup, enclosingEH),
293*67e74705SXin Li EnclosingNormal(enclosingNormal), NormalBlock(nullptr),
294*67e74705SXin Li ActiveFlag(nullptr), ExtInfo(nullptr), FixupDepth(fixupDepth) {
295*67e74705SXin Li CleanupBits.IsNormalCleanup = isNormal;
296*67e74705SXin Li CleanupBits.IsEHCleanup = isEH;
297*67e74705SXin Li CleanupBits.IsActive = isActive;
298*67e74705SXin Li CleanupBits.IsLifetimeMarker = false;
299*67e74705SXin Li CleanupBits.TestFlagInNormalCleanup = false;
300*67e74705SXin Li CleanupBits.TestFlagInEHCleanup = false;
301*67e74705SXin Li CleanupBits.CleanupSize = cleanupSize;
302*67e74705SXin Li
303*67e74705SXin Li assert(CleanupBits.CleanupSize == cleanupSize && "cleanup size overflow");
304*67e74705SXin Li }
305*67e74705SXin Li
Destroy()306*67e74705SXin Li void Destroy() {
307*67e74705SXin Li delete ExtInfo;
308*67e74705SXin Li }
309*67e74705SXin Li // Objects of EHCleanupScope are not destructed. Use Destroy().
310*67e74705SXin Li ~EHCleanupScope() = delete;
311*67e74705SXin Li
isNormalCleanup()312*67e74705SXin Li bool isNormalCleanup() const { return CleanupBits.IsNormalCleanup; }
getNormalBlock()313*67e74705SXin Li llvm::BasicBlock *getNormalBlock() const { return NormalBlock; }
setNormalBlock(llvm::BasicBlock * BB)314*67e74705SXin Li void setNormalBlock(llvm::BasicBlock *BB) { NormalBlock = BB; }
315*67e74705SXin Li
isEHCleanup()316*67e74705SXin Li bool isEHCleanup() const { return CleanupBits.IsEHCleanup; }
317*67e74705SXin Li
isActive()318*67e74705SXin Li bool isActive() const { return CleanupBits.IsActive; }
setActive(bool A)319*67e74705SXin Li void setActive(bool A) { CleanupBits.IsActive = A; }
320*67e74705SXin Li
isLifetimeMarker()321*67e74705SXin Li bool isLifetimeMarker() const { return CleanupBits.IsLifetimeMarker; }
setLifetimeMarker()322*67e74705SXin Li void setLifetimeMarker() { CleanupBits.IsLifetimeMarker = true; }
323*67e74705SXin Li
hasActiveFlag()324*67e74705SXin Li bool hasActiveFlag() const { return ActiveFlag != nullptr; }
getActiveFlag()325*67e74705SXin Li Address getActiveFlag() const {
326*67e74705SXin Li return Address(ActiveFlag, CharUnits::One());
327*67e74705SXin Li }
setActiveFlag(Address Var)328*67e74705SXin Li void setActiveFlag(Address Var) {
329*67e74705SXin Li assert(Var.getAlignment().isOne());
330*67e74705SXin Li ActiveFlag = cast<llvm::AllocaInst>(Var.getPointer());
331*67e74705SXin Li }
332*67e74705SXin Li
setTestFlagInNormalCleanup()333*67e74705SXin Li void setTestFlagInNormalCleanup() {
334*67e74705SXin Li CleanupBits.TestFlagInNormalCleanup = true;
335*67e74705SXin Li }
shouldTestFlagInNormalCleanup()336*67e74705SXin Li bool shouldTestFlagInNormalCleanup() const {
337*67e74705SXin Li return CleanupBits.TestFlagInNormalCleanup;
338*67e74705SXin Li }
339*67e74705SXin Li
setTestFlagInEHCleanup()340*67e74705SXin Li void setTestFlagInEHCleanup() {
341*67e74705SXin Li CleanupBits.TestFlagInEHCleanup = true;
342*67e74705SXin Li }
shouldTestFlagInEHCleanup()343*67e74705SXin Li bool shouldTestFlagInEHCleanup() const {
344*67e74705SXin Li return CleanupBits.TestFlagInEHCleanup;
345*67e74705SXin Li }
346*67e74705SXin Li
getFixupDepth()347*67e74705SXin Li unsigned getFixupDepth() const { return FixupDepth; }
getEnclosingNormalCleanup()348*67e74705SXin Li EHScopeStack::stable_iterator getEnclosingNormalCleanup() const {
349*67e74705SXin Li return EnclosingNormal;
350*67e74705SXin Li }
351*67e74705SXin Li
getCleanupSize()352*67e74705SXin Li size_t getCleanupSize() const { return CleanupBits.CleanupSize; }
getCleanupBuffer()353*67e74705SXin Li void *getCleanupBuffer() { return this + 1; }
354*67e74705SXin Li
getCleanup()355*67e74705SXin Li EHScopeStack::Cleanup *getCleanup() {
356*67e74705SXin Li return reinterpret_cast<EHScopeStack::Cleanup*>(getCleanupBuffer());
357*67e74705SXin Li }
358*67e74705SXin Li
359*67e74705SXin Li /// True if this cleanup scope has any branch-afters or branch-throughs.
hasBranches()360*67e74705SXin Li bool hasBranches() const { return ExtInfo && !ExtInfo->Branches.empty(); }
361*67e74705SXin Li
362*67e74705SXin Li /// Add a branch-after to this cleanup scope. A branch-after is a
363*67e74705SXin Li /// branch from a point protected by this (normal) cleanup to a
364*67e74705SXin Li /// point in the normal cleanup scope immediately containing it.
365*67e74705SXin Li /// For example,
366*67e74705SXin Li /// for (;;) { A a; break; }
367*67e74705SXin Li /// contains a branch-after.
368*67e74705SXin Li ///
369*67e74705SXin Li /// Branch-afters each have their own destination out of the
370*67e74705SXin Li /// cleanup, guaranteed distinct from anything else threaded through
371*67e74705SXin Li /// it. Therefore branch-afters usually force a switch after the
372*67e74705SXin Li /// cleanup.
addBranchAfter(llvm::ConstantInt * Index,llvm::BasicBlock * Block)373*67e74705SXin Li void addBranchAfter(llvm::ConstantInt *Index,
374*67e74705SXin Li llvm::BasicBlock *Block) {
375*67e74705SXin Li struct ExtInfo &ExtInfo = getExtInfo();
376*67e74705SXin Li if (ExtInfo.Branches.insert(Block).second)
377*67e74705SXin Li ExtInfo.BranchAfters.push_back(std::make_pair(Block, Index));
378*67e74705SXin Li }
379*67e74705SXin Li
380*67e74705SXin Li /// Return the number of unique branch-afters on this scope.
getNumBranchAfters()381*67e74705SXin Li unsigned getNumBranchAfters() const {
382*67e74705SXin Li return ExtInfo ? ExtInfo->BranchAfters.size() : 0;
383*67e74705SXin Li }
384*67e74705SXin Li
getBranchAfterBlock(unsigned I)385*67e74705SXin Li llvm::BasicBlock *getBranchAfterBlock(unsigned I) const {
386*67e74705SXin Li assert(I < getNumBranchAfters());
387*67e74705SXin Li return ExtInfo->BranchAfters[I].first;
388*67e74705SXin Li }
389*67e74705SXin Li
getBranchAfterIndex(unsigned I)390*67e74705SXin Li llvm::ConstantInt *getBranchAfterIndex(unsigned I) const {
391*67e74705SXin Li assert(I < getNumBranchAfters());
392*67e74705SXin Li return ExtInfo->BranchAfters[I].second;
393*67e74705SXin Li }
394*67e74705SXin Li
395*67e74705SXin Li /// Add a branch-through to this cleanup scope. A branch-through is
396*67e74705SXin Li /// a branch from a scope protected by this (normal) cleanup to an
397*67e74705SXin Li /// enclosing scope other than the immediately-enclosing normal
398*67e74705SXin Li /// cleanup scope.
399*67e74705SXin Li ///
400*67e74705SXin Li /// In the following example, the branch through B's scope is a
401*67e74705SXin Li /// branch-through, while the branch through A's scope is a
402*67e74705SXin Li /// branch-after:
403*67e74705SXin Li /// for (;;) { A a; B b; break; }
404*67e74705SXin Li ///
405*67e74705SXin Li /// All branch-throughs have a common destination out of the
406*67e74705SXin Li /// cleanup, one possibly shared with the fall-through. Therefore
407*67e74705SXin Li /// branch-throughs usually don't force a switch after the cleanup.
408*67e74705SXin Li ///
409*67e74705SXin Li /// \return true if the branch-through was new to this scope
addBranchThrough(llvm::BasicBlock * Block)410*67e74705SXin Li bool addBranchThrough(llvm::BasicBlock *Block) {
411*67e74705SXin Li return getExtInfo().Branches.insert(Block).second;
412*67e74705SXin Li }
413*67e74705SXin Li
414*67e74705SXin Li /// Determines if this cleanup scope has any branch throughs.
hasBranchThroughs()415*67e74705SXin Li bool hasBranchThroughs() const {
416*67e74705SXin Li if (!ExtInfo) return false;
417*67e74705SXin Li return (ExtInfo->BranchAfters.size() != ExtInfo->Branches.size());
418*67e74705SXin Li }
419*67e74705SXin Li
classof(const EHScope * Scope)420*67e74705SXin Li static bool classof(const EHScope *Scope) {
421*67e74705SXin Li return (Scope->getKind() == Cleanup);
422*67e74705SXin Li }
423*67e74705SXin Li };
424*67e74705SXin Li // NOTE: there's a bunch of different data classes tacked on after an
425*67e74705SXin Li // EHCleanupScope. It is asserted (in EHScopeStack::pushCleanup*) that
426*67e74705SXin Li // they don't require greater alignment than ScopeStackAlignment. So,
427*67e74705SXin Li // EHCleanupScope ought to have alignment equal to that -- not more
428*67e74705SXin Li // (would be misaligned by the stack allocator), and not less (would
429*67e74705SXin Li // break the appended classes).
430*67e74705SXin Li static_assert(llvm::AlignOf<EHCleanupScope>::Alignment ==
431*67e74705SXin Li EHScopeStack::ScopeStackAlignment,
432*67e74705SXin Li "EHCleanupScope expected alignment");
433*67e74705SXin Li
434*67e74705SXin Li /// An exceptions scope which filters exceptions thrown through it.
435*67e74705SXin Li /// Only exceptions matching the filter types will be permitted to be
436*67e74705SXin Li /// thrown.
437*67e74705SXin Li ///
438*67e74705SXin Li /// This is used to implement C++ exception specifications.
439*67e74705SXin Li class EHFilterScope : public EHScope {
440*67e74705SXin Li // Essentially ends in a flexible array member:
441*67e74705SXin Li // llvm::Value *FilterTypes[0];
442*67e74705SXin Li
getFilters()443*67e74705SXin Li llvm::Value **getFilters() {
444*67e74705SXin Li return reinterpret_cast<llvm::Value**>(this+1);
445*67e74705SXin Li }
446*67e74705SXin Li
getFilters()447*67e74705SXin Li llvm::Value * const *getFilters() const {
448*67e74705SXin Li return reinterpret_cast<llvm::Value* const *>(this+1);
449*67e74705SXin Li }
450*67e74705SXin Li
451*67e74705SXin Li public:
EHFilterScope(unsigned numFilters)452*67e74705SXin Li EHFilterScope(unsigned numFilters)
453*67e74705SXin Li : EHScope(Filter, EHScopeStack::stable_end()) {
454*67e74705SXin Li FilterBits.NumFilters = numFilters;
455*67e74705SXin Li assert(FilterBits.NumFilters == numFilters && "NumFilters overflow");
456*67e74705SXin Li }
457*67e74705SXin Li
getSizeForNumFilters(unsigned numFilters)458*67e74705SXin Li static size_t getSizeForNumFilters(unsigned numFilters) {
459*67e74705SXin Li return sizeof(EHFilterScope) + numFilters * sizeof(llvm::Value*);
460*67e74705SXin Li }
461*67e74705SXin Li
getNumFilters()462*67e74705SXin Li unsigned getNumFilters() const { return FilterBits.NumFilters; }
463*67e74705SXin Li
setFilter(unsigned i,llvm::Value * filterValue)464*67e74705SXin Li void setFilter(unsigned i, llvm::Value *filterValue) {
465*67e74705SXin Li assert(i < getNumFilters());
466*67e74705SXin Li getFilters()[i] = filterValue;
467*67e74705SXin Li }
468*67e74705SXin Li
getFilter(unsigned i)469*67e74705SXin Li llvm::Value *getFilter(unsigned i) const {
470*67e74705SXin Li assert(i < getNumFilters());
471*67e74705SXin Li return getFilters()[i];
472*67e74705SXin Li }
473*67e74705SXin Li
classof(const EHScope * scope)474*67e74705SXin Li static bool classof(const EHScope *scope) {
475*67e74705SXin Li return scope->getKind() == Filter;
476*67e74705SXin Li }
477*67e74705SXin Li };
478*67e74705SXin Li
479*67e74705SXin Li /// An exceptions scope which calls std::terminate if any exception
480*67e74705SXin Li /// reaches it.
481*67e74705SXin Li class EHTerminateScope : public EHScope {
482*67e74705SXin Li public:
EHTerminateScope(EHScopeStack::stable_iterator enclosingEHScope)483*67e74705SXin Li EHTerminateScope(EHScopeStack::stable_iterator enclosingEHScope)
484*67e74705SXin Li : EHScope(Terminate, enclosingEHScope) {}
getSize()485*67e74705SXin Li static size_t getSize() { return sizeof(EHTerminateScope); }
486*67e74705SXin Li
classof(const EHScope * scope)487*67e74705SXin Li static bool classof(const EHScope *scope) {
488*67e74705SXin Li return scope->getKind() == Terminate;
489*67e74705SXin Li }
490*67e74705SXin Li };
491*67e74705SXin Li
492*67e74705SXin Li class EHPadEndScope : public EHScope {
493*67e74705SXin Li public:
EHPadEndScope(EHScopeStack::stable_iterator enclosingEHScope)494*67e74705SXin Li EHPadEndScope(EHScopeStack::stable_iterator enclosingEHScope)
495*67e74705SXin Li : EHScope(PadEnd, enclosingEHScope) {}
getSize()496*67e74705SXin Li static size_t getSize() { return sizeof(EHPadEndScope); }
497*67e74705SXin Li
classof(const EHScope * scope)498*67e74705SXin Li static bool classof(const EHScope *scope) {
499*67e74705SXin Li return scope->getKind() == PadEnd;
500*67e74705SXin Li }
501*67e74705SXin Li };
502*67e74705SXin Li
503*67e74705SXin Li /// A non-stable pointer into the scope stack.
504*67e74705SXin Li class EHScopeStack::iterator {
505*67e74705SXin Li char *Ptr;
506*67e74705SXin Li
507*67e74705SXin Li friend class EHScopeStack;
iterator(char * Ptr)508*67e74705SXin Li explicit iterator(char *Ptr) : Ptr(Ptr) {}
509*67e74705SXin Li
510*67e74705SXin Li public:
iterator()511*67e74705SXin Li iterator() : Ptr(nullptr) {}
512*67e74705SXin Li
get()513*67e74705SXin Li EHScope *get() const {
514*67e74705SXin Li return reinterpret_cast<EHScope*>(Ptr);
515*67e74705SXin Li }
516*67e74705SXin Li
517*67e74705SXin Li EHScope *operator->() const { return get(); }
518*67e74705SXin Li EHScope &operator*() const { return *get(); }
519*67e74705SXin Li
520*67e74705SXin Li iterator &operator++() {
521*67e74705SXin Li size_t Size;
522*67e74705SXin Li switch (get()->getKind()) {
523*67e74705SXin Li case EHScope::Catch:
524*67e74705SXin Li Size = EHCatchScope::getSizeForNumHandlers(
525*67e74705SXin Li static_cast<const EHCatchScope *>(get())->getNumHandlers());
526*67e74705SXin Li break;
527*67e74705SXin Li
528*67e74705SXin Li case EHScope::Filter:
529*67e74705SXin Li Size = EHFilterScope::getSizeForNumFilters(
530*67e74705SXin Li static_cast<const EHFilterScope *>(get())->getNumFilters());
531*67e74705SXin Li break;
532*67e74705SXin Li
533*67e74705SXin Li case EHScope::Cleanup:
534*67e74705SXin Li Size = static_cast<const EHCleanupScope *>(get())->getAllocatedSize();
535*67e74705SXin Li break;
536*67e74705SXin Li
537*67e74705SXin Li case EHScope::Terminate:
538*67e74705SXin Li Size = EHTerminateScope::getSize();
539*67e74705SXin Li break;
540*67e74705SXin Li
541*67e74705SXin Li case EHScope::PadEnd:
542*67e74705SXin Li Size = EHPadEndScope::getSize();
543*67e74705SXin Li break;
544*67e74705SXin Li }
545*67e74705SXin Li Ptr += llvm::alignTo(Size, ScopeStackAlignment);
546*67e74705SXin Li return *this;
547*67e74705SXin Li }
548*67e74705SXin Li
next()549*67e74705SXin Li iterator next() {
550*67e74705SXin Li iterator copy = *this;
551*67e74705SXin Li ++copy;
552*67e74705SXin Li return copy;
553*67e74705SXin Li }
554*67e74705SXin Li
555*67e74705SXin Li iterator operator++(int) {
556*67e74705SXin Li iterator copy = *this;
557*67e74705SXin Li operator++();
558*67e74705SXin Li return copy;
559*67e74705SXin Li }
560*67e74705SXin Li
encloses(iterator other)561*67e74705SXin Li bool encloses(iterator other) const { return Ptr >= other.Ptr; }
strictlyEncloses(iterator other)562*67e74705SXin Li bool strictlyEncloses(iterator other) const { return Ptr > other.Ptr; }
563*67e74705SXin Li
564*67e74705SXin Li bool operator==(iterator other) const { return Ptr == other.Ptr; }
565*67e74705SXin Li bool operator!=(iterator other) const { return Ptr != other.Ptr; }
566*67e74705SXin Li };
567*67e74705SXin Li
begin()568*67e74705SXin Li inline EHScopeStack::iterator EHScopeStack::begin() const {
569*67e74705SXin Li return iterator(StartOfData);
570*67e74705SXin Li }
571*67e74705SXin Li
end()572*67e74705SXin Li inline EHScopeStack::iterator EHScopeStack::end() const {
573*67e74705SXin Li return iterator(EndOfBuffer);
574*67e74705SXin Li }
575*67e74705SXin Li
popCatch()576*67e74705SXin Li inline void EHScopeStack::popCatch() {
577*67e74705SXin Li assert(!empty() && "popping exception stack when not empty");
578*67e74705SXin Li
579*67e74705SXin Li EHCatchScope &scope = cast<EHCatchScope>(*begin());
580*67e74705SXin Li InnermostEHScope = scope.getEnclosingEHScope();
581*67e74705SXin Li deallocate(EHCatchScope::getSizeForNumHandlers(scope.getNumHandlers()));
582*67e74705SXin Li }
583*67e74705SXin Li
popTerminate()584*67e74705SXin Li inline void EHScopeStack::popTerminate() {
585*67e74705SXin Li assert(!empty() && "popping exception stack when not empty");
586*67e74705SXin Li
587*67e74705SXin Li EHTerminateScope &scope = cast<EHTerminateScope>(*begin());
588*67e74705SXin Li InnermostEHScope = scope.getEnclosingEHScope();
589*67e74705SXin Li deallocate(EHTerminateScope::getSize());
590*67e74705SXin Li }
591*67e74705SXin Li
find(stable_iterator sp)592*67e74705SXin Li inline EHScopeStack::iterator EHScopeStack::find(stable_iterator sp) const {
593*67e74705SXin Li assert(sp.isValid() && "finding invalid savepoint");
594*67e74705SXin Li assert(sp.Size <= stable_begin().Size && "finding savepoint after pop");
595*67e74705SXin Li return iterator(EndOfBuffer - sp.Size);
596*67e74705SXin Li }
597*67e74705SXin Li
598*67e74705SXin Li inline EHScopeStack::stable_iterator
stabilize(iterator ir)599*67e74705SXin Li EHScopeStack::stabilize(iterator ir) const {
600*67e74705SXin Li assert(StartOfData <= ir.Ptr && ir.Ptr <= EndOfBuffer);
601*67e74705SXin Li return stable_iterator(EndOfBuffer - ir.Ptr);
602*67e74705SXin Li }
603*67e74705SXin Li
604*67e74705SXin Li /// The exceptions personality for a function.
605*67e74705SXin Li struct EHPersonality {
606*67e74705SXin Li const char *PersonalityFn;
607*67e74705SXin Li
608*67e74705SXin Li // If this is non-null, this personality requires a non-standard
609*67e74705SXin Li // function for rethrowing an exception after a catchall cleanup.
610*67e74705SXin Li // This function must have prototype void(void*).
611*67e74705SXin Li const char *CatchallRethrowFn;
612*67e74705SXin Li
613*67e74705SXin Li static const EHPersonality &get(CodeGenModule &CGM, const FunctionDecl *FD);
614*67e74705SXin Li static const EHPersonality &get(CodeGenFunction &CGF);
615*67e74705SXin Li
616*67e74705SXin Li static const EHPersonality GNU_C;
617*67e74705SXin Li static const EHPersonality GNU_C_SJLJ;
618*67e74705SXin Li static const EHPersonality GNU_C_SEH;
619*67e74705SXin Li static const EHPersonality GNU_ObjC;
620*67e74705SXin Li static const EHPersonality GNUstep_ObjC;
621*67e74705SXin Li static const EHPersonality GNU_ObjCXX;
622*67e74705SXin Li static const EHPersonality NeXT_ObjC;
623*67e74705SXin Li static const EHPersonality GNU_CPlusPlus;
624*67e74705SXin Li static const EHPersonality GNU_CPlusPlus_SJLJ;
625*67e74705SXin Li static const EHPersonality GNU_CPlusPlus_SEH;
626*67e74705SXin Li static const EHPersonality MSVC_except_handler;
627*67e74705SXin Li static const EHPersonality MSVC_C_specific_handler;
628*67e74705SXin Li static const EHPersonality MSVC_CxxFrameHandler3;
629*67e74705SXin Li
630*67e74705SXin Li /// Does this personality use landingpads or the family of pad instructions
631*67e74705SXin Li /// designed to form funclets?
usesFuncletPadsEHPersonality632*67e74705SXin Li bool usesFuncletPads() const { return isMSVCPersonality(); }
633*67e74705SXin Li
isMSVCPersonalityEHPersonality634*67e74705SXin Li bool isMSVCPersonality() const {
635*67e74705SXin Li return this == &MSVC_except_handler || this == &MSVC_C_specific_handler ||
636*67e74705SXin Li this == &MSVC_CxxFrameHandler3;
637*67e74705SXin Li }
638*67e74705SXin Li
isMSVCXXPersonalityEHPersonality639*67e74705SXin Li bool isMSVCXXPersonality() const { return this == &MSVC_CxxFrameHandler3; }
640*67e74705SXin Li };
641*67e74705SXin Li }
642*67e74705SXin Li }
643*67e74705SXin Li
644*67e74705SXin Li #endif
645