xref: /aosp_15_r20/external/clang/lib/CodeGen/CGCleanup.h (revision 67e74705e28f6214e480b399dd47ea732279e315)
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