| Mike Stump | 376e3c0 | 2009-03-04 15:35:22 +0000 | [diff] [blame] | 1 | //===-- CGBlocks.h - state for LLVM CodeGen for blocks ----------*- C++ -*-===// | 
|  | 2 | // | 
|  | 3 | //                     The LLVM Compiler Infrastructure | 
|  | 4 | // | 
|  | 5 | // This file is distributed under the University of Illinois Open Source | 
|  | 6 | // License. See LICENSE.TXT for details. | 
|  | 7 | // | 
|  | 8 | //===----------------------------------------------------------------------===// | 
|  | 9 | // | 
|  | 10 | // This is the internal state used for llvm translation for block literals. | 
|  | 11 | // | 
|  | 12 | //===----------------------------------------------------------------------===// | 
|  | 13 |  | 
|  | 14 | #ifndef CLANG_CODEGEN_CGBLOCKS_H | 
|  | 15 | #define CLANG_CODEGEN_CGBLOCKS_H | 
|  | 16 |  | 
| Mike Stump | 9543567 | 2009-03-04 18:17:45 +0000 | [diff] [blame] | 17 | #include "CodeGenTypes.h" | 
|  | 18 | #include "clang/AST/Type.h" | 
| Owen Anderson | 170229f | 2009-07-14 23:10:40 +0000 | [diff] [blame] | 19 | #include "llvm/Module.h" | 
| Mike Stump | 9543567 | 2009-03-04 18:17:45 +0000 | [diff] [blame] | 20 | #include "llvm/ADT/DenseMap.h" | 
|  | 21 | #include "llvm/ADT/SmallVector.h" | 
|  | 22 | #include "clang/Basic/TargetInfo.h" | 
|  | 23 | #include "clang/AST/Expr.h" | 
|  | 24 | #include "clang/AST/ExprCXX.h" | 
|  | 25 | #include "clang/AST/ExprObjC.h" | 
|  | 26 |  | 
|  | 27 | #include <vector> | 
|  | 28 | #include <map> | 
|  | 29 |  | 
|  | 30 | #include "CGBuilder.h" | 
|  | 31 | #include "CGCall.h" | 
|  | 32 | #include "CGValue.h" | 
|  | 33 |  | 
|  | 34 | namespace llvm { | 
|  | 35 | class Module; | 
|  | 36 | class Constant; | 
|  | 37 | class Function; | 
|  | 38 | class GlobalValue; | 
|  | 39 | class TargetData; | 
|  | 40 | class FunctionType; | 
|  | 41 | class Value; | 
| Benjamin Kramer | 9cd050a | 2009-08-11 17:46:57 +0000 | [diff] [blame] | 42 | class LLVMContext; | 
| Mike Stump | 9543567 | 2009-03-04 18:17:45 +0000 | [diff] [blame] | 43 | } | 
|  | 44 |  | 
| Mike Stump | 376e3c0 | 2009-03-04 15:35:22 +0000 | [diff] [blame] | 45 | namespace clang { | 
|  | 46 |  | 
|  | 47 | namespace CodeGen { | 
| Mike Stump | 6c39666 | 2009-03-04 18:47:42 +0000 | [diff] [blame] | 48 | class CodeGenModule; | 
| Mike Stump | 376e3c0 | 2009-03-04 15:35:22 +0000 | [diff] [blame] | 49 |  | 
|  | 50 | class BlockBase { | 
|  | 51 | public: | 
|  | 52 | enum { | 
|  | 53 | BLOCK_NEEDS_FREE =        (1 << 24), | 
|  | 54 | BLOCK_HAS_COPY_DISPOSE =  (1 << 25), | 
|  | 55 | BLOCK_HAS_CXX_OBJ =       (1 << 26), | 
|  | 56 | BLOCK_IS_GC =             (1 << 27), | 
|  | 57 | BLOCK_IS_GLOBAL =         (1 << 28), | 
|  | 58 | BLOCK_HAS_DESCRIPTOR =    (1 << 29) | 
|  | 59 | }; | 
|  | 60 | }; | 
|  | 61 |  | 
|  | 62 | class BlockModule : public BlockBase { | 
| Mike Stump | 9543567 | 2009-03-04 18:17:45 +0000 | [diff] [blame] | 63 | ASTContext &Context; | 
|  | 64 | llvm::Module &TheModule; | 
| Mike Stump | 6c39666 | 2009-03-04 18:47:42 +0000 | [diff] [blame] | 65 | const llvm::TargetData &TheTargetData; | 
| Mike Stump | 9543567 | 2009-03-04 18:17:45 +0000 | [diff] [blame] | 66 | CodeGenTypes &Types; | 
| Mike Stump | 6c39666 | 2009-03-04 18:47:42 +0000 | [diff] [blame] | 67 | CodeGenModule &CGM; | 
| Owen Anderson | 170229f | 2009-07-14 23:10:40 +0000 | [diff] [blame] | 68 | llvm::LLVMContext &VMContext; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 69 |  | 
| Mike Stump | 9543567 | 2009-03-04 18:17:45 +0000 | [diff] [blame] | 70 | ASTContext &getContext() const { return Context; } | 
|  | 71 | llvm::Module &getModule() const { return TheModule; } | 
|  | 72 | CodeGenTypes &getTypes() { return Types; } | 
| Mike Stump | 6c39666 | 2009-03-04 18:47:42 +0000 | [diff] [blame] | 73 | const llvm::TargetData &getTargetData() const { return TheTargetData; } | 
| Mike Stump | 9543567 | 2009-03-04 18:17:45 +0000 | [diff] [blame] | 74 | public: | 
|  | 75 | llvm::Constant *getNSConcreteGlobalBlock(); | 
|  | 76 | llvm::Constant *getNSConcreteStackBlock(); | 
|  | 77 | int getGlobalUniqueCount() { return ++Block.GlobalUniqueCount; } | 
|  | 78 | const llvm::Type *getBlockDescriptorType(); | 
|  | 79 |  | 
|  | 80 | const llvm::Type *getGenericBlockLiteralType(); | 
|  | 81 | const llvm::Type *getGenericExtendedBlockLiteralType(); | 
|  | 82 |  | 
| Mike Stump | 6c39666 | 2009-03-04 18:47:42 +0000 | [diff] [blame] | 83 | llvm::Constant *GetAddrOfGlobalBlock(const BlockExpr *BE, const char *); | 
|  | 84 |  | 
| Mike Stump | 9543567 | 2009-03-04 18:17:45 +0000 | [diff] [blame] | 85 | /// NSConcreteGlobalBlock - Cached reference to the class pointer for global | 
|  | 86 | /// blocks. | 
|  | 87 | llvm::Constant *NSConcreteGlobalBlock; | 
|  | 88 |  | 
|  | 89 | /// NSConcreteStackBlock - Cached reference to the class poinnter for stack | 
|  | 90 | /// blocks. | 
|  | 91 | llvm::Constant *NSConcreteStackBlock; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 92 |  | 
| Mike Stump | 9543567 | 2009-03-04 18:17:45 +0000 | [diff] [blame] | 93 | const llvm::Type *BlockDescriptorType; | 
|  | 94 | const llvm::Type *GenericBlockLiteralType; | 
|  | 95 | const llvm::Type *GenericExtendedBlockLiteralType; | 
|  | 96 | struct { | 
|  | 97 | int GlobalUniqueCount; | 
|  | 98 | } Block; | 
|  | 99 |  | 
| Mike Stump | f89230d | 2009-03-06 06:12:24 +0000 | [diff] [blame] | 100 | llvm::Value *BlockObjectAssign; | 
| Mike Stump | 626aecc | 2009-03-05 01:23:13 +0000 | [diff] [blame] | 101 | llvm::Value *BlockObjectDispose; | 
|  | 102 | const llvm::Type *PtrToInt8Ty; | 
|  | 103 |  | 
| Mike Stump | cbc2bca | 2009-06-05 23:26:36 +0000 | [diff] [blame] | 104 | std::map<uint64_t, llvm::Constant *> AssignCache; | 
|  | 105 | std::map<uint64_t, llvm::Constant *> DestroyCache; | 
|  | 106 |  | 
| Mike Stump | 6c39666 | 2009-03-04 18:47:42 +0000 | [diff] [blame] | 107 | BlockModule(ASTContext &C, llvm::Module &M, const llvm::TargetData &TD, | 
|  | 108 | CodeGenTypes &T, CodeGenModule &CodeGen) | 
|  | 109 | : Context(C), TheModule(M), TheTargetData(TD), Types(T), | 
| Owen Anderson | 170229f | 2009-07-14 23:10:40 +0000 | [diff] [blame] | 110 | CGM(CodeGen), VMContext(M.getContext()), | 
| Mike Stump | 6c39666 | 2009-03-04 18:47:42 +0000 | [diff] [blame] | 111 | NSConcreteGlobalBlock(0), NSConcreteStackBlock(0), BlockDescriptorType(0), | 
| Mike Stump | aeb0ffd | 2009-03-07 02:35:30 +0000 | [diff] [blame] | 112 | GenericBlockLiteralType(0), GenericExtendedBlockLiteralType(0), | 
|  | 113 | BlockObjectAssign(0), BlockObjectDispose(0) { | 
| Mike Stump | 9543567 | 2009-03-04 18:17:45 +0000 | [diff] [blame] | 114 | Block.GlobalUniqueCount = 0; | 
| Benjamin Kramer | abd5b90 | 2009-10-13 10:07:13 +0000 | [diff] [blame] | 115 | PtrToInt8Ty = llvm::Type::getInt8PtrTy(M.getContext()); | 
| Mike Stump | 9543567 | 2009-03-04 18:17:45 +0000 | [diff] [blame] | 116 | } | 
| Mike Stump | 2114d7c | 2009-09-22 02:12:52 +0000 | [diff] [blame] | 117 |  | 
| Mike Stump | 41eb02d | 2009-10-21 18:23:01 +0000 | [diff] [blame] | 118 | bool BlockRequiresCopying(QualType Ty) | 
|  | 119 | { return getContext().BlockRequiresCopying(Ty); } | 
| Mike Stump | 376e3c0 | 2009-03-04 15:35:22 +0000 | [diff] [blame] | 120 | }; | 
|  | 121 |  | 
|  | 122 | class BlockFunction : public BlockBase { | 
| Mike Stump | 626aecc | 2009-03-05 01:23:13 +0000 | [diff] [blame] | 123 | CodeGenModule &CGM; | 
| Mike Stump | 0c74327 | 2009-03-06 01:33:24 +0000 | [diff] [blame] | 124 | CodeGenFunction &CGF; | 
| Mike Stump | 4446dcf | 2009-03-05 08:32:30 +0000 | [diff] [blame] | 125 | ASTContext &getContext() const; | 
| Mike Stump | 626aecc | 2009-03-05 01:23:13 +0000 | [diff] [blame] | 126 |  | 
| Owen Anderson | 170229f | 2009-07-14 23:10:40 +0000 | [diff] [blame] | 127 | protected: | 
|  | 128 | llvm::LLVMContext &VMContext; | 
|  | 129 |  | 
| Mike Stump | 376e3c0 | 2009-03-04 15:35:22 +0000 | [diff] [blame] | 130 | public: | 
| Mike Stump | 626aecc | 2009-03-05 01:23:13 +0000 | [diff] [blame] | 131 | const llvm::Type *PtrToInt8Ty; | 
| Mike Stump | aeb0ffd | 2009-03-07 02:35:30 +0000 | [diff] [blame] | 132 | struct HelperInfo { | 
|  | 133 | int index; | 
|  | 134 | int flag; | 
|  | 135 | bool RequiresCopying; | 
|  | 136 | }; | 
|  | 137 |  | 
| Mike Stump | 626aecc | 2009-03-05 01:23:13 +0000 | [diff] [blame] | 138 | enum { | 
| Mike Stump | 376e3c0 | 2009-03-04 15:35:22 +0000 | [diff] [blame] | 139 | BLOCK_FIELD_IS_OBJECT   =  3,  /* id, NSObject, __attribute__((NSObject)), | 
|  | 140 | block, ... */ | 
|  | 141 | BLOCK_FIELD_IS_BLOCK    =  7,  /* a block variable */ | 
|  | 142 | BLOCK_FIELD_IS_BYREF    =  8,  /* the on stack structure holding the __block | 
|  | 143 | variable */ | 
|  | 144 | BLOCK_FIELD_IS_WEAK     = 16,  /* declared __weak, only used in byref copy | 
|  | 145 | helpers */ | 
| Mike Stump | cbc2bca | 2009-06-05 23:26:36 +0000 | [diff] [blame] | 146 | BLOCK_BYREF_CALLER      = 128,  /* called from __block (byref) copy/dispose | 
| Mike Stump | 376e3c0 | 2009-03-04 15:35:22 +0000 | [diff] [blame] | 147 | support routines */ | 
| Mike Stump | cbc2bca | 2009-06-05 23:26:36 +0000 | [diff] [blame] | 148 | BLOCK_BYREF_CURRENT_MAX = 256 | 
| Mike Stump | 376e3c0 | 2009-03-04 15:35:22 +0000 | [diff] [blame] | 149 | }; | 
| Mike Stump | 06acea8a | 2009-03-04 18:57:26 +0000 | [diff] [blame] | 150 |  | 
| Mike Stump | 8473a12 | 2009-03-04 19:03:44 +0000 | [diff] [blame] | 151 | /// BlockInfo - Information to generate a block literal. | 
|  | 152 | struct BlockInfo { | 
|  | 153 | /// BlockLiteralTy - The type of the block literal. | 
|  | 154 | const llvm::Type *BlockLiteralTy; | 
|  | 155 |  | 
| Daniel Dunbar | 0a2171c | 2009-05-14 16:42:16 +0000 | [diff] [blame] | 156 | /// Name - the name of the function this block was created for, if any. | 
| Mike Stump | 8473a12 | 2009-03-04 19:03:44 +0000 | [diff] [blame] | 157 | const char *Name; | 
|  | 158 |  | 
|  | 159 | /// ByCopyDeclRefs - Variables from parent scopes that have been imported | 
|  | 160 | /// into this block. | 
| Mike Stump | 7fe9cc1 | 2009-10-21 03:49:08 +0000 | [diff] [blame] | 161 | llvm::SmallVector<const BlockDeclRefExpr *, 8> DeclRefs; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 162 |  | 
| Mike Stump | 8473a12 | 2009-03-04 19:03:44 +0000 | [diff] [blame] | 163 | BlockInfo(const llvm::Type *blt, const char *n) | 
| Daniel Dunbar | 0a2171c | 2009-05-14 16:42:16 +0000 | [diff] [blame] | 164 | : BlockLiteralTy(blt), Name(n) { | 
|  | 165 | // Skip asm prefix, if any. | 
|  | 166 | if (Name && Name[0] == '\01') | 
|  | 167 | ++Name; | 
|  | 168 | } | 
| Mike Stump | 8473a12 | 2009-03-04 19:03:44 +0000 | [diff] [blame] | 169 | }; | 
|  | 170 |  | 
| Mike Stump | 06acea8a | 2009-03-04 18:57:26 +0000 | [diff] [blame] | 171 | CGBuilderTy &Builder; | 
|  | 172 |  | 
| Mike Stump | aeb0ffd | 2009-03-07 02:35:30 +0000 | [diff] [blame] | 173 | BlockFunction(CodeGenModule &cgm, CodeGenFunction &cgf, CGBuilderTy &B); | 
|  | 174 |  | 
|  | 175 | /// BlockOffset - The offset in bytes for the next allocation of an | 
|  | 176 | /// imported block variable. | 
|  | 177 | uint64_t BlockOffset; | 
|  | 178 | /// BlockAlign - Maximal alignment needed for the Block expressed in bytes. | 
|  | 179 | uint64_t BlockAlign; | 
|  | 180 |  | 
|  | 181 | /// getBlockOffset - Allocate an offset for the ValueDecl from a | 
|  | 182 | /// BlockDeclRefExpr in a block literal (BlockExpr). | 
|  | 183 | uint64_t getBlockOffset(const BlockDeclRefExpr *E); | 
|  | 184 |  | 
|  | 185 | /// BlockHasCopyDispose - True iff the block uses copy/dispose. | 
|  | 186 | bool BlockHasCopyDispose; | 
|  | 187 |  | 
|  | 188 | /// BlockDeclRefDecls - Decls from BlockDeclRefExprs in apperance order | 
|  | 189 | /// in a block literal.  Decls without names are used for padding. | 
|  | 190 | llvm::SmallVector<const Expr *, 8> BlockDeclRefDecls; | 
|  | 191 |  | 
|  | 192 | /// BlockDecls - Offsets for all Decls in BlockDeclRefExprs. | 
|  | 193 | std::map<const Decl*, uint64_t> BlockDecls; | 
| Mike Stump | 06acea8a | 2009-03-04 18:57:26 +0000 | [diff] [blame] | 194 |  | 
| Mike Stump | 8473a12 | 2009-03-04 19:03:44 +0000 | [diff] [blame] | 195 | ImplicitParamDecl *BlockStructDecl; | 
|  | 196 | ImplicitParamDecl *getBlockStructDecl() { return BlockStructDecl; } | 
|  | 197 |  | 
| Mike Stump | aeb0ffd | 2009-03-07 02:35:30 +0000 | [diff] [blame] | 198 | llvm::Constant *GenerateCopyHelperFunction(bool, const llvm::StructType *, | 
| Mike Stump | 6764593 | 2009-04-10 18:52:28 +0000 | [diff] [blame] | 199 | std::vector<HelperInfo> *); | 
| Mike Stump | aeb0ffd | 2009-03-07 02:35:30 +0000 | [diff] [blame] | 200 | llvm::Constant *GenerateDestroyHelperFunction(bool, const llvm::StructType *, | 
| Mike Stump | 6764593 | 2009-04-10 18:52:28 +0000 | [diff] [blame] | 201 | std::vector<HelperInfo> *); | 
| Mike Stump | 0c74327 | 2009-03-06 01:33:24 +0000 | [diff] [blame] | 202 |  | 
| Mike Stump | aeb0ffd | 2009-03-07 02:35:30 +0000 | [diff] [blame] | 203 | llvm::Constant *BuildCopyHelper(const llvm::StructType *, | 
| Mike Stump | 6764593 | 2009-04-10 18:52:28 +0000 | [diff] [blame] | 204 | std::vector<HelperInfo> *); | 
| Mike Stump | aeb0ffd | 2009-03-07 02:35:30 +0000 | [diff] [blame] | 205 | llvm::Constant *BuildDestroyHelper(const llvm::StructType *, | 
| Mike Stump | 6764593 | 2009-04-10 18:52:28 +0000 | [diff] [blame] | 206 | std::vector<HelperInfo> *); | 
| Mike Stump | ee2a5ee | 2009-03-06 02:29:21 +0000 | [diff] [blame] | 207 |  | 
| Mike Stump | f89230d | 2009-03-06 06:12:24 +0000 | [diff] [blame] | 208 | llvm::Constant *GeneratebyrefCopyHelperFunction(const llvm::Type *, int flag); | 
| Mike Stump | fbe25dd | 2009-03-06 04:53:30 +0000 | [diff] [blame] | 209 | llvm::Constant *GeneratebyrefDestroyHelperFunction(const llvm::Type *T, int); | 
| Mike Stump | ee2a5ee | 2009-03-06 02:29:21 +0000 | [diff] [blame] | 210 |  | 
| Mike Stump | cbc2bca | 2009-06-05 23:26:36 +0000 | [diff] [blame] | 211 | llvm::Constant *BuildbyrefCopyHelper(const llvm::Type *T, int flag, | 
|  | 212 | unsigned Align); | 
|  | 213 | llvm::Constant *BuildbyrefDestroyHelper(const llvm::Type *T, int flag, | 
|  | 214 | unsigned Align); | 
| Mike Stump | 06acea8a | 2009-03-04 18:57:26 +0000 | [diff] [blame] | 215 |  | 
| Mike Stump | f89230d | 2009-03-06 06:12:24 +0000 | [diff] [blame] | 216 | llvm::Value *getBlockObjectAssign(); | 
| Mike Stump | 626aecc | 2009-03-05 01:23:13 +0000 | [diff] [blame] | 217 | llvm::Value *getBlockObjectDispose(); | 
| Mike Stump | fbe25dd | 2009-03-06 04:53:30 +0000 | [diff] [blame] | 218 | void BuildBlockRelease(llvm::Value *DeclPtr, int flag = BLOCK_FIELD_IS_BYREF); | 
| Mike Stump | 4446dcf | 2009-03-05 08:32:30 +0000 | [diff] [blame] | 219 |  | 
| Mike Stump | 41eb02d | 2009-10-21 18:23:01 +0000 | [diff] [blame] | 220 | bool BlockRequiresCopying(QualType Ty) | 
|  | 221 | { return getContext().BlockRequiresCopying(Ty); } | 
| Mike Stump | 376e3c0 | 2009-03-04 15:35:22 +0000 | [diff] [blame] | 222 | }; | 
|  | 223 |  | 
|  | 224 | }  // end namespace CodeGen | 
|  | 225 | }  // end namespace clang | 
|  | 226 |  | 
|  | 227 | #endif |