| Chris Lattner | ddc135e | 2006-11-10 06:34:16 +0000 | [diff] [blame] | 1 | //===--- ASTContext.cpp - Context to hold long-lived AST nodes ------------===// | 
|  | 2 | // | 
|  | 3 | //                     The LLVM Compiler Infrastructure | 
|  | 4 | // | 
| Chris Lattner | 5b12ab8 | 2007-12-29 19:59:25 +0000 | [diff] [blame] | 5 | // This file is distributed under the University of Illinois Open Source | 
|  | 6 | // License. See LICENSE.TXT for details. | 
| Chris Lattner | ddc135e | 2006-11-10 06:34:16 +0000 | [diff] [blame] | 7 | // | 
|  | 8 | //===----------------------------------------------------------------------===// | 
|  | 9 | // | 
|  | 10 | //  This file implements the ASTContext interface. | 
|  | 11 | // | 
|  | 12 | //===----------------------------------------------------------------------===// | 
|  | 13 |  | 
|  | 14 | #include "clang/AST/ASTContext.h" | 
| Chandler Carruth | 3a02247 | 2012-12-04 09:13:33 +0000 | [diff] [blame] | 15 | #include "CXXABI.h" | 
| Benjamin Kramer | ea70eb3 | 2012-12-01 15:09:41 +0000 | [diff] [blame] | 16 | #include "clang/AST/ASTMutationListener.h" | 
|  | 17 | #include "clang/AST/Attr.h" | 
| Ken Dyck | 8c89d59 | 2009-12-22 14:23:30 +0000 | [diff] [blame] | 18 | #include "clang/AST/CharUnits.h" | 
| Benjamin Kramer | ea70eb3 | 2012-12-01 15:09:41 +0000 | [diff] [blame] | 19 | #include "clang/AST/Comment.h" | 
| Dmitri Gribenko | ca7f80a | 2012-08-09 00:03:17 +0000 | [diff] [blame] | 20 | #include "clang/AST/CommentCommandTraits.h" | 
| Argyrios Kyrtzidis | faf0876 | 2008-08-07 20:55:28 +0000 | [diff] [blame] | 21 | #include "clang/AST/DeclCXX.h" | 
| Chandler Carruth | aa36b89 | 2015-12-30 03:40:23 +0000 | [diff] [blame] | 22 | #include "clang/AST/DeclContextInternals.h" | 
| Steve Naroff | 67391b8 | 2007-10-01 19:00:59 +0000 | [diff] [blame] | 23 | #include "clang/AST/DeclObjC.h" | 
| Douglas Gregor | ded2d7b | 2009-02-04 19:02:06 +0000 | [diff] [blame] | 24 | #include "clang/AST/DeclTemplate.h" | 
| Daniel Dunbar | 221fa94 | 2008-08-11 04:54:23 +0000 | [diff] [blame] | 25 | #include "clang/AST/Expr.h" | 
| John McCall | 87fe5d5 | 2010-05-20 01:18:31 +0000 | [diff] [blame] | 26 | #include "clang/AST/ExprCXX.h" | 
| Douglas Gregor | ef84c4b | 2009-04-09 22:27:44 +0000 | [diff] [blame] | 27 | #include "clang/AST/ExternalASTSource.h" | 
| Peter Collingbourne | 0ff0b37 | 2011-01-13 18:57:25 +0000 | [diff] [blame] | 28 | #include "clang/AST/Mangle.h" | 
| Reid Kleckner | d8110b6 | 2013-09-10 20:14:30 +0000 | [diff] [blame] | 29 | #include "clang/AST/MangleNumberingContext.h" | 
| Benjamin Kramer | ea70eb3 | 2012-12-01 15:09:41 +0000 | [diff] [blame] | 30 | #include "clang/AST/RecordLayout.h" | 
| Reid Kleckner | 2ab0ac5 | 2013-06-17 12:56:08 +0000 | [diff] [blame] | 31 | #include "clang/AST/RecursiveASTVisitor.h" | 
| Benjamin Kramer | ea70eb3 | 2012-12-01 15:09:41 +0000 | [diff] [blame] | 32 | #include "clang/AST/TypeLoc.h" | 
| Reid Kleckner | 96f8f93 | 2014-02-05 17:27:08 +0000 | [diff] [blame] | 33 | #include "clang/AST/VTableBuilder.h" | 
| Chris Lattner | 15ba949 | 2009-06-14 01:54:56 +0000 | [diff] [blame] | 34 | #include "clang/Basic/Builtins.h" | 
| Chris Lattner | d286851 | 2009-03-28 03:45:20 +0000 | [diff] [blame] | 35 | #include "clang/Basic/SourceManager.h" | 
| Chris Lattner | 4dc8a6f | 2007-05-20 23:50:58 +0000 | [diff] [blame] | 36 | #include "clang/Basic/TargetInfo.h" | 
| Anders Carlsson | d849982 | 2007-10-29 05:01:08 +0000 | [diff] [blame] | 37 | #include "llvm/ADT/StringExtras.h" | 
| Robert Lytton | eaf6f36 | 2013-11-12 10:09:34 +0000 | [diff] [blame] | 38 | #include "llvm/ADT/Triple.h" | 
| Benjamin Kramer | ea70eb3 | 2012-12-01 15:09:41 +0000 | [diff] [blame] | 39 | #include "llvm/Support/Capacity.h" | 
| Nate Begeman | b699c9b | 2009-01-18 06:42:49 +0000 | [diff] [blame] | 40 | #include "llvm/Support/MathExtras.h" | 
| Benjamin Kramer | 1402ce3 | 2009-10-24 09:57:09 +0000 | [diff] [blame] | 41 | #include "llvm/Support/raw_ostream.h" | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 42 | #include <map> | 
| Anders Carlsson | a4267a6 | 2009-07-18 21:19:52 +0000 | [diff] [blame] | 43 |  | 
| Chris Lattner | ddc135e | 2006-11-10 06:34:16 +0000 | [diff] [blame] | 44 | using namespace clang; | 
|  | 45 |  | 
| Douglas Gregor | 9672f92 | 2010-07-03 00:47:00 +0000 | [diff] [blame] | 46 | unsigned ASTContext::NumImplicitDefaultConstructors; | 
|  | 47 | unsigned ASTContext::NumImplicitDefaultConstructorsDeclared; | 
| Douglas Gregor | a6d6950 | 2010-07-02 23:41:54 +0000 | [diff] [blame] | 48 | unsigned ASTContext::NumImplicitCopyConstructors; | 
|  | 49 | unsigned ASTContext::NumImplicitCopyConstructorsDeclared; | 
| Alexis Hunt | fcaeae4 | 2011-05-25 20:50:04 +0000 | [diff] [blame] | 50 | unsigned ASTContext::NumImplicitMoveConstructors; | 
|  | 51 | unsigned ASTContext::NumImplicitMoveConstructorsDeclared; | 
| Douglas Gregor | 330b9cf | 2010-07-02 21:50:04 +0000 | [diff] [blame] | 52 | unsigned ASTContext::NumImplicitCopyAssignmentOperators; | 
|  | 53 | unsigned ASTContext::NumImplicitCopyAssignmentOperatorsDeclared; | 
| Alexis Hunt | fcaeae4 | 2011-05-25 20:50:04 +0000 | [diff] [blame] | 54 | unsigned ASTContext::NumImplicitMoveAssignmentOperators; | 
|  | 55 | unsigned ASTContext::NumImplicitMoveAssignmentOperatorsDeclared; | 
| Douglas Gregor | 7454c56 | 2010-07-02 20:37:36 +0000 | [diff] [blame] | 56 | unsigned ASTContext::NumImplicitDestructors; | 
|  | 57 | unsigned ASTContext::NumImplicitDestructorsDeclared; | 
|  | 58 |  | 
| Steve Naroff | 0af9120 | 2007-04-27 21:51:21 +0000 | [diff] [blame] | 59 | enum FloatingRank { | 
| Nemanja Ivanovic | bb1ea2d | 2016-05-09 08:52:33 +0000 | [diff] [blame] | 60 | HalfRank, FloatRank, DoubleRank, LongDoubleRank, Float128Rank | 
| Steve Naroff | 0af9120 | 2007-04-27 21:51:21 +0000 | [diff] [blame] | 61 | }; | 
|  | 62 |  | 
| Dmitri Gribenko | f26054f | 2012-07-11 21:38:39 +0000 | [diff] [blame] | 63 | RawComment *ASTContext::getRawCommentForDeclNoCache(const Decl *D) const { | 
| Dmitri Gribenko | aab8383 | 2012-06-20 00:34:58 +0000 | [diff] [blame] | 64 | if (!CommentsLoaded && ExternalSource) { | 
|  | 65 | ExternalSource->ReadComments(); | 
| Dmitri Gribenko | 9ee0e30 | 2014-03-27 15:40:39 +0000 | [diff] [blame] | 66 |  | 
|  | 67 | #ifndef NDEBUG | 
|  | 68 | ArrayRef<RawComment *> RawComments = Comments.getComments(); | 
|  | 69 | assert(std::is_sorted(RawComments.begin(), RawComments.end(), | 
|  | 70 | BeforeThanCompare<RawComment>(SourceMgr))); | 
|  | 71 | #endif | 
|  | 72 |  | 
| Dmitri Gribenko | aab8383 | 2012-06-20 00:34:58 +0000 | [diff] [blame] | 73 | CommentsLoaded = true; | 
|  | 74 | } | 
|  | 75 |  | 
|  | 76 | assert(D); | 
|  | 77 |  | 
| Dmitri Gribenko | df17d64 | 2012-06-28 16:19:39 +0000 | [diff] [blame] | 78 | // User can not attach documentation to implicit declarations. | 
|  | 79 | if (D->isImplicit()) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 80 | return nullptr; | 
| Dmitri Gribenko | df17d64 | 2012-06-28 16:19:39 +0000 | [diff] [blame] | 81 |  | 
| Dmitri Gribenko | b261088 | 2012-08-14 17:17:18 +0000 | [diff] [blame] | 82 | // User can not attach documentation to implicit instantiations. | 
|  | 83 | if (const FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) { | 
|  | 84 | if (FD->getTemplateSpecializationKind() == TSK_ImplicitInstantiation) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 85 | return nullptr; | 
| Dmitri Gribenko | b261088 | 2012-08-14 17:17:18 +0000 | [diff] [blame] | 86 | } | 
|  | 87 |  | 
| Dmitri Gribenko | b1ad993 | 2012-08-20 22:36:31 +0000 | [diff] [blame] | 88 | if (const VarDecl *VD = dyn_cast<VarDecl>(D)) { | 
|  | 89 | if (VD->isStaticDataMember() && | 
|  | 90 | VD->getTemplateSpecializationKind() == TSK_ImplicitInstantiation) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 91 | return nullptr; | 
| Dmitri Gribenko | b1ad993 | 2012-08-20 22:36:31 +0000 | [diff] [blame] | 92 | } | 
|  | 93 |  | 
|  | 94 | if (const CXXRecordDecl *CRD = dyn_cast<CXXRecordDecl>(D)) { | 
|  | 95 | if (CRD->getTemplateSpecializationKind() == TSK_ImplicitInstantiation) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 96 | return nullptr; | 
| Dmitri Gribenko | b1ad993 | 2012-08-20 22:36:31 +0000 | [diff] [blame] | 97 | } | 
|  | 98 |  | 
| Dmitri Gribenko | 01b0651 | 2013-01-27 21:18:39 +0000 | [diff] [blame] | 99 | if (const ClassTemplateSpecializationDecl *CTSD = | 
|  | 100 | dyn_cast<ClassTemplateSpecializationDecl>(D)) { | 
|  | 101 | TemplateSpecializationKind TSK = CTSD->getSpecializationKind(); | 
|  | 102 | if (TSK == TSK_ImplicitInstantiation || | 
|  | 103 | TSK == TSK_Undeclared) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 104 | return nullptr; | 
| Dmitri Gribenko | 01b0651 | 2013-01-27 21:18:39 +0000 | [diff] [blame] | 105 | } | 
|  | 106 |  | 
| Dmitri Gribenko | b1ad993 | 2012-08-20 22:36:31 +0000 | [diff] [blame] | 107 | if (const EnumDecl *ED = dyn_cast<EnumDecl>(D)) { | 
|  | 108 | if (ED->getTemplateSpecializationKind() == TSK_ImplicitInstantiation) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 109 | return nullptr; | 
| Dmitri Gribenko | b1ad993 | 2012-08-20 22:36:31 +0000 | [diff] [blame] | 110 | } | 
| Fariborz Jahanian | 799a403 | 2013-04-17 21:05:20 +0000 | [diff] [blame] | 111 | if (const TagDecl *TD = dyn_cast<TagDecl>(D)) { | 
|  | 112 | // When tag declaration (but not definition!) is part of the | 
|  | 113 | // decl-specifier-seq of some other declaration, it doesn't get comment | 
|  | 114 | if (TD->isEmbeddedInDeclarator() && !TD->isCompleteDefinition()) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 115 | return nullptr; | 
| Fariborz Jahanian | 799a403 | 2013-04-17 21:05:20 +0000 | [diff] [blame] | 116 | } | 
| Dmitri Gribenko | aab8383 | 2012-06-20 00:34:58 +0000 | [diff] [blame] | 117 | // TODO: handle comments for function parameters properly. | 
|  | 118 | if (isa<ParmVarDecl>(D)) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 119 | return nullptr; | 
| Dmitri Gribenko | aab8383 | 2012-06-20 00:34:58 +0000 | [diff] [blame] | 120 |  | 
| Dmitri Gribenko | 34df220 | 2012-07-31 22:37:06 +0000 | [diff] [blame] | 121 | // TODO: we could look up template parameter documentation in the template | 
|  | 122 | // documentation. | 
|  | 123 | if (isa<TemplateTypeParmDecl>(D) || | 
|  | 124 | isa<NonTypeTemplateParmDecl>(D) || | 
|  | 125 | isa<TemplateTemplateParmDecl>(D)) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 126 | return nullptr; | 
| Dmitri Gribenko | 34df220 | 2012-07-31 22:37:06 +0000 | [diff] [blame] | 127 |  | 
| Dmitri Gribenko | 7dd29d4 | 2012-07-06 18:19:34 +0000 | [diff] [blame] | 128 | ArrayRef<RawComment *> RawComments = Comments.getComments(); | 
| Dmitri Gribenko | aab8383 | 2012-06-20 00:34:58 +0000 | [diff] [blame] | 129 |  | 
|  | 130 | // If there are no comments anywhere, we won't find anything. | 
|  | 131 | if (RawComments.empty()) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 132 | return nullptr; | 
| Dmitri Gribenko | aab8383 | 2012-06-20 00:34:58 +0000 | [diff] [blame] | 133 |  | 
| Dmitri Gribenko | e7bb944 | 2012-07-13 01:06:46 +0000 | [diff] [blame] | 134 | // Find declaration location. | 
|  | 135 | // For Objective-C declarations we generally don't expect to have multiple | 
|  | 136 | // declarators, thus use declaration starting location as the "declaration | 
|  | 137 | // location". | 
|  | 138 | // For all other declarations multiple declarators are used quite frequently, | 
|  | 139 | // so we use the location of the identifier as the "declaration location". | 
|  | 140 | SourceLocation DeclLoc; | 
|  | 141 | if (isa<ObjCMethodDecl>(D) || isa<ObjCContainerDecl>(D) || | 
| Dmitri Gribenko | 34df220 | 2012-07-31 22:37:06 +0000 | [diff] [blame] | 142 | isa<ObjCPropertyDecl>(D) || | 
| Dmitri Gribenko | 7f4b377 | 2012-08-02 20:49:51 +0000 | [diff] [blame] | 143 | isa<RedeclarableTemplateDecl>(D) || | 
|  | 144 | isa<ClassTemplateSpecializationDecl>(D)) | 
| Dmitri Gribenko | e7bb944 | 2012-07-13 01:06:46 +0000 | [diff] [blame] | 145 | DeclLoc = D->getLocStart(); | 
| Fariborz Jahanian | b64e95f | 2013-07-24 22:58:51 +0000 | [diff] [blame] | 146 | else { | 
| Dmitri Gribenko | e7bb944 | 2012-07-13 01:06:46 +0000 | [diff] [blame] | 147 | DeclLoc = D->getLocation(); | 
| Dmitri Gribenko | ef099dc | 2014-03-27 16:40:51 +0000 | [diff] [blame] | 148 | if (DeclLoc.isMacroID()) { | 
|  | 149 | if (isa<TypedefDecl>(D)) { | 
|  | 150 | // If location of the typedef name is in a macro, it is because being | 
|  | 151 | // declared via a macro. Try using declaration's starting location as | 
|  | 152 | // the "declaration location". | 
|  | 153 | DeclLoc = D->getLocStart(); | 
|  | 154 | } else if (const TagDecl *TD = dyn_cast<TagDecl>(D)) { | 
|  | 155 | // If location of the tag decl is inside a macro, but the spelling of | 
|  | 156 | // the tag name comes from a macro argument, it looks like a special | 
|  | 157 | // macro like NS_ENUM is being used to define the tag decl.  In that | 
|  | 158 | // case, adjust the source location to the expansion loc so that we can | 
|  | 159 | // attach the comment to the tag decl. | 
|  | 160 | if (SourceMgr.isMacroArgExpansion(DeclLoc) && | 
|  | 161 | TD->isCompleteDefinition()) | 
|  | 162 | DeclLoc = SourceMgr.getExpansionLoc(DeclLoc); | 
|  | 163 | } | 
|  | 164 | } | 
| Fariborz Jahanian | b64e95f | 2013-07-24 22:58:51 +0000 | [diff] [blame] | 165 | } | 
| Dmitri Gribenko | e7bb944 | 2012-07-13 01:06:46 +0000 | [diff] [blame] | 166 |  | 
| Dmitri Gribenko | aab8383 | 2012-06-20 00:34:58 +0000 | [diff] [blame] | 167 | // If the declaration doesn't map directly to a location in a file, we | 
|  | 168 | // can't find the comment. | 
| Dmitri Gribenko | aab8383 | 2012-06-20 00:34:58 +0000 | [diff] [blame] | 169 | if (DeclLoc.isInvalid() || !DeclLoc.isFileID()) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 170 | return nullptr; | 
| Dmitri Gribenko | aab8383 | 2012-06-20 00:34:58 +0000 | [diff] [blame] | 171 |  | 
|  | 172 | // Find the comment that occurs just after this declaration. | 
| Dmitri Gribenko | 82ea947 | 2012-07-17 22:01:09 +0000 | [diff] [blame] | 173 | ArrayRef<RawComment *>::iterator Comment; | 
|  | 174 | { | 
|  | 175 | // When searching for comments during parsing, the comment we are looking | 
|  | 176 | // for is usually among the last two comments we parsed -- check them | 
|  | 177 | // first. | 
| Dmitri Gribenko | a7d16ce | 2013-04-10 15:35:17 +0000 | [diff] [blame] | 178 | RawComment CommentAtDeclLoc( | 
|  | 179 | SourceMgr, SourceRange(DeclLoc), false, | 
|  | 180 | LangOpts.CommentOpts.ParseAllComments); | 
| Dmitri Gribenko | 82ea947 | 2012-07-17 22:01:09 +0000 | [diff] [blame] | 181 | BeforeThanCompare<RawComment> Compare(SourceMgr); | 
|  | 182 | ArrayRef<RawComment *>::iterator MaybeBeforeDecl = RawComments.end() - 1; | 
|  | 183 | bool Found = Compare(*MaybeBeforeDecl, &CommentAtDeclLoc); | 
|  | 184 | if (!Found && RawComments.size() >= 2) { | 
|  | 185 | MaybeBeforeDecl--; | 
|  | 186 | Found = Compare(*MaybeBeforeDecl, &CommentAtDeclLoc); | 
|  | 187 | } | 
|  | 188 |  | 
|  | 189 | if (Found) { | 
|  | 190 | Comment = MaybeBeforeDecl + 1; | 
|  | 191 | assert(Comment == std::lower_bound(RawComments.begin(), RawComments.end(), | 
|  | 192 | &CommentAtDeclLoc, Compare)); | 
|  | 193 | } else { | 
|  | 194 | // Slow path. | 
|  | 195 | Comment = std::lower_bound(RawComments.begin(), RawComments.end(), | 
|  | 196 | &CommentAtDeclLoc, Compare); | 
|  | 197 | } | 
|  | 198 | } | 
| Dmitri Gribenko | aab8383 | 2012-06-20 00:34:58 +0000 | [diff] [blame] | 199 |  | 
|  | 200 | // Decompose the location for the declaration and find the beginning of the | 
|  | 201 | // file buffer. | 
|  | 202 | std::pair<FileID, unsigned> DeclLocDecomp = SourceMgr.getDecomposedLoc(DeclLoc); | 
|  | 203 |  | 
|  | 204 | // First check whether we have a trailing comment. | 
|  | 205 | if (Comment != RawComments.end() && | 
| Dmitri Gribenko | 7dd29d4 | 2012-07-06 18:19:34 +0000 | [diff] [blame] | 206 | (*Comment)->isDocumentation() && (*Comment)->isTrailingComment() && | 
| Fariborz Jahanian | fad2854 | 2013-08-06 23:29:00 +0000 | [diff] [blame] | 207 | (isa<FieldDecl>(D) || isa<EnumConstantDecl>(D) || isa<VarDecl>(D) || | 
| Fariborz Jahanian | 3ab6222 | 2013-08-07 16:40:29 +0000 | [diff] [blame] | 208 | isa<ObjCMethodDecl>(D) || isa<ObjCPropertyDecl>(D))) { | 
| Dmitri Gribenko | aab8383 | 2012-06-20 00:34:58 +0000 | [diff] [blame] | 209 | std::pair<FileID, unsigned> CommentBeginDecomp | 
| Dmitri Gribenko | 7dd29d4 | 2012-07-06 18:19:34 +0000 | [diff] [blame] | 210 | = SourceMgr.getDecomposedLoc((*Comment)->getSourceRange().getBegin()); | 
| Dmitri Gribenko | aab8383 | 2012-06-20 00:34:58 +0000 | [diff] [blame] | 211 | // Check that Doxygen trailing comment comes after the declaration, starts | 
|  | 212 | // on the same line and in the same file as the declaration. | 
|  | 213 | if (DeclLocDecomp.first == CommentBeginDecomp.first && | 
|  | 214 | SourceMgr.getLineNumber(DeclLocDecomp.first, DeclLocDecomp.second) | 
|  | 215 | == SourceMgr.getLineNumber(CommentBeginDecomp.first, | 
|  | 216 | CommentBeginDecomp.second)) { | 
| Dmitri Gribenko | 7dd29d4 | 2012-07-06 18:19:34 +0000 | [diff] [blame] | 217 | return *Comment; | 
| Dmitri Gribenko | aab8383 | 2012-06-20 00:34:58 +0000 | [diff] [blame] | 218 | } | 
|  | 219 | } | 
|  | 220 |  | 
|  | 221 | // The comment just after the declaration was not a trailing comment. | 
|  | 222 | // Let's look at the previous comment. | 
|  | 223 | if (Comment == RawComments.begin()) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 224 | return nullptr; | 
| Dmitri Gribenko | aab8383 | 2012-06-20 00:34:58 +0000 | [diff] [blame] | 225 | --Comment; | 
|  | 226 |  | 
|  | 227 | // Check that we actually have a non-member Doxygen comment. | 
| Dmitri Gribenko | 7dd29d4 | 2012-07-06 18:19:34 +0000 | [diff] [blame] | 228 | if (!(*Comment)->isDocumentation() || (*Comment)->isTrailingComment()) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 229 | return nullptr; | 
| Dmitri Gribenko | aab8383 | 2012-06-20 00:34:58 +0000 | [diff] [blame] | 230 |  | 
|  | 231 | // Decompose the end of the comment. | 
|  | 232 | std::pair<FileID, unsigned> CommentEndDecomp | 
| Dmitri Gribenko | 7dd29d4 | 2012-07-06 18:19:34 +0000 | [diff] [blame] | 233 | = SourceMgr.getDecomposedLoc((*Comment)->getSourceRange().getEnd()); | 
| Dmitri Gribenko | aab8383 | 2012-06-20 00:34:58 +0000 | [diff] [blame] | 234 |  | 
|  | 235 | // If the comment and the declaration aren't in the same file, then they | 
|  | 236 | // aren't related. | 
|  | 237 | if (DeclLocDecomp.first != CommentEndDecomp.first) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 238 | return nullptr; | 
| Dmitri Gribenko | aab8383 | 2012-06-20 00:34:58 +0000 | [diff] [blame] | 239 |  | 
|  | 240 | // Get the corresponding buffer. | 
|  | 241 | bool Invalid = false; | 
|  | 242 | const char *Buffer = SourceMgr.getBufferData(DeclLocDecomp.first, | 
|  | 243 | &Invalid).data(); | 
|  | 244 | if (Invalid) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 245 | return nullptr; | 
| Dmitri Gribenko | aab8383 | 2012-06-20 00:34:58 +0000 | [diff] [blame] | 246 |  | 
|  | 247 | // Extract text between the comment and declaration. | 
|  | 248 | StringRef Text(Buffer + CommentEndDecomp.second, | 
|  | 249 | DeclLocDecomp.second - CommentEndDecomp.second); | 
|  | 250 |  | 
| Dmitri Gribenko | 7e8729b | 2012-06-27 23:43:37 +0000 | [diff] [blame] | 251 | // There should be no other declarations or preprocessor directives between | 
|  | 252 | // comment and declaration. | 
| Argyrios Kyrtzidis | b534d3a | 2013-07-26 18:38:12 +0000 | [diff] [blame] | 253 | if (Text.find_first_of(";{}#@") != StringRef::npos) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 254 | return nullptr; | 
| Dmitri Gribenko | aab8383 | 2012-06-20 00:34:58 +0000 | [diff] [blame] | 255 |  | 
| Dmitri Gribenko | 7dd29d4 | 2012-07-06 18:19:34 +0000 | [diff] [blame] | 256 | return *Comment; | 
| Dmitri Gribenko | aab8383 | 2012-06-20 00:34:58 +0000 | [diff] [blame] | 257 | } | 
|  | 258 |  | 
| Dmitri Gribenko | b261088 | 2012-08-14 17:17:18 +0000 | [diff] [blame] | 259 | namespace { | 
|  | 260 | /// If we have a 'templated' declaration for a template, adjust 'D' to | 
|  | 261 | /// refer to the actual template. | 
| Dmitri Gribenko | 9063180 | 2012-08-22 17:44:32 +0000 | [diff] [blame] | 262 | /// If we have an implicit instantiation, adjust 'D' to refer to template. | 
| Dmitri Gribenko | b261088 | 2012-08-14 17:17:18 +0000 | [diff] [blame] | 263 | const Decl *adjustDeclToTemplate(const Decl *D) { | 
| Douglas Gregor | 35ceb27 | 2012-08-13 16:37:30 +0000 | [diff] [blame] | 264 | if (const FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) { | 
| Dmitri Gribenko | 9063180 | 2012-08-22 17:44:32 +0000 | [diff] [blame] | 265 | // Is this function declaration part of a function template? | 
| Douglas Gregor | 35ceb27 | 2012-08-13 16:37:30 +0000 | [diff] [blame] | 266 | if (const FunctionTemplateDecl *FTD = FD->getDescribedFunctionTemplate()) | 
| Dmitri Gribenko | 9063180 | 2012-08-22 17:44:32 +0000 | [diff] [blame] | 267 | return FTD; | 
|  | 268 |  | 
|  | 269 | // Nothing to do if function is not an implicit instantiation. | 
|  | 270 | if (FD->getTemplateSpecializationKind() != TSK_ImplicitInstantiation) | 
|  | 271 | return D; | 
|  | 272 |  | 
|  | 273 | // Function is an implicit instantiation of a function template? | 
|  | 274 | if (const FunctionTemplateDecl *FTD = FD->getPrimaryTemplate()) | 
|  | 275 | return FTD; | 
|  | 276 |  | 
|  | 277 | // Function is instantiated from a member definition of a class template? | 
|  | 278 | if (const FunctionDecl *MemberDecl = | 
|  | 279 | FD->getInstantiatedFromMemberFunction()) | 
|  | 280 | return MemberDecl; | 
|  | 281 |  | 
|  | 282 | return D; | 
| Douglas Gregor | 35ceb27 | 2012-08-13 16:37:30 +0000 | [diff] [blame] | 283 | } | 
| Dmitri Gribenko | 9063180 | 2012-08-22 17:44:32 +0000 | [diff] [blame] | 284 | if (const VarDecl *VD = dyn_cast<VarDecl>(D)) { | 
|  | 285 | // Static data member is instantiated from a member definition of a class | 
|  | 286 | // template? | 
|  | 287 | if (VD->isStaticDataMember()) | 
|  | 288 | if (const VarDecl *MemberDecl = VD->getInstantiatedFromStaticDataMember()) | 
|  | 289 | return MemberDecl; | 
|  | 290 |  | 
|  | 291 | return D; | 
|  | 292 | } | 
|  | 293 | if (const CXXRecordDecl *CRD = dyn_cast<CXXRecordDecl>(D)) { | 
|  | 294 | // Is this class declaration part of a class template? | 
|  | 295 | if (const ClassTemplateDecl *CTD = CRD->getDescribedClassTemplate()) | 
|  | 296 | return CTD; | 
|  | 297 |  | 
|  | 298 | // Class is an implicit instantiation of a class template or partial | 
|  | 299 | // specialization? | 
|  | 300 | if (const ClassTemplateSpecializationDecl *CTSD = | 
|  | 301 | dyn_cast<ClassTemplateSpecializationDecl>(CRD)) { | 
|  | 302 | if (CTSD->getSpecializationKind() != TSK_ImplicitInstantiation) | 
|  | 303 | return D; | 
|  | 304 | llvm::PointerUnion<ClassTemplateDecl *, | 
|  | 305 | ClassTemplatePartialSpecializationDecl *> | 
|  | 306 | PU = CTSD->getSpecializedTemplateOrPartial(); | 
|  | 307 | return PU.is<ClassTemplateDecl*>() ? | 
|  | 308 | static_cast<const Decl*>(PU.get<ClassTemplateDecl *>()) : | 
|  | 309 | static_cast<const Decl*>( | 
|  | 310 | PU.get<ClassTemplatePartialSpecializationDecl *>()); | 
|  | 311 | } | 
|  | 312 |  | 
|  | 313 | // Class is instantiated from a member definition of a class template? | 
|  | 314 | if (const MemberSpecializationInfo *Info = | 
|  | 315 | CRD->getMemberSpecializationInfo()) | 
|  | 316 | return Info->getInstantiatedFrom(); | 
|  | 317 |  | 
|  | 318 | return D; | 
|  | 319 | } | 
|  | 320 | if (const EnumDecl *ED = dyn_cast<EnumDecl>(D)) { | 
|  | 321 | // Enum is instantiated from a member definition of a class template? | 
|  | 322 | if (const EnumDecl *MemberDecl = ED->getInstantiatedFromMemberEnum()) | 
|  | 323 | return MemberDecl; | 
|  | 324 |  | 
|  | 325 | return D; | 
|  | 326 | } | 
|  | 327 | // FIXME: Adjust alias templates? | 
| Dmitri Gribenko | b261088 | 2012-08-14 17:17:18 +0000 | [diff] [blame] | 328 | return D; | 
|  | 329 | } | 
| Eugene Zelenko | d4304d2 | 2015-11-04 21:37:17 +0000 | [diff] [blame] | 330 | } // anonymous namespace | 
| Dmitri Gribenko | b261088 | 2012-08-14 17:17:18 +0000 | [diff] [blame] | 331 |  | 
| Dmitri Gribenko | 4ae66a3 | 2012-08-16 18:19:43 +0000 | [diff] [blame] | 332 | const RawComment *ASTContext::getRawCommentForAnyRedecl( | 
|  | 333 | const Decl *D, | 
|  | 334 | const Decl **OriginalDecl) const { | 
| Dmitri Gribenko | b261088 | 2012-08-14 17:17:18 +0000 | [diff] [blame] | 335 | D = adjustDeclToTemplate(D); | 
| Douglas Gregor | 35ceb27 | 2012-08-13 16:37:30 +0000 | [diff] [blame] | 336 |  | 
| Dmitri Gribenko | a43ec18 | 2012-08-11 00:51:43 +0000 | [diff] [blame] | 337 | // Check whether we have cached a comment for this declaration already. | 
|  | 338 | { | 
|  | 339 | llvm::DenseMap<const Decl *, RawCommentAndCacheFlags>::iterator Pos = | 
|  | 340 | RedeclComments.find(D); | 
|  | 341 | if (Pos != RedeclComments.end()) { | 
|  | 342 | const RawCommentAndCacheFlags &Raw = Pos->second; | 
| Dmitri Gribenko | 4ae66a3 | 2012-08-16 18:19:43 +0000 | [diff] [blame] | 343 | if (Raw.getKind() != RawCommentAndCacheFlags::NoCommentInDecl) { | 
|  | 344 | if (OriginalDecl) | 
|  | 345 | *OriginalDecl = Raw.getOriginalDecl(); | 
| Dmitri Gribenko | a43ec18 | 2012-08-11 00:51:43 +0000 | [diff] [blame] | 346 | return Raw.getRaw(); | 
| Dmitri Gribenko | 4ae66a3 | 2012-08-16 18:19:43 +0000 | [diff] [blame] | 347 | } | 
| Dmitri Gribenko | a43ec18 | 2012-08-11 00:51:43 +0000 | [diff] [blame] | 348 | } | 
| Dmitri Gribenko | ec92531 | 2012-07-06 00:28:32 +0000 | [diff] [blame] | 349 | } | 
| Dmitri Gribenko | aab8383 | 2012-06-20 00:34:58 +0000 | [diff] [blame] | 350 |  | 
| Dmitri Gribenko | a43ec18 | 2012-08-11 00:51:43 +0000 | [diff] [blame] | 351 | // Search for comments attached to declarations in the redeclaration chain. | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 352 | const RawComment *RC = nullptr; | 
|  | 353 | const Decl *OriginalDeclForRC = nullptr; | 
| Aaron Ballman | 86c9390 | 2014-03-06 23:45:36 +0000 | [diff] [blame] | 354 | for (auto I : D->redecls()) { | 
| Dmitri Gribenko | a43ec18 | 2012-08-11 00:51:43 +0000 | [diff] [blame] | 355 | llvm::DenseMap<const Decl *, RawCommentAndCacheFlags>::iterator Pos = | 
| Aaron Ballman | 86c9390 | 2014-03-06 23:45:36 +0000 | [diff] [blame] | 356 | RedeclComments.find(I); | 
| Dmitri Gribenko | a43ec18 | 2012-08-11 00:51:43 +0000 | [diff] [blame] | 357 | if (Pos != RedeclComments.end()) { | 
|  | 358 | const RawCommentAndCacheFlags &Raw = Pos->second; | 
|  | 359 | if (Raw.getKind() != RawCommentAndCacheFlags::NoCommentInDecl) { | 
|  | 360 | RC = Raw.getRaw(); | 
| Dmitri Gribenko | 4ae66a3 | 2012-08-16 18:19:43 +0000 | [diff] [blame] | 361 | OriginalDeclForRC = Raw.getOriginalDecl(); | 
| Dmitri Gribenko | a43ec18 | 2012-08-11 00:51:43 +0000 | [diff] [blame] | 362 | break; | 
|  | 363 | } | 
|  | 364 | } else { | 
| Aaron Ballman | 86c9390 | 2014-03-06 23:45:36 +0000 | [diff] [blame] | 365 | RC = getRawCommentForDeclNoCache(I); | 
|  | 366 | OriginalDeclForRC = I; | 
| Dmitri Gribenko | a43ec18 | 2012-08-11 00:51:43 +0000 | [diff] [blame] | 367 | RawCommentAndCacheFlags Raw; | 
|  | 368 | if (RC) { | 
| Will Wilson | f9de536 | 2015-10-27 17:01:10 +0000 | [diff] [blame] | 369 | // Call order swapped to work around ICE in VS2015 RTM (Release Win32) | 
|  | 370 | // https://connect.microsoft.com/VisualStudio/feedback/details/1741530 | 
| Dmitri Gribenko | a43ec18 | 2012-08-11 00:51:43 +0000 | [diff] [blame] | 371 | Raw.setKind(RawCommentAndCacheFlags::FromDecl); | 
| Will Wilson | f9de536 | 2015-10-27 17:01:10 +0000 | [diff] [blame] | 372 | Raw.setRaw(RC); | 
| Dmitri Gribenko | a43ec18 | 2012-08-11 00:51:43 +0000 | [diff] [blame] | 373 | } else | 
|  | 374 | Raw.setKind(RawCommentAndCacheFlags::NoCommentInDecl); | 
| Aaron Ballman | 86c9390 | 2014-03-06 23:45:36 +0000 | [diff] [blame] | 375 | Raw.setOriginalDecl(I); | 
|  | 376 | RedeclComments[I] = Raw; | 
| Dmitri Gribenko | a43ec18 | 2012-08-11 00:51:43 +0000 | [diff] [blame] | 377 | if (RC) | 
|  | 378 | break; | 
|  | 379 | } | 
|  | 380 | } | 
|  | 381 |  | 
| Dmitri Gribenko | 5c8897d | 2012-06-28 16:25:36 +0000 | [diff] [blame] | 382 | // If we found a comment, it should be a documentation comment. | 
|  | 383 | assert(!RC || RC->isDocumentation()); | 
| Dmitri Gribenko | a43ec18 | 2012-08-11 00:51:43 +0000 | [diff] [blame] | 384 |  | 
| Dmitri Gribenko | 4ae66a3 | 2012-08-16 18:19:43 +0000 | [diff] [blame] | 385 | if (OriginalDecl) | 
|  | 386 | *OriginalDecl = OriginalDeclForRC; | 
|  | 387 |  | 
| Dmitri Gribenko | a43ec18 | 2012-08-11 00:51:43 +0000 | [diff] [blame] | 388 | // Update cache for every declaration in the redeclaration chain. | 
|  | 389 | RawCommentAndCacheFlags Raw; | 
|  | 390 | Raw.setRaw(RC); | 
|  | 391 | Raw.setKind(RawCommentAndCacheFlags::FromRedecl); | 
| Dmitri Gribenko | 4ae66a3 | 2012-08-16 18:19:43 +0000 | [diff] [blame] | 392 | Raw.setOriginalDecl(OriginalDeclForRC); | 
| Dmitri Gribenko | a43ec18 | 2012-08-11 00:51:43 +0000 | [diff] [blame] | 393 |  | 
| Aaron Ballman | 86c9390 | 2014-03-06 23:45:36 +0000 | [diff] [blame] | 394 | for (auto I : D->redecls()) { | 
|  | 395 | RawCommentAndCacheFlags &R = RedeclComments[I]; | 
| Dmitri Gribenko | a43ec18 | 2012-08-11 00:51:43 +0000 | [diff] [blame] | 396 | if (R.getKind() == RawCommentAndCacheFlags::NoCommentInDecl) | 
|  | 397 | R = Raw; | 
|  | 398 | } | 
|  | 399 |  | 
| Dmitri Gribenko | aab8383 | 2012-06-20 00:34:58 +0000 | [diff] [blame] | 400 | return RC; | 
|  | 401 | } | 
|  | 402 |  | 
| Fariborz Jahanian | 1c883b9 | 2012-10-10 18:34:52 +0000 | [diff] [blame] | 403 | static void addRedeclaredMethods(const ObjCMethodDecl *ObjCMethod, | 
|  | 404 | SmallVectorImpl<const NamedDecl *> &Redeclared) { | 
|  | 405 | const DeclContext *DC = ObjCMethod->getDeclContext(); | 
|  | 406 | if (const ObjCImplDecl *IMD = dyn_cast<ObjCImplDecl>(DC)) { | 
|  | 407 | const ObjCInterfaceDecl *ID = IMD->getClassInterface(); | 
|  | 408 | if (!ID) | 
|  | 409 | return; | 
|  | 410 | // Add redeclared method here. | 
| Aaron Ballman | b4a5345 | 2014-03-13 21:57:01 +0000 | [diff] [blame] | 411 | for (const auto *Ext : ID->known_extensions()) { | 
| Fariborz Jahanian | 1c883b9 | 2012-10-10 18:34:52 +0000 | [diff] [blame] | 412 | if (ObjCMethodDecl *RedeclaredMethod = | 
| Douglas Gregor | 048fbfa | 2013-01-16 23:00:23 +0000 | [diff] [blame] | 413 | Ext->getMethod(ObjCMethod->getSelector(), | 
| Fariborz Jahanian | 1c883b9 | 2012-10-10 18:34:52 +0000 | [diff] [blame] | 414 | ObjCMethod->isInstanceMethod())) | 
|  | 415 | Redeclared.push_back(RedeclaredMethod); | 
|  | 416 | } | 
|  | 417 | } | 
|  | 418 | } | 
|  | 419 |  | 
| Fariborz Jahanian | 42e3132 | 2012-10-11 23:52:50 +0000 | [diff] [blame] | 420 | comments::FullComment *ASTContext::cloneFullComment(comments::FullComment *FC, | 
|  | 421 | const Decl *D) const { | 
|  | 422 | comments::DeclInfo *ThisDeclInfo = new (*this) comments::DeclInfo; | 
|  | 423 | ThisDeclInfo->CommentDecl = D; | 
|  | 424 | ThisDeclInfo->IsFilled = false; | 
|  | 425 | ThisDeclInfo->fill(); | 
|  | 426 | ThisDeclInfo->CommentDecl = FC->getDecl(); | 
| Argyrios Kyrtzidis | 7daabbd | 2014-04-27 22:53:03 +0000 | [diff] [blame] | 427 | if (!ThisDeclInfo->TemplateParameters) | 
|  | 428 | ThisDeclInfo->TemplateParameters = FC->getDeclInfo()->TemplateParameters; | 
| Fariborz Jahanian | 42e3132 | 2012-10-11 23:52:50 +0000 | [diff] [blame] | 429 | comments::FullComment *CFC = | 
|  | 430 | new (*this) comments::FullComment(FC->getBlocks(), | 
|  | 431 | ThisDeclInfo); | 
|  | 432 | return CFC; | 
| Fariborz Jahanian | 42e3132 | 2012-10-11 23:52:50 +0000 | [diff] [blame] | 433 | } | 
|  | 434 |  | 
| Richard Smith | b39b9d5 | 2013-05-21 05:24:00 +0000 | [diff] [blame] | 435 | comments::FullComment *ASTContext::getLocalCommentForDeclUncached(const Decl *D) const { | 
|  | 436 | const RawComment *RC = getRawCommentForDeclNoCache(D); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 437 | return RC ? RC->parse(*this, nullptr, D) : nullptr; | 
| Richard Smith | b39b9d5 | 2013-05-21 05:24:00 +0000 | [diff] [blame] | 438 | } | 
|  | 439 |  | 
| Dmitri Gribenko | 6743e04 | 2012-09-29 11:40:46 +0000 | [diff] [blame] | 440 | comments::FullComment *ASTContext::getCommentForDecl( | 
|  | 441 | const Decl *D, | 
|  | 442 | const Preprocessor *PP) const { | 
| Fariborz Jahanian | 096f7c1 | 2013-05-13 17:27:00 +0000 | [diff] [blame] | 443 | if (D->isInvalidDecl()) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 444 | return nullptr; | 
| Dmitri Gribenko | b261088 | 2012-08-14 17:17:18 +0000 | [diff] [blame] | 445 | D = adjustDeclToTemplate(D); | 
| Fariborz Jahanian | 1c883b9 | 2012-10-10 18:34:52 +0000 | [diff] [blame] | 446 |  | 
| Dmitri Gribenko | b261088 | 2012-08-14 17:17:18 +0000 | [diff] [blame] | 447 | const Decl *Canonical = D->getCanonicalDecl(); | 
|  | 448 | llvm::DenseMap<const Decl *, comments::FullComment *>::iterator Pos = | 
|  | 449 | ParsedComments.find(Canonical); | 
| Fariborz Jahanian | 1c883b9 | 2012-10-10 18:34:52 +0000 | [diff] [blame] | 450 |  | 
|  | 451 | if (Pos != ParsedComments.end()) { | 
| Fariborz Jahanian | 42e3132 | 2012-10-11 23:52:50 +0000 | [diff] [blame] | 452 | if (Canonical != D) { | 
| Fariborz Jahanian | 1c883b9 | 2012-10-10 18:34:52 +0000 | [diff] [blame] | 453 | comments::FullComment *FC = Pos->second; | 
| Fariborz Jahanian | 42e3132 | 2012-10-11 23:52:50 +0000 | [diff] [blame] | 454 | comments::FullComment *CFC = cloneFullComment(FC, D); | 
| Fariborz Jahanian | 1c883b9 | 2012-10-10 18:34:52 +0000 | [diff] [blame] | 455 | return CFC; | 
|  | 456 | } | 
| Dmitri Gribenko | b261088 | 2012-08-14 17:17:18 +0000 | [diff] [blame] | 457 | return Pos->second; | 
| Fariborz Jahanian | 1c883b9 | 2012-10-10 18:34:52 +0000 | [diff] [blame] | 458 | } | 
|  | 459 |  | 
| Dmitri Gribenko | 4ae66a3 | 2012-08-16 18:19:43 +0000 | [diff] [blame] | 460 | const Decl *OriginalDecl; | 
| Fariborz Jahanian | 1c883b9 | 2012-10-10 18:34:52 +0000 | [diff] [blame] | 461 |  | 
| Dmitri Gribenko | 4ae66a3 | 2012-08-16 18:19:43 +0000 | [diff] [blame] | 462 | const RawComment *RC = getRawCommentForAnyRedecl(D, &OriginalDecl); | 
| Fariborz Jahanian | 1c883b9 | 2012-10-10 18:34:52 +0000 | [diff] [blame] | 463 | if (!RC) { | 
|  | 464 | if (isa<ObjCMethodDecl>(D) || isa<FunctionDecl>(D)) { | 
| Dmitri Gribenko | 941ab0f | 2012-11-03 14:24:57 +0000 | [diff] [blame] | 465 | SmallVector<const NamedDecl*, 8> Overridden; | 
| Fariborz Jahanian | 37494a1 | 2013-01-12 00:28:34 +0000 | [diff] [blame] | 466 | const ObjCMethodDecl *OMD = dyn_cast<ObjCMethodDecl>(D); | 
| Fariborz Jahanian | 66024d0 | 2013-01-25 23:08:39 +0000 | [diff] [blame] | 467 | if (OMD && OMD->isPropertyAccessor()) | 
|  | 468 | if (const ObjCPropertyDecl *PDecl = OMD->findPropertyDecl()) | 
|  | 469 | if (comments::FullComment *FC = getCommentForDecl(PDecl, PP)) | 
|  | 470 | return cloneFullComment(FC, D); | 
| Fariborz Jahanian | 37494a1 | 2013-01-12 00:28:34 +0000 | [diff] [blame] | 471 | if (OMD) | 
| Dmitri Gribenko | 941ab0f | 2012-11-03 14:24:57 +0000 | [diff] [blame] | 472 | addRedeclaredMethods(OMD, Overridden); | 
|  | 473 | getOverriddenMethods(dyn_cast<NamedDecl>(D), Overridden); | 
| Fariborz Jahanian | 66024d0 | 2013-01-25 23:08:39 +0000 | [diff] [blame] | 474 | for (unsigned i = 0, e = Overridden.size(); i < e; i++) | 
|  | 475 | if (comments::FullComment *FC = getCommentForDecl(Overridden[i], PP)) | 
|  | 476 | return cloneFullComment(FC, D); | 
| Fariborz Jahanian | 1c883b9 | 2012-10-10 18:34:52 +0000 | [diff] [blame] | 477 | } | 
| Fariborz Jahanian | 6384fbb | 2013-05-02 15:44:16 +0000 | [diff] [blame] | 478 | else if (const TypedefNameDecl *TD = dyn_cast<TypedefNameDecl>(D)) { | 
| Dmitri Gribenko | 01b0651 | 2013-01-27 21:18:39 +0000 | [diff] [blame] | 479 | // Attach any tag type's documentation to its typedef if latter | 
| Fariborz Jahanian | 66024d0 | 2013-01-25 23:08:39 +0000 | [diff] [blame] | 480 | // does not have one of its own. | 
| Fariborz Jahanian | 40abf34 | 2013-01-25 22:48:32 +0000 | [diff] [blame] | 481 | QualType QT = TD->getUnderlyingType(); | 
| Dmitri Gribenko | 01b0651 | 2013-01-27 21:18:39 +0000 | [diff] [blame] | 482 | if (const TagType *TT = QT->getAs<TagType>()) | 
|  | 483 | if (const Decl *TD = TT->getDecl()) | 
|  | 484 | if (comments::FullComment *FC = getCommentForDecl(TD, PP)) | 
| Fariborz Jahanian | 66024d0 | 2013-01-25 23:08:39 +0000 | [diff] [blame] | 485 | return cloneFullComment(FC, D); | 
| Fariborz Jahanian | 40abf34 | 2013-01-25 22:48:32 +0000 | [diff] [blame] | 486 | } | 
| Fariborz Jahanian | e970c1b | 2013-04-26 20:55:38 +0000 | [diff] [blame] | 487 | else if (const ObjCInterfaceDecl *IC = dyn_cast<ObjCInterfaceDecl>(D)) { | 
|  | 488 | while (IC->getSuperClass()) { | 
|  | 489 | IC = IC->getSuperClass(); | 
|  | 490 | if (comments::FullComment *FC = getCommentForDecl(IC, PP)) | 
|  | 491 | return cloneFullComment(FC, D); | 
|  | 492 | } | 
|  | 493 | } | 
|  | 494 | else if (const ObjCCategoryDecl *CD = dyn_cast<ObjCCategoryDecl>(D)) { | 
|  | 495 | if (const ObjCInterfaceDecl *IC = CD->getClassInterface()) | 
|  | 496 | if (comments::FullComment *FC = getCommentForDecl(IC, PP)) | 
|  | 497 | return cloneFullComment(FC, D); | 
|  | 498 | } | 
|  | 499 | else if (const CXXRecordDecl *RD = dyn_cast<CXXRecordDecl>(D)) { | 
|  | 500 | if (!(RD = RD->getDefinition())) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 501 | return nullptr; | 
| Fariborz Jahanian | e970c1b | 2013-04-26 20:55:38 +0000 | [diff] [blame] | 502 | // Check non-virtual bases. | 
| Aaron Ballman | 574705e | 2014-03-13 15:41:46 +0000 | [diff] [blame] | 503 | for (const auto &I : RD->bases()) { | 
|  | 504 | if (I.isVirtual() || (I.getAccessSpecifier() != AS_public)) | 
| Fariborz Jahanian | e970c1b | 2013-04-26 20:55:38 +0000 | [diff] [blame] | 505 | continue; | 
| Aaron Ballman | 574705e | 2014-03-13 15:41:46 +0000 | [diff] [blame] | 506 | QualType Ty = I.getType(); | 
| Fariborz Jahanian | e970c1b | 2013-04-26 20:55:38 +0000 | [diff] [blame] | 507 | if (Ty.isNull()) | 
|  | 508 | continue; | 
|  | 509 | if (const CXXRecordDecl *NonVirtualBase = Ty->getAsCXXRecordDecl()) { | 
|  | 510 | if (!(NonVirtualBase= NonVirtualBase->getDefinition())) | 
|  | 511 | continue; | 
|  | 512 |  | 
|  | 513 | if (comments::FullComment *FC = getCommentForDecl((NonVirtualBase), PP)) | 
|  | 514 | return cloneFullComment(FC, D); | 
|  | 515 | } | 
|  | 516 | } | 
|  | 517 | // Check virtual bases. | 
| Aaron Ballman | 445a939 | 2014-03-13 16:15:17 +0000 | [diff] [blame] | 518 | for (const auto &I : RD->vbases()) { | 
|  | 519 | if (I.getAccessSpecifier() != AS_public) | 
| Fariborz Jahanian | 5a2e4a2 | 2013-04-26 23:34:36 +0000 | [diff] [blame] | 520 | continue; | 
| Aaron Ballman | 445a939 | 2014-03-13 16:15:17 +0000 | [diff] [blame] | 521 | QualType Ty = I.getType(); | 
| Fariborz Jahanian | e970c1b | 2013-04-26 20:55:38 +0000 | [diff] [blame] | 522 | if (Ty.isNull()) | 
|  | 523 | continue; | 
|  | 524 | if (const CXXRecordDecl *VirtualBase = Ty->getAsCXXRecordDecl()) { | 
|  | 525 | if (!(VirtualBase= VirtualBase->getDefinition())) | 
|  | 526 | continue; | 
|  | 527 | if (comments::FullComment *FC = getCommentForDecl((VirtualBase), PP)) | 
|  | 528 | return cloneFullComment(FC, D); | 
|  | 529 | } | 
|  | 530 | } | 
|  | 531 | } | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 532 | return nullptr; | 
| Fariborz Jahanian | 1c883b9 | 2012-10-10 18:34:52 +0000 | [diff] [blame] | 533 | } | 
|  | 534 |  | 
| Dmitri Gribenko | bfda9f7 | 2012-08-22 18:12:19 +0000 | [diff] [blame] | 535 | // If the RawComment was attached to other redeclaration of this Decl, we | 
|  | 536 | // should parse the comment in context of that other Decl.  This is important | 
|  | 537 | // because comments can contain references to parameter names which can be | 
|  | 538 | // different across redeclarations. | 
| Dmitri Gribenko | 4ae66a3 | 2012-08-16 18:19:43 +0000 | [diff] [blame] | 539 | if (D != OriginalDecl) | 
| Dmitri Gribenko | 6743e04 | 2012-09-29 11:40:46 +0000 | [diff] [blame] | 540 | return getCommentForDecl(OriginalDecl, PP); | 
| Dmitri Gribenko | 4ae66a3 | 2012-08-16 18:19:43 +0000 | [diff] [blame] | 541 |  | 
| Dmitri Gribenko | 6743e04 | 2012-09-29 11:40:46 +0000 | [diff] [blame] | 542 | comments::FullComment *FC = RC->parse(*this, PP, D); | 
| Dmitri Gribenko | b261088 | 2012-08-14 17:17:18 +0000 | [diff] [blame] | 543 | ParsedComments[Canonical] = FC; | 
|  | 544 | return FC; | 
| Dmitri Gribenko | ec92531 | 2012-07-06 00:28:32 +0000 | [diff] [blame] | 545 | } | 
|  | 546 |  | 
| Douglas Gregor | 7dbfb46 | 2010-06-16 21:09:37 +0000 | [diff] [blame] | 547 | void | 
|  | 548 | ASTContext::CanonicalTemplateTemplateParm::Profile(llvm::FoldingSetNodeID &ID, | 
|  | 549 | TemplateTemplateParmDecl *Parm) { | 
|  | 550 | ID.AddInteger(Parm->getDepth()); | 
|  | 551 | ID.AddInteger(Parm->getPosition()); | 
| Douglas Gregor | f550077 | 2011-01-05 15:48:55 +0000 | [diff] [blame] | 552 | ID.AddBoolean(Parm->isParameterPack()); | 
| Douglas Gregor | 7dbfb46 | 2010-06-16 21:09:37 +0000 | [diff] [blame] | 553 |  | 
|  | 554 | TemplateParameterList *Params = Parm->getTemplateParameters(); | 
|  | 555 | ID.AddInteger(Params->size()); | 
|  | 556 | for (TemplateParameterList::const_iterator P = Params->begin(), | 
|  | 557 | PEnd = Params->end(); | 
|  | 558 | P != PEnd; ++P) { | 
|  | 559 | if (TemplateTypeParmDecl *TTP = dyn_cast<TemplateTypeParmDecl>(*P)) { | 
|  | 560 | ID.AddInteger(0); | 
|  | 561 | ID.AddBoolean(TTP->isParameterPack()); | 
|  | 562 | continue; | 
|  | 563 | } | 
|  | 564 |  | 
|  | 565 | if (NonTypeTemplateParmDecl *NTTP = dyn_cast<NonTypeTemplateParmDecl>(*P)) { | 
|  | 566 | ID.AddInteger(1); | 
| Douglas Gregor | f550077 | 2011-01-05 15:48:55 +0000 | [diff] [blame] | 567 | ID.AddBoolean(NTTP->isParameterPack()); | 
| Eli Friedman | 205a429 | 2012-03-07 01:09:33 +0000 | [diff] [blame] | 568 | ID.AddPointer(NTTP->getType().getCanonicalType().getAsOpaquePtr()); | 
| Douglas Gregor | 0231d8d | 2011-01-19 20:10:05 +0000 | [diff] [blame] | 569 | if (NTTP->isExpandedParameterPack()) { | 
|  | 570 | ID.AddBoolean(true); | 
|  | 571 | ID.AddInteger(NTTP->getNumExpansionTypes()); | 
| Eli Friedman | 205a429 | 2012-03-07 01:09:33 +0000 | [diff] [blame] | 572 | for (unsigned I = 0, N = NTTP->getNumExpansionTypes(); I != N; ++I) { | 
|  | 573 | QualType T = NTTP->getExpansionType(I); | 
|  | 574 | ID.AddPointer(T.getCanonicalType().getAsOpaquePtr()); | 
|  | 575 | } | 
| Douglas Gregor | 0231d8d | 2011-01-19 20:10:05 +0000 | [diff] [blame] | 576 | } else | 
|  | 577 | ID.AddBoolean(false); | 
| Douglas Gregor | 7dbfb46 | 2010-06-16 21:09:37 +0000 | [diff] [blame] | 578 | continue; | 
|  | 579 | } | 
|  | 580 |  | 
|  | 581 | TemplateTemplateParmDecl *TTP = cast<TemplateTemplateParmDecl>(*P); | 
|  | 582 | ID.AddInteger(2); | 
|  | 583 | Profile(ID, TTP); | 
|  | 584 | } | 
|  | 585 | } | 
|  | 586 |  | 
|  | 587 | TemplateTemplateParmDecl * | 
|  | 588 | ASTContext::getCanonicalTemplateTemplateParmDecl( | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 589 | TemplateTemplateParmDecl *TTP) const { | 
| Douglas Gregor | 7dbfb46 | 2010-06-16 21:09:37 +0000 | [diff] [blame] | 590 | // Check if we already have a canonical template template parameter. | 
|  | 591 | llvm::FoldingSetNodeID ID; | 
|  | 592 | CanonicalTemplateTemplateParm::Profile(ID, TTP); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 593 | void *InsertPos = nullptr; | 
| Douglas Gregor | 7dbfb46 | 2010-06-16 21:09:37 +0000 | [diff] [blame] | 594 | CanonicalTemplateTemplateParm *Canonical | 
|  | 595 | = CanonTemplateTemplateParms.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 596 | if (Canonical) | 
|  | 597 | return Canonical->getParam(); | 
|  | 598 |  | 
|  | 599 | // Build a canonical template parameter list. | 
|  | 600 | TemplateParameterList *Params = TTP->getTemplateParameters(); | 
| Chris Lattner | 0e62c1c | 2011-07-23 10:55:15 +0000 | [diff] [blame] | 601 | SmallVector<NamedDecl *, 4> CanonParams; | 
| Douglas Gregor | 7dbfb46 | 2010-06-16 21:09:37 +0000 | [diff] [blame] | 602 | CanonParams.reserve(Params->size()); | 
|  | 603 | for (TemplateParameterList::const_iterator P = Params->begin(), | 
|  | 604 | PEnd = Params->end(); | 
|  | 605 | P != PEnd; ++P) { | 
|  | 606 | if (TemplateTypeParmDecl *TTP = dyn_cast<TemplateTypeParmDecl>(*P)) | 
|  | 607 | CanonParams.push_back( | 
|  | 608 | TemplateTypeParmDecl::Create(*this, getTranslationUnitDecl(), | 
| Abramo Bagnara | b3185b0 | 2011-03-06 15:48:19 +0000 | [diff] [blame] | 609 | SourceLocation(), | 
|  | 610 | SourceLocation(), | 
|  | 611 | TTP->getDepth(), | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 612 | TTP->getIndex(), nullptr, false, | 
| Douglas Gregor | 7dbfb46 | 2010-06-16 21:09:37 +0000 | [diff] [blame] | 613 | TTP->isParameterPack())); | 
|  | 614 | else if (NonTypeTemplateParmDecl *NTTP | 
| Douglas Gregor | 0231d8d | 2011-01-19 20:10:05 +0000 | [diff] [blame] | 615 | = dyn_cast<NonTypeTemplateParmDecl>(*P)) { | 
|  | 616 | QualType T = getCanonicalType(NTTP->getType()); | 
|  | 617 | TypeSourceInfo *TInfo = getTrivialTypeSourceInfo(T); | 
|  | 618 | NonTypeTemplateParmDecl *Param; | 
|  | 619 | if (NTTP->isExpandedParameterPack()) { | 
| Chris Lattner | 0e62c1c | 2011-07-23 10:55:15 +0000 | [diff] [blame] | 620 | SmallVector<QualType, 2> ExpandedTypes; | 
|  | 621 | SmallVector<TypeSourceInfo *, 2> ExpandedTInfos; | 
| Douglas Gregor | 0231d8d | 2011-01-19 20:10:05 +0000 | [diff] [blame] | 622 | for (unsigned I = 0, N = NTTP->getNumExpansionTypes(); I != N; ++I) { | 
|  | 623 | ExpandedTypes.push_back(getCanonicalType(NTTP->getExpansionType(I))); | 
|  | 624 | ExpandedTInfos.push_back( | 
|  | 625 | getTrivialTypeSourceInfo(ExpandedTypes.back())); | 
|  | 626 | } | 
|  | 627 |  | 
|  | 628 | Param = NonTypeTemplateParmDecl::Create(*this, getTranslationUnitDecl(), | 
| Abramo Bagnara | dff1930 | 2011-03-08 08:55:46 +0000 | [diff] [blame] | 629 | SourceLocation(), | 
|  | 630 | SourceLocation(), | 
| Douglas Gregor | 0231d8d | 2011-01-19 20:10:05 +0000 | [diff] [blame] | 631 | NTTP->getDepth(), | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 632 | NTTP->getPosition(), nullptr, | 
| Douglas Gregor | 0231d8d | 2011-01-19 20:10:05 +0000 | [diff] [blame] | 633 | T, | 
|  | 634 | TInfo, | 
| David Majnemer | dfecf1a | 2016-07-06 04:19:16 +0000 | [diff] [blame] | 635 | ExpandedTypes, | 
|  | 636 | ExpandedTInfos); | 
| Douglas Gregor | 0231d8d | 2011-01-19 20:10:05 +0000 | [diff] [blame] | 637 | } else { | 
|  | 638 | Param = NonTypeTemplateParmDecl::Create(*this, getTranslationUnitDecl(), | 
| Abramo Bagnara | dff1930 | 2011-03-08 08:55:46 +0000 | [diff] [blame] | 639 | SourceLocation(), | 
|  | 640 | SourceLocation(), | 
| Douglas Gregor | 0231d8d | 2011-01-19 20:10:05 +0000 | [diff] [blame] | 641 | NTTP->getDepth(), | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 642 | NTTP->getPosition(), nullptr, | 
| Douglas Gregor | 0231d8d | 2011-01-19 20:10:05 +0000 | [diff] [blame] | 643 | T, | 
|  | 644 | NTTP->isParameterPack(), | 
|  | 645 | TInfo); | 
|  | 646 | } | 
|  | 647 | CanonParams.push_back(Param); | 
|  | 648 |  | 
|  | 649 | } else | 
| Douglas Gregor | 7dbfb46 | 2010-06-16 21:09:37 +0000 | [diff] [blame] | 650 | CanonParams.push_back(getCanonicalTemplateTemplateParmDecl( | 
|  | 651 | cast<TemplateTemplateParmDecl>(*P))); | 
|  | 652 | } | 
|  | 653 |  | 
| Hubert Tong | e4a0c0e | 2016-07-30 22:33:34 +0000 | [diff] [blame] | 654 | assert(!TTP->getRequiresClause() && | 
|  | 655 | "Unexpected requires-clause on template template-parameter"); | 
|  | 656 | LLVM_CONSTEXPR Expr *const CanonRequiresClause = nullptr; | 
|  | 657 |  | 
| Douglas Gregor | 7dbfb46 | 2010-06-16 21:09:37 +0000 | [diff] [blame] | 658 | TemplateTemplateParmDecl *CanonTTP | 
|  | 659 | = TemplateTemplateParmDecl::Create(*this, getTranslationUnitDecl(), | 
|  | 660 | SourceLocation(), TTP->getDepth(), | 
| Douglas Gregor | f550077 | 2011-01-05 15:48:55 +0000 | [diff] [blame] | 661 | TTP->getPosition(), | 
|  | 662 | TTP->isParameterPack(), | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 663 | nullptr, | 
| Douglas Gregor | 7dbfb46 | 2010-06-16 21:09:37 +0000 | [diff] [blame] | 664 | TemplateParameterList::Create(*this, SourceLocation(), | 
|  | 665 | SourceLocation(), | 
| David Majnemer | 902f8c6 | 2015-12-27 07:16:27 +0000 | [diff] [blame] | 666 | CanonParams, | 
| Hubert Tong | e4a0c0e | 2016-07-30 22:33:34 +0000 | [diff] [blame] | 667 | SourceLocation(), | 
|  | 668 | CanonRequiresClause)); | 
| Douglas Gregor | 7dbfb46 | 2010-06-16 21:09:37 +0000 | [diff] [blame] | 669 |  | 
|  | 670 | // Get the new insert position for the node we care about. | 
|  | 671 | Canonical = CanonTemplateTemplateParms.FindNodeOrInsertPos(ID, InsertPos); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 672 | assert(!Canonical && "Shouldn't be in the map!"); | 
| Douglas Gregor | 7dbfb46 | 2010-06-16 21:09:37 +0000 | [diff] [blame] | 673 | (void)Canonical; | 
|  | 674 |  | 
|  | 675 | // Create the canonical template template parameter entry. | 
|  | 676 | Canonical = new (*this) CanonicalTemplateTemplateParm(CanonTTP); | 
|  | 677 | CanonTemplateTemplateParms.InsertNode(Canonical, InsertPos); | 
|  | 678 | return CanonTTP; | 
|  | 679 | } | 
|  | 680 |  | 
| Charles Davis | 53c59df | 2010-08-16 03:33:14 +0000 | [diff] [blame] | 681 | CXXABI *ASTContext::createCXXABI(const TargetInfo &T) { | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 682 | if (!LangOpts.CPlusPlus) return nullptr; | 
| John McCall | 8635341 | 2010-08-21 22:46:04 +0000 | [diff] [blame] | 683 |  | 
| John McCall | 359b885 | 2013-01-25 22:30:49 +0000 | [diff] [blame] | 684 | switch (T.getCXXABI().getKind()) { | 
| Joerg Sonnenberger | daa13aa | 2014-05-13 11:20:16 +0000 | [diff] [blame] | 685 | case TargetCXXABI::GenericARM: // Same as Itanium at this level | 
| John McCall | 359b885 | 2013-01-25 22:30:49 +0000 | [diff] [blame] | 686 | case TargetCXXABI::iOS: | 
| Tim Northover | a2ee433 | 2014-03-29 15:09:45 +0000 | [diff] [blame] | 687 | case TargetCXXABI::iOS64: | 
| Tim Northover | 756447a | 2015-10-30 16:30:36 +0000 | [diff] [blame] | 688 | case TargetCXXABI::WatchOS: | 
| Joerg Sonnenberger | daa13aa | 2014-05-13 11:20:16 +0000 | [diff] [blame] | 689 | case TargetCXXABI::GenericAArch64: | 
| Zoran Jovanovic | 26a1216 | 2015-02-18 15:21:35 +0000 | [diff] [blame] | 690 | case TargetCXXABI::GenericMIPS: | 
| John McCall | 359b885 | 2013-01-25 22:30:49 +0000 | [diff] [blame] | 691 | case TargetCXXABI::GenericItanium: | 
| Dan Gohman | c285307 | 2015-09-03 22:51:53 +0000 | [diff] [blame] | 692 | case TargetCXXABI::WebAssembly: | 
| Charles Davis | 53c59df | 2010-08-16 03:33:14 +0000 | [diff] [blame] | 693 | return CreateItaniumCXXABI(*this); | 
| John McCall | 359b885 | 2013-01-25 22:30:49 +0000 | [diff] [blame] | 694 | case TargetCXXABI::Microsoft: | 
| Charles Davis | 6bcb07a | 2010-08-19 02:18:14 +0000 | [diff] [blame] | 695 | return CreateMicrosoftCXXABI(*this); | 
|  | 696 | } | 
| David Blaikie | 8a40f70 | 2012-01-17 06:56:22 +0000 | [diff] [blame] | 697 | llvm_unreachable("Invalid CXXABI type!"); | 
| Charles Davis | 53c59df | 2010-08-16 03:33:14 +0000 | [diff] [blame] | 698 | } | 
|  | 699 |  | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 700 | static const LangAS::Map *getAddressSpaceMap(const TargetInfo &T, | 
| Peter Collingbourne | 599cb8e | 2011-03-18 22:38:29 +0000 | [diff] [blame] | 701 | const LangOptions &LOpts) { | 
|  | 702 | if (LOpts.FakeAddressSpaceMap) { | 
|  | 703 | // The fake address space map must have a distinct entry for each | 
|  | 704 | // language-specific address space. | 
|  | 705 | static const unsigned FakeAddrSpaceMap[] = { | 
|  | 706 | 1, // opencl_global | 
|  | 707 | 2, // opencl_local | 
| Peter Collingbourne | f44bdf9 | 2012-05-20 21:08:35 +0000 | [diff] [blame] | 708 | 3, // opencl_constant | 
| Anastasia Stulova | 2c8dcfb | 2014-11-26 14:10:06 +0000 | [diff] [blame] | 709 | 4, // opencl_generic | 
|  | 710 | 5, // cuda_device | 
|  | 711 | 6, // cuda_constant | 
|  | 712 | 7  // cuda_shared | 
| Peter Collingbourne | 599cb8e | 2011-03-18 22:38:29 +0000 | [diff] [blame] | 713 | }; | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 714 | return &FakeAddrSpaceMap; | 
| Peter Collingbourne | 599cb8e | 2011-03-18 22:38:29 +0000 | [diff] [blame] | 715 | } else { | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 716 | return &T.getAddressSpaceMap(); | 
| Peter Collingbourne | 599cb8e | 2011-03-18 22:38:29 +0000 | [diff] [blame] | 717 | } | 
|  | 718 | } | 
|  | 719 |  | 
| David Tweed | 31d09b0 | 2013-09-13 12:04:22 +0000 | [diff] [blame] | 720 | static bool isAddrSpaceMapManglingEnabled(const TargetInfo &TI, | 
|  | 721 | const LangOptions &LangOpts) { | 
|  | 722 | switch (LangOpts.getAddressSpaceMapMangling()) { | 
| David Tweed | 31d09b0 | 2013-09-13 12:04:22 +0000 | [diff] [blame] | 723 | case LangOptions::ASMM_Target: | 
|  | 724 | return TI.useAddressSpaceMapMangling(); | 
|  | 725 | case LangOptions::ASMM_On: | 
|  | 726 | return true; | 
|  | 727 | case LangOptions::ASMM_Off: | 
|  | 728 | return false; | 
|  | 729 | } | 
| NAKAMURA Takumi | 5c81ca4 | 2013-09-13 17:12:09 +0000 | [diff] [blame] | 730 | llvm_unreachable("getAddressSpaceMapMangling() doesn't cover anything."); | 
| David Tweed | 31d09b0 | 2013-09-13 12:04:22 +0000 | [diff] [blame] | 731 | } | 
|  | 732 |  | 
| Alexey Samsonov | 0b15e34 | 2014-10-15 22:17:27 +0000 | [diff] [blame] | 733 | ASTContext::ASTContext(LangOptions &LOpts, SourceManager &SM, | 
| Daniel Dunbar | 221fa94 | 2008-08-11 04:54:23 +0000 | [diff] [blame] | 734 | IdentifierTable &idents, SelectorTable &sels, | 
| Alp Toker | 0804343 | 2014-05-03 03:46:04 +0000 | [diff] [blame] | 735 | Builtin::Context &builtins) | 
| Alexey Samsonov | 0b15e34 | 2014-10-15 22:17:27 +0000 | [diff] [blame] | 736 | : FunctionProtoTypes(this_()), TemplateSpecializationTypes(this_()), | 
|  | 737 | DependentTemplateSpecializationTypes(this_()), | 
|  | 738 | SubstTemplateTemplateParmPacks(this_()), | 
|  | 739 | GlobalNestedNameSpecifier(nullptr), Int128Decl(nullptr), | 
| Nemanja Ivanovic | bb1ea2d | 2016-05-09 08:52:33 +0000 | [diff] [blame] | 740 | UInt128Decl(nullptr), BuiltinVaListDecl(nullptr), | 
|  | 741 | BuiltinMSVaListDecl(nullptr), ObjCIdDecl(nullptr), ObjCSelDecl(nullptr), | 
|  | 742 | ObjCClassDecl(nullptr), ObjCProtocolClassDecl(nullptr), BOOLDecl(nullptr), | 
| Ben Langmuir | f541674 | 2016-02-04 00:55:24 +0000 | [diff] [blame] | 743 | CFConstantStringTagDecl(nullptr), CFConstantStringTypeDecl(nullptr), | 
|  | 744 | ObjCInstanceTypeDecl(nullptr), FILEDecl(nullptr), jmp_bufDecl(nullptr), | 
|  | 745 | sigjmp_bufDecl(nullptr), ucontext_tDecl(nullptr), | 
|  | 746 | BlockDescriptorType(nullptr), BlockDescriptorExtendedType(nullptr), | 
|  | 747 | cudaConfigureCallDecl(nullptr), FirstLocalImport(), LastLocalImport(), | 
| Eric Fiselier | 6ad6855 | 2016-07-01 01:24:09 +0000 | [diff] [blame] | 748 | ExternCContext(nullptr), MakeIntegerSeqDecl(nullptr), | 
|  | 749 | TypePackElementDecl(nullptr), SourceMgr(SM), LangOpts(LOpts), | 
| Alexey Samsonov | a511cdd | 2015-02-04 17:40:08 +0000 | [diff] [blame] | 750 | SanitizerBL(new SanitizerBlacklist(LangOpts.SanitizerBlacklistFiles, SM)), | 
| Artem Belevich | b5bc923 | 2015-09-22 17:23:22 +0000 | [diff] [blame] | 751 | AddrSpaceMap(nullptr), Target(nullptr), AuxTarget(nullptr), | 
|  | 752 | PrintingPolicy(LOpts), Idents(idents), Selectors(sels), | 
|  | 753 | BuiltinInfo(builtins), DeclarationNames(*this), ExternalSource(nullptr), | 
|  | 754 | Listener(nullptr), Comments(SM), CommentsLoaded(false), | 
| Alexey Samsonov | 0b15e34 | 2014-10-15 22:17:27 +0000 | [diff] [blame] | 755 | CommentCommandTraits(BumpAlloc, LOpts.CommentOpts), LastSDM(nullptr, 0) { | 
| Daniel Dunbar | 221fa94 | 2008-08-11 04:54:23 +0000 | [diff] [blame] | 756 | TUDecl = TranslationUnitDecl::Create(*this); | 
|  | 757 | } | 
|  | 758 |  | 
| Chris Lattner | d5973eb | 2006-11-12 00:53:46 +0000 | [diff] [blame] | 759 | ASTContext::~ASTContext() { | 
| Manuel Klimek | 95403e6 | 2014-05-21 13:28:59 +0000 | [diff] [blame] | 760 | ReleaseParentMapEntries(); | 
|  | 761 |  | 
| Ted Kremenek | da4e0d3 | 2010-02-11 07:12:28 +0000 | [diff] [blame] | 762 | // Release the DenseMaps associated with DeclContext objects. | 
|  | 763 | // FIXME: Is this the ideal solution? | 
|  | 764 | ReleaseDeclContextMaps(); | 
| Douglas Gregor | 832940b | 2010-03-02 23:58:15 +0000 | [diff] [blame] | 765 |  | 
| Manuel Klimek | a732899 | 2013-06-03 13:51:33 +0000 | [diff] [blame] | 766 | // Call all of the deallocation functions on all of their targets. | 
| Chandler Carruth | ff5a01a | 2015-12-30 03:00:23 +0000 | [diff] [blame] | 767 | for (auto &Pair : Deallocations) | 
|  | 768 | (Pair.first)(Pair.second); | 
| Manuel Klimek | a732899 | 2013-06-03 13:51:33 +0000 | [diff] [blame] | 769 |  | 
| Ted Kremenek | 076baeb | 2010-06-08 23:00:58 +0000 | [diff] [blame] | 770 | // ASTRecordLayout objects in ASTRecordLayouts must always be destroyed | 
| Douglas Gregor | 5b11d49 | 2010-07-25 17:53:33 +0000 | [diff] [blame] | 771 | // because they can contain DenseMaps. | 
|  | 772 | for (llvm::DenseMap<const ObjCContainerDecl*, | 
|  | 773 | const ASTRecordLayout*>::iterator | 
|  | 774 | I = ObjCLayouts.begin(), E = ObjCLayouts.end(); I != E; ) | 
|  | 775 | // Increment in loop to prevent using deallocated memory. | 
|  | 776 | if (ASTRecordLayout *R = const_cast<ASTRecordLayout*>((I++)->second)) | 
|  | 777 | R->Destroy(*this); | 
|  | 778 |  | 
| Ted Kremenek | 076baeb | 2010-06-08 23:00:58 +0000 | [diff] [blame] | 779 | for (llvm::DenseMap<const RecordDecl*, const ASTRecordLayout*>::iterator | 
|  | 780 | I = ASTRecordLayouts.begin(), E = ASTRecordLayouts.end(); I != E; ) { | 
|  | 781 | // Increment in loop to prevent using deallocated memory. | 
|  | 782 | if (ASTRecordLayout *R = const_cast<ASTRecordLayout*>((I++)->second)) | 
|  | 783 | R->Destroy(*this); | 
|  | 784 | } | 
| Douglas Gregor | 561eceb | 2010-08-30 16:49:28 +0000 | [diff] [blame] | 785 |  | 
|  | 786 | for (llvm::DenseMap<const Decl*, AttrVec*>::iterator A = DeclAttrs.begin(), | 
|  | 787 | AEnd = DeclAttrs.end(); | 
|  | 788 | A != AEnd; ++A) | 
|  | 789 | A->second->~AttrVec(); | 
| Reid Kleckner | d8110b6 | 2013-09-10 20:14:30 +0000 | [diff] [blame] | 790 |  | 
| David Majnemer | e694f3e | 2015-08-14 14:43:50 +0000 | [diff] [blame] | 791 | for (std::pair<const MaterializeTemporaryExpr *, APValue *> &MTVPair : | 
|  | 792 | MaterializedTemporaryValues) | 
|  | 793 | MTVPair.second->~APValue(); | 
|  | 794 |  | 
| Richard Smith | 423f46f | 2016-07-20 21:38:26 +0000 | [diff] [blame] | 795 | for (const auto &Value : ModuleInitializers) | 
|  | 796 | Value.second->~PerModuleInitializers(); | 
|  | 797 |  | 
| Reid Kleckner | 588c937 | 2014-02-19 23:44:52 +0000 | [diff] [blame] | 798 | llvm::DeleteContainerSeconds(MangleNumberingContexts); | 
| Douglas Gregor | 561eceb | 2010-08-30 16:49:28 +0000 | [diff] [blame] | 799 | } | 
| Douglas Gregor | f21eb49 | 2009-03-26 23:50:42 +0000 | [diff] [blame] | 800 |  | 
| Manuel Klimek | 95403e6 | 2014-05-21 13:28:59 +0000 | [diff] [blame] | 801 | void ASTContext::ReleaseParentMapEntries() { | 
| Benjamin Kramer | 94355ae | 2015-10-23 09:04:55 +0000 | [diff] [blame] | 802 | if (!PointerParents) return; | 
|  | 803 | for (const auto &Entry : *PointerParents) { | 
|  | 804 | if (Entry.second.is<ast_type_traits::DynTypedNode *>()) { | 
|  | 805 | delete Entry.second.get<ast_type_traits::DynTypedNode *>(); | 
|  | 806 | } else if (Entry.second.is<ParentVector *>()) { | 
|  | 807 | delete Entry.second.get<ParentVector *>(); | 
|  | 808 | } | 
|  | 809 | } | 
|  | 810 | for (const auto &Entry : *OtherParents) { | 
| Manuel Klimek | 95403e6 | 2014-05-21 13:28:59 +0000 | [diff] [blame] | 811 | if (Entry.second.is<ast_type_traits::DynTypedNode *>()) { | 
|  | 812 | delete Entry.second.get<ast_type_traits::DynTypedNode *>(); | 
| Benjamin Kramer | 8e4a176 | 2015-10-22 11:21:40 +0000 | [diff] [blame] | 813 | } else if (Entry.second.is<ParentVector *>()) { | 
| Manuel Klimek | 95403e6 | 2014-05-21 13:28:59 +0000 | [diff] [blame] | 814 | delete Entry.second.get<ParentVector *>(); | 
|  | 815 | } | 
|  | 816 | } | 
|  | 817 | } | 
|  | 818 |  | 
| Douglas Gregor | 1a80933 | 2010-05-23 18:26:36 +0000 | [diff] [blame] | 819 | void ASTContext::AddDeallocation(void (*Callback)(void*), void *Data) { | 
| Chandler Carruth | ff5a01a | 2015-12-30 03:00:23 +0000 | [diff] [blame] | 820 | Deallocations.push_back({Callback, Data}); | 
| Douglas Gregor | 1a80933 | 2010-05-23 18:26:36 +0000 | [diff] [blame] | 821 | } | 
|  | 822 |  | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 823 | void | 
| Argyrios Kyrtzidis | 1b7ed91 | 2014-02-27 04:11:59 +0000 | [diff] [blame] | 824 | ASTContext::setExternalSource(IntrusiveRefCntPtr<ExternalASTSource> Source) { | 
| Benjamin Kramer | d6da1a0 | 2016-06-12 20:05:23 +0000 | [diff] [blame] | 825 | ExternalSource = std::move(Source); | 
| Douglas Gregor | ef84c4b | 2009-04-09 22:27:44 +0000 | [diff] [blame] | 826 | } | 
|  | 827 |  | 
| Chris Lattner | 4eb445d | 2007-01-26 01:27:23 +0000 | [diff] [blame] | 828 | void ASTContext::PrintStats() const { | 
| Chandler Carruth | 3c147a7 | 2011-07-04 05:32:14 +0000 | [diff] [blame] | 829 | llvm::errs() << "\n*** AST Context Stats:\n"; | 
|  | 830 | llvm::errs() << "  " << Types.size() << " types total.\n"; | 
| Sebastian Redl | 0f8b23f | 2009-03-16 23:22:08 +0000 | [diff] [blame] | 831 |  | 
| Douglas Gregor | a30d046 | 2009-05-26 14:40:08 +0000 | [diff] [blame] | 832 | unsigned counts[] = { | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 833 | #define TYPE(Name, Parent) 0, | 
| Douglas Gregor | a30d046 | 2009-05-26 14:40:08 +0000 | [diff] [blame] | 834 | #define ABSTRACT_TYPE(Name, Parent) | 
|  | 835 | #include "clang/AST/TypeNodes.def" | 
|  | 836 | 0 // Extra | 
|  | 837 | }; | 
| Douglas Gregor | b1fe2c9 | 2009-04-07 17:20:56 +0000 | [diff] [blame] | 838 |  | 
| Chris Lattner | 4eb445d | 2007-01-26 01:27:23 +0000 | [diff] [blame] | 839 | for (unsigned i = 0, e = Types.size(); i != e; ++i) { | 
|  | 840 | Type *T = Types[i]; | 
| Douglas Gregor | a30d046 | 2009-05-26 14:40:08 +0000 | [diff] [blame] | 841 | counts[(unsigned)T->getTypeClass()]++; | 
| Chris Lattner | 4eb445d | 2007-01-26 01:27:23 +0000 | [diff] [blame] | 842 | } | 
|  | 843 |  | 
| Douglas Gregor | a30d046 | 2009-05-26 14:40:08 +0000 | [diff] [blame] | 844 | unsigned Idx = 0; | 
|  | 845 | unsigned TotalBytes = 0; | 
|  | 846 | #define TYPE(Name, Parent)                                              \ | 
|  | 847 | if (counts[Idx])                                                      \ | 
| Chandler Carruth | 3c147a7 | 2011-07-04 05:32:14 +0000 | [diff] [blame] | 848 | llvm::errs() << "    " << counts[Idx] << " " << #Name               \ | 
|  | 849 | << " types\n";                                         \ | 
| Douglas Gregor | a30d046 | 2009-05-26 14:40:08 +0000 | [diff] [blame] | 850 | TotalBytes += counts[Idx] * sizeof(Name##Type);                       \ | 
|  | 851 | ++Idx; | 
|  | 852 | #define ABSTRACT_TYPE(Name, Parent) | 
|  | 853 | #include "clang/AST/TypeNodes.def" | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 854 |  | 
| Chandler Carruth | 3c147a7 | 2011-07-04 05:32:14 +0000 | [diff] [blame] | 855 | llvm::errs() << "Total bytes = " << TotalBytes << "\n"; | 
|  | 856 |  | 
| Douglas Gregor | 7454c56 | 2010-07-02 20:37:36 +0000 | [diff] [blame] | 857 | // Implicit special member functions. | 
| Chandler Carruth | 3c147a7 | 2011-07-04 05:32:14 +0000 | [diff] [blame] | 858 | llvm::errs() << NumImplicitDefaultConstructorsDeclared << "/" | 
|  | 859 | << NumImplicitDefaultConstructors | 
|  | 860 | << " implicit default constructors created\n"; | 
|  | 861 | llvm::errs() << NumImplicitCopyConstructorsDeclared << "/" | 
|  | 862 | << NumImplicitCopyConstructors | 
|  | 863 | << " implicit copy constructors created\n"; | 
| David Blaikie | bbafb8a | 2012-03-11 07:00:24 +0000 | [diff] [blame] | 864 | if (getLangOpts().CPlusPlus) | 
| Chandler Carruth | 3c147a7 | 2011-07-04 05:32:14 +0000 | [diff] [blame] | 865 | llvm::errs() << NumImplicitMoveConstructorsDeclared << "/" | 
|  | 866 | << NumImplicitMoveConstructors | 
|  | 867 | << " implicit move constructors created\n"; | 
|  | 868 | llvm::errs() << NumImplicitCopyAssignmentOperatorsDeclared << "/" | 
|  | 869 | << NumImplicitCopyAssignmentOperators | 
|  | 870 | << " implicit copy assignment operators created\n"; | 
| David Blaikie | bbafb8a | 2012-03-11 07:00:24 +0000 | [diff] [blame] | 871 | if (getLangOpts().CPlusPlus) | 
| Chandler Carruth | 3c147a7 | 2011-07-04 05:32:14 +0000 | [diff] [blame] | 872 | llvm::errs() << NumImplicitMoveAssignmentOperatorsDeclared << "/" | 
|  | 873 | << NumImplicitMoveAssignmentOperators | 
|  | 874 | << " implicit move assignment operators created\n"; | 
|  | 875 | llvm::errs() << NumImplicitDestructorsDeclared << "/" | 
|  | 876 | << NumImplicitDestructors | 
|  | 877 | << " implicit destructors created\n"; | 
|  | 878 |  | 
| Argyrios Kyrtzidis | 1b7ed91 | 2014-02-27 04:11:59 +0000 | [diff] [blame] | 879 | if (ExternalSource) { | 
| Chandler Carruth | 3c147a7 | 2011-07-04 05:32:14 +0000 | [diff] [blame] | 880 | llvm::errs() << "\n"; | 
| Douglas Gregor | ef84c4b | 2009-04-09 22:27:44 +0000 | [diff] [blame] | 881 | ExternalSource->PrintStats(); | 
|  | 882 | } | 
| Chandler Carruth | 3c147a7 | 2011-07-04 05:32:14 +0000 | [diff] [blame] | 883 |  | 
| Douglas Gregor | 5b11d49 | 2010-07-25 17:53:33 +0000 | [diff] [blame] | 884 | BumpAlloc.PrintStats(); | 
| Chris Lattner | 4eb445d | 2007-01-26 01:27:23 +0000 | [diff] [blame] | 885 | } | 
|  | 886 |  | 
| Richard Smith | 4241314 | 2015-05-15 20:05:43 +0000 | [diff] [blame] | 887 | void ASTContext::mergeDefinitionIntoModule(NamedDecl *ND, Module *M, | 
|  | 888 | bool NotifyListeners) { | 
|  | 889 | if (NotifyListeners) | 
|  | 890 | if (auto *Listener = getASTMutationListener()) | 
|  | 891 | Listener->RedefinedHiddenDefinition(ND, M); | 
|  | 892 |  | 
|  | 893 | if (getLangOpts().ModulesLocalVisibility) | 
|  | 894 | MergedDefModules[ND].push_back(M); | 
|  | 895 | else | 
|  | 896 | ND->setHidden(false); | 
|  | 897 | } | 
|  | 898 |  | 
|  | 899 | void ASTContext::deduplicateMergedDefinitonsFor(NamedDecl *ND) { | 
|  | 900 | auto It = MergedDefModules.find(ND); | 
|  | 901 | if (It == MergedDefModules.end()) | 
|  | 902 | return; | 
|  | 903 |  | 
|  | 904 | auto &Merged = It->second; | 
|  | 905 | llvm::DenseSet<Module*> Found; | 
|  | 906 | for (Module *&M : Merged) | 
|  | 907 | if (!Found.insert(M).second) | 
|  | 908 | M = nullptr; | 
|  | 909 | Merged.erase(std::remove(Merged.begin(), Merged.end(), nullptr), Merged.end()); | 
|  | 910 | } | 
|  | 911 |  | 
| Richard Smith | dc1f042 | 2016-07-20 19:10:16 +0000 | [diff] [blame] | 912 | void ASTContext::PerModuleInitializers::resolve(ASTContext &Ctx) { | 
|  | 913 | if (LazyInitializers.empty()) | 
|  | 914 | return; | 
|  | 915 |  | 
|  | 916 | auto *Source = Ctx.getExternalSource(); | 
|  | 917 | assert(Source && "lazy initializers but no external source"); | 
|  | 918 |  | 
|  | 919 | auto LazyInits = std::move(LazyInitializers); | 
|  | 920 | LazyInitializers.clear(); | 
|  | 921 |  | 
|  | 922 | for (auto ID : LazyInits) | 
|  | 923 | Initializers.push_back(Source->GetExternalDecl(ID)); | 
|  | 924 |  | 
|  | 925 | assert(LazyInitializers.empty() && | 
|  | 926 | "GetExternalDecl for lazy module initializer added more inits"); | 
|  | 927 | } | 
|  | 928 |  | 
|  | 929 | void ASTContext::addModuleInitializer(Module *M, Decl *D) { | 
|  | 930 | // One special case: if we add a module initializer that imports another | 
|  | 931 | // module, and that module's only initializer is an ImportDecl, simplify. | 
|  | 932 | if (auto *ID = dyn_cast<ImportDecl>(D)) { | 
|  | 933 | auto It = ModuleInitializers.find(ID->getImportedModule()); | 
|  | 934 |  | 
|  | 935 | // Maybe the ImportDecl does nothing at all. (Common case.) | 
|  | 936 | if (It == ModuleInitializers.end()) | 
|  | 937 | return; | 
|  | 938 |  | 
|  | 939 | // Maybe the ImportDecl only imports another ImportDecl. | 
|  | 940 | auto &Imported = *It->second; | 
|  | 941 | if (Imported.Initializers.size() + Imported.LazyInitializers.size() == 1) { | 
|  | 942 | Imported.resolve(*this); | 
|  | 943 | auto *OnlyDecl = Imported.Initializers.front(); | 
|  | 944 | if (isa<ImportDecl>(OnlyDecl)) | 
|  | 945 | D = OnlyDecl; | 
|  | 946 | } | 
|  | 947 | } | 
|  | 948 |  | 
|  | 949 | auto *&Inits = ModuleInitializers[M]; | 
|  | 950 | if (!Inits) | 
|  | 951 | Inits = new (*this) PerModuleInitializers; | 
|  | 952 | Inits->Initializers.push_back(D); | 
|  | 953 | } | 
|  | 954 |  | 
|  | 955 | void ASTContext::addLazyModuleInitializers(Module *M, ArrayRef<uint32_t> IDs) { | 
|  | 956 | auto *&Inits = ModuleInitializers[M]; | 
|  | 957 | if (!Inits) | 
|  | 958 | Inits = new (*this) PerModuleInitializers; | 
|  | 959 | Inits->LazyInitializers.insert(Inits->LazyInitializers.end(), | 
|  | 960 | IDs.begin(), IDs.end()); | 
|  | 961 | } | 
|  | 962 |  | 
|  | 963 | ArrayRef<Decl*> ASTContext::getModuleInitializers(Module *M) { | 
|  | 964 | auto It = ModuleInitializers.find(M); | 
|  | 965 | if (It == ModuleInitializers.end()) | 
|  | 966 | return None; | 
|  | 967 |  | 
|  | 968 | auto *Inits = It->second; | 
|  | 969 | Inits->resolve(*this); | 
|  | 970 | return Inits->Initializers; | 
|  | 971 | } | 
|  | 972 |  | 
| Richard Smith | f19e127 | 2015-03-07 00:04:49 +0000 | [diff] [blame] | 973 | ExternCContextDecl *ASTContext::getExternCContextDecl() const { | 
|  | 974 | if (!ExternCContext) | 
|  | 975 | ExternCContext = ExternCContextDecl::Create(*this, getTranslationUnitDecl()); | 
|  | 976 |  | 
|  | 977 | return ExternCContext; | 
|  | 978 | } | 
|  | 979 |  | 
| David Majnemer | d9b1a4f | 2015-11-04 03:40:30 +0000 | [diff] [blame] | 980 | BuiltinTemplateDecl * | 
|  | 981 | ASTContext::buildBuiltinTemplateDecl(BuiltinTemplateKind BTK, | 
|  | 982 | const IdentifierInfo *II) const { | 
|  | 983 | auto *BuiltinTemplate = BuiltinTemplateDecl::Create(*this, TUDecl, II, BTK); | 
|  | 984 | BuiltinTemplate->setImplicit(); | 
|  | 985 | TUDecl->addDecl(BuiltinTemplate); | 
|  | 986 |  | 
|  | 987 | return BuiltinTemplate; | 
|  | 988 | } | 
|  | 989 |  | 
|  | 990 | BuiltinTemplateDecl * | 
|  | 991 | ASTContext::getMakeIntegerSeqDecl() const { | 
|  | 992 | if (!MakeIntegerSeqDecl) | 
|  | 993 | MakeIntegerSeqDecl = buildBuiltinTemplateDecl(BTK__make_integer_seq, | 
|  | 994 | getMakeIntegerSeqName()); | 
|  | 995 | return MakeIntegerSeqDecl; | 
|  | 996 | } | 
|  | 997 |  | 
| Eric Fiselier | 6ad6855 | 2016-07-01 01:24:09 +0000 | [diff] [blame] | 998 | BuiltinTemplateDecl * | 
|  | 999 | ASTContext::getTypePackElementDecl() const { | 
|  | 1000 | if (!TypePackElementDecl) | 
|  | 1001 | TypePackElementDecl = buildBuiltinTemplateDecl(BTK__type_pack_element, | 
|  | 1002 | getTypePackElementName()); | 
|  | 1003 | return TypePackElementDecl; | 
|  | 1004 | } | 
|  | 1005 |  | 
| Alp Toker | 2dea15b | 2013-12-17 01:22:38 +0000 | [diff] [blame] | 1006 | RecordDecl *ASTContext::buildImplicitRecord(StringRef Name, | 
|  | 1007 | RecordDecl::TagKind TK) const { | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 1008 | SourceLocation Loc; | 
|  | 1009 | RecordDecl *NewDecl; | 
| Alp Toker | 2dea15b | 2013-12-17 01:22:38 +0000 | [diff] [blame] | 1010 | if (getLangOpts().CPlusPlus) | 
|  | 1011 | NewDecl = CXXRecordDecl::Create(*this, TK, getTranslationUnitDecl(), Loc, | 
|  | 1012 | Loc, &Idents.get(Name)); | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 1013 | else | 
| Alp Toker | 2dea15b | 2013-12-17 01:22:38 +0000 | [diff] [blame] | 1014 | NewDecl = RecordDecl::Create(*this, TK, getTranslationUnitDecl(), Loc, Loc, | 
|  | 1015 | &Idents.get(Name)); | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 1016 | NewDecl->setImplicit(); | 
| David Majnemer | f863736 | 2015-01-15 08:41:25 +0000 | [diff] [blame] | 1017 | NewDecl->addAttr(TypeVisibilityAttr::CreateImplicit( | 
|  | 1018 | const_cast<ASTContext &>(*this), TypeVisibilityAttr::Default)); | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 1019 | return NewDecl; | 
|  | 1020 | } | 
|  | 1021 |  | 
|  | 1022 | TypedefDecl *ASTContext::buildImplicitTypedef(QualType T, | 
|  | 1023 | StringRef Name) const { | 
|  | 1024 | TypeSourceInfo *TInfo = getTrivialTypeSourceInfo(T); | 
|  | 1025 | TypedefDecl *NewDecl = TypedefDecl::Create( | 
|  | 1026 | const_cast<ASTContext &>(*this), getTranslationUnitDecl(), | 
|  | 1027 | SourceLocation(), SourceLocation(), &Idents.get(Name), TInfo); | 
|  | 1028 | NewDecl->setImplicit(); | 
|  | 1029 | return NewDecl; | 
|  | 1030 | } | 
|  | 1031 |  | 
| Douglas Gregor | 801c99d | 2011-08-12 06:49:56 +0000 | [diff] [blame] | 1032 | TypedefDecl *ASTContext::getInt128Decl() const { | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 1033 | if (!Int128Decl) | 
|  | 1034 | Int128Decl = buildImplicitTypedef(Int128Ty, "__int128_t"); | 
| Douglas Gregor | 801c99d | 2011-08-12 06:49:56 +0000 | [diff] [blame] | 1035 | return Int128Decl; | 
|  | 1036 | } | 
|  | 1037 |  | 
|  | 1038 | TypedefDecl *ASTContext::getUInt128Decl() const { | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 1039 | if (!UInt128Decl) | 
|  | 1040 | UInt128Decl = buildImplicitTypedef(UnsignedInt128Ty, "__uint128_t"); | 
| Douglas Gregor | 801c99d | 2011-08-12 06:49:56 +0000 | [diff] [blame] | 1041 | return UInt128Decl; | 
|  | 1042 | } | 
| Chris Lattner | 4eb445d | 2007-01-26 01:27:23 +0000 | [diff] [blame] | 1043 |  | 
| John McCall | 48f2d58 | 2009-10-23 23:03:21 +0000 | [diff] [blame] | 1044 | void ASTContext::InitBuiltinType(CanQualType &R, BuiltinType::Kind K) { | 
| John McCall | 90d1c2d | 2009-09-24 23:30:46 +0000 | [diff] [blame] | 1045 | BuiltinType *Ty = new (*this, TypeAlignment) BuiltinType(K); | 
| John McCall | 48f2d58 | 2009-10-23 23:03:21 +0000 | [diff] [blame] | 1046 | R = CanQualType::CreateUnsafe(QualType(Ty, 0)); | 
| John McCall | 90d1c2d | 2009-09-24 23:30:46 +0000 | [diff] [blame] | 1047 | Types.push_back(Ty); | 
| Chris Lattner | d5973eb | 2006-11-12 00:53:46 +0000 | [diff] [blame] | 1048 | } | 
|  | 1049 |  | 
| Artem Belevich | b5bc923 | 2015-09-22 17:23:22 +0000 | [diff] [blame] | 1050 | void ASTContext::InitBuiltinTypes(const TargetInfo &Target, | 
|  | 1051 | const TargetInfo *AuxTarget) { | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 1052 | assert((!this->Target || this->Target == &Target) && | 
|  | 1053 | "Incorrect target reinitialization"); | 
| Chris Lattner | 970e54e | 2006-11-12 00:37:36 +0000 | [diff] [blame] | 1054 | assert(VoidTy.isNull() && "Context reinitialized?"); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 1055 |  | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 1056 | this->Target = &Target; | 
| Artem Belevich | b5bc923 | 2015-09-22 17:23:22 +0000 | [diff] [blame] | 1057 | this->AuxTarget = AuxTarget; | 
|  | 1058 |  | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 1059 | ABI.reset(createCXXABI(Target)); | 
|  | 1060 | AddrSpaceMap = getAddressSpaceMap(Target, LangOpts); | 
| David Tweed | 31d09b0 | 2013-09-13 12:04:22 +0000 | [diff] [blame] | 1061 | AddrSpaceMapMangling = isAddrSpaceMapManglingEnabled(Target, LangOpts); | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 1062 |  | 
| Chris Lattner | 970e54e | 2006-11-12 00:37:36 +0000 | [diff] [blame] | 1063 | // C99 6.2.5p19. | 
| Chris Lattner | 726f97b | 2006-12-03 02:57:32 +0000 | [diff] [blame] | 1064 | InitBuiltinType(VoidTy,              BuiltinType::Void); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 1065 |  | 
| Chris Lattner | 970e54e | 2006-11-12 00:37:36 +0000 | [diff] [blame] | 1066 | // C99 6.2.5p2. | 
| Chris Lattner | 726f97b | 2006-12-03 02:57:32 +0000 | [diff] [blame] | 1067 | InitBuiltinType(BoolTy,              BuiltinType::Bool); | 
| Chris Lattner | 970e54e | 2006-11-12 00:37:36 +0000 | [diff] [blame] | 1068 | // C99 6.2.5p3. | 
| Eli Friedman | 9ffd4a9 | 2009-06-05 07:05:05 +0000 | [diff] [blame] | 1069 | if (LangOpts.CharIsSigned) | 
| Chris Lattner | b16f455 | 2007-06-03 07:25:34 +0000 | [diff] [blame] | 1070 | InitBuiltinType(CharTy,            BuiltinType::Char_S); | 
|  | 1071 | else | 
|  | 1072 | InitBuiltinType(CharTy,            BuiltinType::Char_U); | 
| Chris Lattner | 970e54e | 2006-11-12 00:37:36 +0000 | [diff] [blame] | 1073 | // C99 6.2.5p4. | 
| Chris Lattner | 726f97b | 2006-12-03 02:57:32 +0000 | [diff] [blame] | 1074 | InitBuiltinType(SignedCharTy,        BuiltinType::SChar); | 
|  | 1075 | InitBuiltinType(ShortTy,             BuiltinType::Short); | 
|  | 1076 | InitBuiltinType(IntTy,               BuiltinType::Int); | 
|  | 1077 | InitBuiltinType(LongTy,              BuiltinType::Long); | 
|  | 1078 | InitBuiltinType(LongLongTy,          BuiltinType::LongLong); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 1079 |  | 
| Chris Lattner | 970e54e | 2006-11-12 00:37:36 +0000 | [diff] [blame] | 1080 | // C99 6.2.5p6. | 
| Chris Lattner | 726f97b | 2006-12-03 02:57:32 +0000 | [diff] [blame] | 1081 | InitBuiltinType(UnsignedCharTy,      BuiltinType::UChar); | 
|  | 1082 | InitBuiltinType(UnsignedShortTy,     BuiltinType::UShort); | 
|  | 1083 | InitBuiltinType(UnsignedIntTy,       BuiltinType::UInt); | 
|  | 1084 | InitBuiltinType(UnsignedLongTy,      BuiltinType::ULong); | 
|  | 1085 | InitBuiltinType(UnsignedLongLongTy,  BuiltinType::ULongLong); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 1086 |  | 
| Chris Lattner | 970e54e | 2006-11-12 00:37:36 +0000 | [diff] [blame] | 1087 | // C99 6.2.5p10. | 
| Chris Lattner | 726f97b | 2006-12-03 02:57:32 +0000 | [diff] [blame] | 1088 | InitBuiltinType(FloatTy,             BuiltinType::Float); | 
|  | 1089 | InitBuiltinType(DoubleTy,            BuiltinType::Double); | 
|  | 1090 | InitBuiltinType(LongDoubleTy,        BuiltinType::LongDouble); | 
| Argyrios Kyrtzidis | 40e9e48 | 2008-08-09 16:51:54 +0000 | [diff] [blame] | 1091 |  | 
| Nemanja Ivanovic | bb1ea2d | 2016-05-09 08:52:33 +0000 | [diff] [blame] | 1092 | // GNU extension, __float128 for IEEE quadruple precision | 
|  | 1093 | InitBuiltinType(Float128Ty,          BuiltinType::Float128); | 
|  | 1094 |  | 
| Chris Lattner | f122cef | 2009-04-30 02:43:43 +0000 | [diff] [blame] | 1095 | // GNU extension, 128-bit integers. | 
|  | 1096 | InitBuiltinType(Int128Ty,            BuiltinType::Int128); | 
|  | 1097 | InitBuiltinType(UnsignedInt128Ty,    BuiltinType::UInt128); | 
|  | 1098 |  | 
| Hans Wennborg | 0d81e01 | 2013-05-10 10:08:40 +0000 | [diff] [blame] | 1099 | // C++ 3.9.1p5 | 
|  | 1100 | if (TargetInfo::isTypeSigned(Target.getWCharType())) | 
|  | 1101 | InitBuiltinType(WCharTy,           BuiltinType::WChar_S); | 
|  | 1102 | else  // -fshort-wchar makes wchar_t be unsigned. | 
|  | 1103 | InitBuiltinType(WCharTy,           BuiltinType::WChar_U); | 
|  | 1104 | if (LangOpts.CPlusPlus && LangOpts.WChar) | 
|  | 1105 | WideCharTy = WCharTy; | 
|  | 1106 | else { | 
|  | 1107 | // C99 (or C++ using -fno-wchar). | 
|  | 1108 | WideCharTy = getFromTargetType(Target.getWCharType()); | 
|  | 1109 | } | 
| Argyrios Kyrtzidis | 40e9e48 | 2008-08-09 16:51:54 +0000 | [diff] [blame] | 1110 |  | 
| James Molloy | 3636554 | 2012-05-04 10:55:22 +0000 | [diff] [blame] | 1111 | WIntTy = getFromTargetType(Target.getWIntType()); | 
|  | 1112 |  | 
| Alisdair Meredith | a9ad47d | 2009-07-14 06:30:34 +0000 | [diff] [blame] | 1113 | if (LangOpts.CPlusPlus) // C++0x 3.9.1p5, extension for C++ | 
|  | 1114 | InitBuiltinType(Char16Ty,           BuiltinType::Char16); | 
|  | 1115 | else // C99 | 
|  | 1116 | Char16Ty = getFromTargetType(Target.getChar16Type()); | 
|  | 1117 |  | 
|  | 1118 | if (LangOpts.CPlusPlus) // C++0x 3.9.1p5, extension for C++ | 
|  | 1119 | InitBuiltinType(Char32Ty,           BuiltinType::Char32); | 
|  | 1120 | else // C99 | 
|  | 1121 | Char32Ty = getFromTargetType(Target.getChar32Type()); | 
|  | 1122 |  | 
| Douglas Gregor | 4619e43 | 2008-12-05 23:32:09 +0000 | [diff] [blame] | 1123 | // Placeholder type for type-dependent expressions whose type is | 
|  | 1124 | // completely unknown. No code should ever check a type against | 
|  | 1125 | // DependentTy and users should never see it; however, it is here to | 
|  | 1126 | // help diagnose failures to properly check for type-dependent | 
|  | 1127 | // expressions. | 
|  | 1128 | InitBuiltinType(DependentTy,         BuiltinType::Dependent); | 
| Douglas Gregor | 5251f1b | 2008-10-21 16:13:35 +0000 | [diff] [blame] | 1129 |  | 
| John McCall | 36e7fe3 | 2010-10-12 00:20:44 +0000 | [diff] [blame] | 1130 | // Placeholder type for functions. | 
|  | 1131 | InitBuiltinType(OverloadTy,          BuiltinType::Overload); | 
|  | 1132 |  | 
| John McCall | 0009fcc | 2011-04-26 20:42:42 +0000 | [diff] [blame] | 1133 | // Placeholder type for bound members. | 
|  | 1134 | InitBuiltinType(BoundMemberTy,       BuiltinType::BoundMember); | 
|  | 1135 |  | 
| John McCall | 526ab47 | 2011-10-25 17:37:35 +0000 | [diff] [blame] | 1136 | // Placeholder type for pseudo-objects. | 
|  | 1137 | InitBuiltinType(PseudoObjectTy,      BuiltinType::PseudoObject); | 
|  | 1138 |  | 
| John McCall | 3199634 | 2011-04-07 08:22:57 +0000 | [diff] [blame] | 1139 | // "any" type; useful for debugger-like clients. | 
|  | 1140 | InitBuiltinType(UnknownAnyTy,        BuiltinType::UnknownAny); | 
|  | 1141 |  | 
| John McCall | 8a6b59a | 2011-10-17 18:09:15 +0000 | [diff] [blame] | 1142 | // Placeholder type for unbridged ARC casts. | 
|  | 1143 | InitBuiltinType(ARCUnbridgedCastTy,  BuiltinType::ARCUnbridgedCast); | 
|  | 1144 |  | 
| Eli Friedman | 34866c7 | 2012-08-31 00:14:07 +0000 | [diff] [blame] | 1145 | // Placeholder type for builtin functions. | 
|  | 1146 | InitBuiltinType(BuiltinFnTy,  BuiltinType::BuiltinFn); | 
|  | 1147 |  | 
| Alexey Bataev | 1a3320e | 2015-08-25 14:24:04 +0000 | [diff] [blame] | 1148 | // Placeholder type for OMP array sections. | 
|  | 1149 | if (LangOpts.OpenMP) | 
|  | 1150 | InitBuiltinType(OMPArraySectionTy, BuiltinType::OMPArraySection); | 
|  | 1151 |  | 
| Chris Lattner | 970e54e | 2006-11-12 00:37:36 +0000 | [diff] [blame] | 1152 | // C99 6.2.5p11. | 
| Chris Lattner | c639593 | 2007-06-22 20:56:16 +0000 | [diff] [blame] | 1153 | FloatComplexTy      = getComplexType(FloatTy); | 
|  | 1154 | DoubleComplexTy     = getComplexType(DoubleTy); | 
|  | 1155 | LongDoubleComplexTy = getComplexType(LongDoubleTy); | 
| Nemanja Ivanovic | bb1ea2d | 2016-05-09 08:52:33 +0000 | [diff] [blame] | 1156 | Float128ComplexTy   = getComplexType(Float128Ty); | 
| Douglas Gregor | 5251f1b | 2008-10-21 16:13:35 +0000 | [diff] [blame] | 1157 |  | 
| Fariborz Jahanian | 252ba5f | 2009-11-21 19:53:08 +0000 | [diff] [blame] | 1158 | // Builtin types for 'id', 'Class', and 'SEL'. | 
| Steve Naroff | 1329fa0 | 2009-07-15 18:40:39 +0000 | [diff] [blame] | 1159 | InitBuiltinType(ObjCBuiltinIdTy, BuiltinType::ObjCId); | 
|  | 1160 | InitBuiltinType(ObjCBuiltinClassTy, BuiltinType::ObjCClass); | 
| Fariborz Jahanian | 252ba5f | 2009-11-21 19:53:08 +0000 | [diff] [blame] | 1161 | InitBuiltinType(ObjCBuiltinSelTy, BuiltinType::ObjCSel); | 
| Guy Benyei | d8a08ea | 2012-12-18 14:38:23 +0000 | [diff] [blame] | 1162 |  | 
| Alexey Bader | 954ba21 | 2016-04-08 13:40:33 +0000 | [diff] [blame] | 1163 | if (LangOpts.OpenCL) { | 
|  | 1164 | #define IMAGE_TYPE(ImgType, Id, SingletonId, Access, Suffix) \ | 
|  | 1165 | InitBuiltinType(SingletonId, BuiltinType::Id); | 
| Alexey Bader | b62f144 | 2016-04-13 08:33:41 +0000 | [diff] [blame] | 1166 | #include "clang/Basic/OpenCLImageTypes.def" | 
| Guy Benyei | 1b4fb3e | 2013-01-20 12:31:11 +0000 | [diff] [blame] | 1167 |  | 
| Guy Benyei | 6105419 | 2013-02-07 10:55:47 +0000 | [diff] [blame] | 1168 | InitBuiltinType(OCLSamplerTy, BuiltinType::OCLSampler); | 
| Guy Benyei | 1b4fb3e | 2013-01-20 12:31:11 +0000 | [diff] [blame] | 1169 | InitBuiltinType(OCLEventTy, BuiltinType::OCLEvent); | 
| Alexey Bader | 9c8453f | 2015-09-15 11:18:52 +0000 | [diff] [blame] | 1170 | InitBuiltinType(OCLClkEventTy, BuiltinType::OCLClkEvent); | 
|  | 1171 | InitBuiltinType(OCLQueueTy, BuiltinType::OCLQueue); | 
|  | 1172 | InitBuiltinType(OCLNDRangeTy, BuiltinType::OCLNDRange); | 
|  | 1173 | InitBuiltinType(OCLReserveIDTy, BuiltinType::OCLReserveID); | 
| Guy Benyei | d8a08ea | 2012-12-18 14:38:23 +0000 | [diff] [blame] | 1174 | } | 
| Ted Kremenek | e65b086 | 2012-03-06 20:05:56 +0000 | [diff] [blame] | 1175 |  | 
|  | 1176 | // Builtin type for __objc_yes and __objc_no | 
| Fariborz Jahanian | 29898f4 | 2012-04-16 21:03:30 +0000 | [diff] [blame] | 1177 | ObjCBuiltinBoolTy = (Target.useSignedCharForObjCBool() ? | 
|  | 1178 | SignedCharTy : BoolTy); | 
| Ted Kremenek | e65b086 | 2012-03-06 20:05:56 +0000 | [diff] [blame] | 1179 |  | 
| Ted Kremenek | 1b0ea82 | 2008-01-07 19:49:32 +0000 | [diff] [blame] | 1180 | ObjCConstantStringType = QualType(); | 
| Fariborz Jahanian | cb6c867 | 2013-01-04 18:45:40 +0000 | [diff] [blame] | 1181 |  | 
|  | 1182 | ObjCSuperType = QualType(); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 1183 |  | 
| Fariborz Jahanian | 797f24c | 2007-10-29 22:57:28 +0000 | [diff] [blame] | 1184 | // void * type | 
|  | 1185 | VoidPtrTy = getPointerType(VoidTy); | 
| Sebastian Redl | 576fd42 | 2009-05-10 18:38:11 +0000 | [diff] [blame] | 1186 |  | 
|  | 1187 | // nullptr type (C++0x 2.14.7) | 
|  | 1188 | InitBuiltinType(NullPtrTy,           BuiltinType::NullPtr); | 
| Anton Korobeynikov | f0c267e | 2011-10-14 23:23:15 +0000 | [diff] [blame] | 1189 |  | 
|  | 1190 | // half type (OpenCL 6.1.1.1) / ARM NEON __fp16 | 
|  | 1191 | InitBuiltinType(HalfTy, BuiltinType::Half); | 
| Meador Inge | cfb6090 | 2012-07-01 15:57:25 +0000 | [diff] [blame] | 1192 |  | 
|  | 1193 | // Builtin type used to help define __builtin_va_list. | 
| Richard Smith | 9b88a4c | 2015-07-27 05:40:23 +0000 | [diff] [blame] | 1194 | VaListTagDecl = nullptr; | 
| Chris Lattner | 970e54e | 2006-11-12 00:37:36 +0000 | [diff] [blame] | 1195 | } | 
|  | 1196 |  | 
| David Blaikie | 9c902b5 | 2011-09-25 23:23:43 +0000 | [diff] [blame] | 1197 | DiagnosticsEngine &ASTContext::getDiagnostics() const { | 
| Argyrios Kyrtzidis | ca0d0cd | 2010-09-22 14:32:24 +0000 | [diff] [blame] | 1198 | return SourceMgr.getDiagnostics(); | 
|  | 1199 | } | 
|  | 1200 |  | 
| Douglas Gregor | 561eceb | 2010-08-30 16:49:28 +0000 | [diff] [blame] | 1201 | AttrVec& ASTContext::getDeclAttrs(const Decl *D) { | 
|  | 1202 | AttrVec *&Result = DeclAttrs[D]; | 
|  | 1203 | if (!Result) { | 
|  | 1204 | void *Mem = Allocate(sizeof(AttrVec)); | 
|  | 1205 | Result = new (Mem) AttrVec; | 
|  | 1206 | } | 
|  | 1207 |  | 
|  | 1208 | return *Result; | 
|  | 1209 | } | 
|  | 1210 |  | 
|  | 1211 | /// \brief Erase the attributes corresponding to the given declaration. | 
|  | 1212 | void ASTContext::eraseDeclAttrs(const Decl *D) { | 
|  | 1213 | llvm::DenseMap<const Decl*, AttrVec*>::iterator Pos = DeclAttrs.find(D); | 
|  | 1214 | if (Pos != DeclAttrs.end()) { | 
|  | 1215 | Pos->second->~AttrVec(); | 
|  | 1216 | DeclAttrs.erase(Pos); | 
|  | 1217 | } | 
|  | 1218 | } | 
|  | 1219 |  | 
| Larisse Voufo | 39a1e50 | 2013-08-06 01:03:05 +0000 | [diff] [blame] | 1220 | // FIXME: Remove ? | 
| Douglas Gregor | 86d142a | 2009-10-08 07:24:58 +0000 | [diff] [blame] | 1221 | MemberSpecializationInfo * | 
| Douglas Gregor | 3c74d41 | 2009-10-14 20:14:33 +0000 | [diff] [blame] | 1222 | ASTContext::getInstantiatedFromStaticDataMember(const VarDecl *Var) { | 
| Douglas Gregor | a6ef8f0 | 2009-07-24 20:34:43 +0000 | [diff] [blame] | 1223 | assert(Var->isStaticDataMember() && "Not a static data member"); | 
| Larisse Voufo | 39a1e50 | 2013-08-06 01:03:05 +0000 | [diff] [blame] | 1224 | return getTemplateOrSpecializationInfo(Var) | 
|  | 1225 | .dyn_cast<MemberSpecializationInfo *>(); | 
|  | 1226 | } | 
|  | 1227 |  | 
|  | 1228 | ASTContext::TemplateOrSpecializationInfo | 
|  | 1229 | ASTContext::getTemplateOrSpecializationInfo(const VarDecl *Var) { | 
|  | 1230 | llvm::DenseMap<const VarDecl *, TemplateOrSpecializationInfo>::iterator Pos = | 
|  | 1231 | TemplateOrInstantiation.find(Var); | 
|  | 1232 | if (Pos == TemplateOrInstantiation.end()) | 
|  | 1233 | return TemplateOrSpecializationInfo(); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 1234 |  | 
| Douglas Gregor | a6ef8f0 | 2009-07-24 20:34:43 +0000 | [diff] [blame] | 1235 | return Pos->second; | 
|  | 1236 | } | 
|  | 1237 |  | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 1238 | void | 
| Douglas Gregor | 86d142a | 2009-10-08 07:24:58 +0000 | [diff] [blame] | 1239 | ASTContext::setInstantiatedFromStaticDataMember(VarDecl *Inst, VarDecl *Tmpl, | 
| Argyrios Kyrtzidis | cdb8b3f | 2010-07-04 21:44:00 +0000 | [diff] [blame] | 1240 | TemplateSpecializationKind TSK, | 
|  | 1241 | SourceLocation PointOfInstantiation) { | 
| Douglas Gregor | a6ef8f0 | 2009-07-24 20:34:43 +0000 | [diff] [blame] | 1242 | assert(Inst->isStaticDataMember() && "Not a static data member"); | 
|  | 1243 | assert(Tmpl->isStaticDataMember() && "Not a static data member"); | 
| Larisse Voufo | 39a1e50 | 2013-08-06 01:03:05 +0000 | [diff] [blame] | 1244 | setTemplateOrSpecializationInfo(Inst, new (*this) MemberSpecializationInfo( | 
|  | 1245 | Tmpl, TSK, PointOfInstantiation)); | 
|  | 1246 | } | 
|  | 1247 |  | 
|  | 1248 | void | 
|  | 1249 | ASTContext::setTemplateOrSpecializationInfo(VarDecl *Inst, | 
|  | 1250 | TemplateOrSpecializationInfo TSI) { | 
|  | 1251 | assert(!TemplateOrInstantiation[Inst] && | 
|  | 1252 | "Already noted what the variable was instantiated from"); | 
|  | 1253 | TemplateOrInstantiation[Inst] = TSI; | 
| Douglas Gregor | a6ef8f0 | 2009-07-24 20:34:43 +0000 | [diff] [blame] | 1254 | } | 
|  | 1255 |  | 
| Francois Pichet | 00c7e6c | 2011-08-14 03:52:19 +0000 | [diff] [blame] | 1256 | FunctionDecl *ASTContext::getClassScopeSpecializationPattern( | 
|  | 1257 | const FunctionDecl *FD){ | 
|  | 1258 | assert(FD && "Specialization is 0"); | 
|  | 1259 | llvm::DenseMap<const FunctionDecl*, FunctionDecl *>::const_iterator Pos | 
| Francois Pichet | 5792825 | 2011-08-14 14:28:49 +0000 | [diff] [blame] | 1260 | = ClassScopeSpecializationPattern.find(FD); | 
|  | 1261 | if (Pos == ClassScopeSpecializationPattern.end()) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 1262 | return nullptr; | 
| Francois Pichet | 00c7e6c | 2011-08-14 03:52:19 +0000 | [diff] [blame] | 1263 |  | 
|  | 1264 | return Pos->second; | 
|  | 1265 | } | 
|  | 1266 |  | 
|  | 1267 | void ASTContext::setClassScopeSpecializationPattern(FunctionDecl *FD, | 
|  | 1268 | FunctionDecl *Pattern) { | 
|  | 1269 | assert(FD && "Specialization is 0"); | 
|  | 1270 | assert(Pattern && "Class scope specialization pattern is 0"); | 
| Francois Pichet | 5792825 | 2011-08-14 14:28:49 +0000 | [diff] [blame] | 1271 | ClassScopeSpecializationPattern[FD] = Pattern; | 
| Francois Pichet | 00c7e6c | 2011-08-14 03:52:19 +0000 | [diff] [blame] | 1272 | } | 
|  | 1273 |  | 
| John McCall | e61f2ba | 2009-11-18 02:36:19 +0000 | [diff] [blame] | 1274 | NamedDecl * | 
| John McCall | b96ec56 | 2009-12-04 22:46:56 +0000 | [diff] [blame] | 1275 | ASTContext::getInstantiatedFromUsingDecl(UsingDecl *UUD) { | 
| John McCall | e61f2ba | 2009-11-18 02:36:19 +0000 | [diff] [blame] | 1276 | llvm::DenseMap<UsingDecl *, NamedDecl *>::const_iterator Pos | 
| John McCall | b96ec56 | 2009-12-04 22:46:56 +0000 | [diff] [blame] | 1277 | = InstantiatedFromUsingDecl.find(UUD); | 
|  | 1278 | if (Pos == InstantiatedFromUsingDecl.end()) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 1279 | return nullptr; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 1280 |  | 
| Anders Carlsson | 4bb87ce | 2009-08-29 19:37:28 +0000 | [diff] [blame] | 1281 | return Pos->second; | 
|  | 1282 | } | 
|  | 1283 |  | 
|  | 1284 | void | 
| John McCall | b96ec56 | 2009-12-04 22:46:56 +0000 | [diff] [blame] | 1285 | ASTContext::setInstantiatedFromUsingDecl(UsingDecl *Inst, NamedDecl *Pattern) { | 
|  | 1286 | assert((isa<UsingDecl>(Pattern) || | 
|  | 1287 | isa<UnresolvedUsingValueDecl>(Pattern) || | 
|  | 1288 | isa<UnresolvedUsingTypenameDecl>(Pattern)) && | 
|  | 1289 | "pattern decl is not a using decl"); | 
|  | 1290 | assert(!InstantiatedFromUsingDecl[Inst] && "pattern already exists"); | 
|  | 1291 | InstantiatedFromUsingDecl[Inst] = Pattern; | 
|  | 1292 | } | 
|  | 1293 |  | 
|  | 1294 | UsingShadowDecl * | 
|  | 1295 | ASTContext::getInstantiatedFromUsingShadowDecl(UsingShadowDecl *Inst) { | 
|  | 1296 | llvm::DenseMap<UsingShadowDecl*, UsingShadowDecl*>::const_iterator Pos | 
|  | 1297 | = InstantiatedFromUsingShadowDecl.find(Inst); | 
|  | 1298 | if (Pos == InstantiatedFromUsingShadowDecl.end()) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 1299 | return nullptr; | 
| John McCall | b96ec56 | 2009-12-04 22:46:56 +0000 | [diff] [blame] | 1300 |  | 
|  | 1301 | return Pos->second; | 
|  | 1302 | } | 
|  | 1303 |  | 
|  | 1304 | void | 
|  | 1305 | ASTContext::setInstantiatedFromUsingShadowDecl(UsingShadowDecl *Inst, | 
|  | 1306 | UsingShadowDecl *Pattern) { | 
|  | 1307 | assert(!InstantiatedFromUsingShadowDecl[Inst] && "pattern already exists"); | 
|  | 1308 | InstantiatedFromUsingShadowDecl[Inst] = Pattern; | 
| Anders Carlsson | 4bb87ce | 2009-08-29 19:37:28 +0000 | [diff] [blame] | 1309 | } | 
|  | 1310 |  | 
| Anders Carlsson | 5da8484 | 2009-09-01 04:26:58 +0000 | [diff] [blame] | 1311 | FieldDecl *ASTContext::getInstantiatedFromUnnamedFieldDecl(FieldDecl *Field) { | 
|  | 1312 | llvm::DenseMap<FieldDecl *, FieldDecl *>::iterator Pos | 
|  | 1313 | = InstantiatedFromUnnamedFieldDecl.find(Field); | 
|  | 1314 | if (Pos == InstantiatedFromUnnamedFieldDecl.end()) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 1315 | return nullptr; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 1316 |  | 
| Anders Carlsson | 5da8484 | 2009-09-01 04:26:58 +0000 | [diff] [blame] | 1317 | return Pos->second; | 
|  | 1318 | } | 
|  | 1319 |  | 
|  | 1320 | void ASTContext::setInstantiatedFromUnnamedFieldDecl(FieldDecl *Inst, | 
|  | 1321 | FieldDecl *Tmpl) { | 
|  | 1322 | assert(!Inst->getDeclName() && "Instantiated field decl is not unnamed"); | 
|  | 1323 | assert(!Tmpl->getDeclName() && "Template field decl is not unnamed"); | 
|  | 1324 | assert(!InstantiatedFromUnnamedFieldDecl[Inst] && | 
|  | 1325 | "Already noted what unnamed field was instantiated from"); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 1326 |  | 
| Anders Carlsson | 5da8484 | 2009-09-01 04:26:58 +0000 | [diff] [blame] | 1327 | InstantiatedFromUnnamedFieldDecl[Inst] = Tmpl; | 
|  | 1328 | } | 
|  | 1329 |  | 
| Douglas Gregor | 832940b | 2010-03-02 23:58:15 +0000 | [diff] [blame] | 1330 | ASTContext::overridden_cxx_method_iterator | 
|  | 1331 | ASTContext::overridden_methods_begin(const CXXMethodDecl *Method) const { | 
| Clement Courbet | 6ecaec8 | 2016-07-05 07:49:31 +0000 | [diff] [blame] | 1332 | llvm::DenseMap<const CXXMethodDecl *, CXXMethodVector>::const_iterator Pos = | 
|  | 1333 | OverriddenMethods.find(Method->getCanonicalDecl()); | 
| Douglas Gregor | 832940b | 2010-03-02 23:58:15 +0000 | [diff] [blame] | 1334 | if (Pos == OverriddenMethods.end()) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 1335 | return nullptr; | 
| Douglas Gregor | 832940b | 2010-03-02 23:58:15 +0000 | [diff] [blame] | 1336 | return Pos->second.begin(); | 
|  | 1337 | } | 
|  | 1338 |  | 
|  | 1339 | ASTContext::overridden_cxx_method_iterator | 
|  | 1340 | ASTContext::overridden_methods_end(const CXXMethodDecl *Method) const { | 
| Clement Courbet | 6ecaec8 | 2016-07-05 07:49:31 +0000 | [diff] [blame] | 1341 | llvm::DenseMap<const CXXMethodDecl *, CXXMethodVector>::const_iterator Pos = | 
|  | 1342 | OverriddenMethods.find(Method->getCanonicalDecl()); | 
| Douglas Gregor | 832940b | 2010-03-02 23:58:15 +0000 | [diff] [blame] | 1343 | if (Pos == OverriddenMethods.end()) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 1344 | return nullptr; | 
| Douglas Gregor | 832940b | 2010-03-02 23:58:15 +0000 | [diff] [blame] | 1345 | return Pos->second.end(); | 
|  | 1346 | } | 
|  | 1347 |  | 
| Argyrios Kyrtzidis | 6685e8a | 2010-07-04 21:44:35 +0000 | [diff] [blame] | 1348 | unsigned | 
|  | 1349 | ASTContext::overridden_methods_size(const CXXMethodDecl *Method) const { | 
| Clement Courbet | 6ecaec8 | 2016-07-05 07:49:31 +0000 | [diff] [blame] | 1350 | llvm::DenseMap<const CXXMethodDecl *, CXXMethodVector>::const_iterator Pos = | 
|  | 1351 | OverriddenMethods.find(Method->getCanonicalDecl()); | 
| Argyrios Kyrtzidis | 6685e8a | 2010-07-04 21:44:35 +0000 | [diff] [blame] | 1352 | if (Pos == OverriddenMethods.end()) | 
|  | 1353 | return 0; | 
| Chandler Carruth | b1bcd5d | 2016-06-11 04:45:38 +0000 | [diff] [blame] | 1354 | return Pos->second.size(); | 
| Clement Courbet | 8251ebf | 2016-06-10 11:54:43 +0000 | [diff] [blame] | 1355 | } | 
|  | 1356 |  | 
| Clement Courbet | 6ecaec8 | 2016-07-05 07:49:31 +0000 | [diff] [blame] | 1357 | ASTContext::overridden_method_range | 
|  | 1358 | ASTContext::overridden_methods(const CXXMethodDecl *Method) const { | 
|  | 1359 | return overridden_method_range(overridden_methods_begin(Method), | 
|  | 1360 | overridden_methods_end(Method)); | 
|  | 1361 | } | 
|  | 1362 |  | 
| Douglas Gregor | 832940b | 2010-03-02 23:58:15 +0000 | [diff] [blame] | 1363 | void ASTContext::addOverriddenMethod(const CXXMethodDecl *Method, | 
|  | 1364 | const CXXMethodDecl *Overridden) { | 
| Argyrios Kyrtzidis | cc4ca0a | 2012-10-09 01:23:45 +0000 | [diff] [blame] | 1365 | assert(Method->isCanonicalDecl() && Overridden->isCanonicalDecl()); | 
| Douglas Gregor | 832940b | 2010-03-02 23:58:15 +0000 | [diff] [blame] | 1366 | OverriddenMethods[Method].push_back(Overridden); | 
|  | 1367 | } | 
|  | 1368 |  | 
| Dmitri Gribenko | 941ab0f | 2012-11-03 14:24:57 +0000 | [diff] [blame] | 1369 | void ASTContext::getOverriddenMethods( | 
|  | 1370 | const NamedDecl *D, | 
|  | 1371 | SmallVectorImpl<const NamedDecl *> &Overridden) const { | 
| Argyrios Kyrtzidis | b9556e6 | 2012-10-09 01:23:50 +0000 | [diff] [blame] | 1372 | assert(D); | 
|  | 1373 |  | 
|  | 1374 | if (const CXXMethodDecl *CXXMethod = dyn_cast<CXXMethodDecl>(D)) { | 
| Argyrios Kyrtzidis | c8e7008 | 2013-04-17 00:09:03 +0000 | [diff] [blame] | 1375 | Overridden.append(overridden_methods_begin(CXXMethod), | 
|  | 1376 | overridden_methods_end(CXXMethod)); | 
| Argyrios Kyrtzidis | b9556e6 | 2012-10-09 01:23:50 +0000 | [diff] [blame] | 1377 | return; | 
|  | 1378 | } | 
|  | 1379 |  | 
|  | 1380 | const ObjCMethodDecl *Method = dyn_cast<ObjCMethodDecl>(D); | 
|  | 1381 | if (!Method) | 
|  | 1382 | return; | 
|  | 1383 |  | 
| Argyrios Kyrtzidis | 353f6a4 | 2012-10-09 18:19:01 +0000 | [diff] [blame] | 1384 | SmallVector<const ObjCMethodDecl *, 8> OverDecls; | 
|  | 1385 | Method->getOverriddenMethods(OverDecls); | 
| Argyrios Kyrtzidis | a7a1081 | 2012-10-09 20:08:43 +0000 | [diff] [blame] | 1386 | Overridden.append(OverDecls.begin(), OverDecls.end()); | 
| Argyrios Kyrtzidis | b9556e6 | 2012-10-09 01:23:50 +0000 | [diff] [blame] | 1387 | } | 
|  | 1388 |  | 
| Douglas Gregor | 0f2a360 | 2011-12-03 00:30:27 +0000 | [diff] [blame] | 1389 | void ASTContext::addedLocalImportDecl(ImportDecl *Import) { | 
|  | 1390 | assert(!Import->NextLocalImport && "Import declaration already in the chain"); | 
|  | 1391 | assert(!Import->isFromASTFile() && "Non-local import declaration"); | 
|  | 1392 | if (!FirstLocalImport) { | 
|  | 1393 | FirstLocalImport = Import; | 
|  | 1394 | LastLocalImport = Import; | 
|  | 1395 | return; | 
|  | 1396 | } | 
|  | 1397 |  | 
|  | 1398 | LastLocalImport->NextLocalImport = Import; | 
|  | 1399 | LastLocalImport = Import; | 
|  | 1400 | } | 
|  | 1401 |  | 
| Chris Lattner | 53cfe80 | 2007-07-18 17:52:12 +0000 | [diff] [blame] | 1402 | //===----------------------------------------------------------------------===// | 
|  | 1403 | //                         Type Sizing and Analysis | 
|  | 1404 | //===----------------------------------------------------------------------===// | 
| Chris Lattner | 983a8bb | 2007-07-13 22:13:22 +0000 | [diff] [blame] | 1405 |  | 
| Chris Lattner | 9a8d1d9 | 2008-06-30 18:32:54 +0000 | [diff] [blame] | 1406 | /// getFloatTypeSemantics - Return the APFloat 'semantics' for the specified | 
|  | 1407 | /// scalar floating point type. | 
|  | 1408 | const llvm::fltSemantics &ASTContext::getFloatTypeSemantics(QualType T) const { | 
| John McCall | 9dd450b | 2009-09-21 23:43:11 +0000 | [diff] [blame] | 1409 | const BuiltinType *BT = T->getAs<BuiltinType>(); | 
| Chris Lattner | 9a8d1d9 | 2008-06-30 18:32:54 +0000 | [diff] [blame] | 1410 | assert(BT && "Not a floating point type!"); | 
|  | 1411 | switch (BT->getKind()) { | 
| David Blaikie | 83d382b | 2011-09-23 05:06:16 +0000 | [diff] [blame] | 1412 | default: llvm_unreachable("Not a floating point type!"); | 
| Anton Korobeynikov | f0c267e | 2011-10-14 23:23:15 +0000 | [diff] [blame] | 1413 | case BuiltinType::Half:       return Target->getHalfFormat(); | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 1414 | case BuiltinType::Float:      return Target->getFloatFormat(); | 
|  | 1415 | case BuiltinType::Double:     return Target->getDoubleFormat(); | 
|  | 1416 | case BuiltinType::LongDouble: return Target->getLongDoubleFormat(); | 
| Nemanja Ivanovic | bb1ea2d | 2016-05-09 08:52:33 +0000 | [diff] [blame] | 1417 | case BuiltinType::Float128:   return Target->getFloat128Format(); | 
| Chris Lattner | 9a8d1d9 | 2008-06-30 18:32:54 +0000 | [diff] [blame] | 1418 | } | 
|  | 1419 | } | 
|  | 1420 |  | 
| Rafael Espindola | 71eccb3 | 2013-08-08 19:53:46 +0000 | [diff] [blame] | 1421 | CharUnits ASTContext::getDeclAlign(const Decl *D, bool ForAlignof) const { | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 1422 | unsigned Align = Target->getCharWidth(); | 
| Eli Friedman | 19a546c | 2009-02-22 02:56:25 +0000 | [diff] [blame] | 1423 |  | 
| John McCall | 9426870 | 2010-10-08 18:24:19 +0000 | [diff] [blame] | 1424 | bool UseAlignAttrOnly = false; | 
|  | 1425 | if (unsigned AlignFromAttr = D->getMaxAlignment()) { | 
|  | 1426 | Align = AlignFromAttr; | 
| Eli Friedman | 19a546c | 2009-02-22 02:56:25 +0000 | [diff] [blame] | 1427 |  | 
| John McCall | 9426870 | 2010-10-08 18:24:19 +0000 | [diff] [blame] | 1428 | // __attribute__((aligned)) can increase or decrease alignment | 
|  | 1429 | // *except* on a struct or struct member, where it only increases | 
|  | 1430 | // alignment unless 'packed' is also specified. | 
|  | 1431 | // | 
| Peter Collingbourne | 2f3cf4b | 2011-09-29 18:04:28 +0000 | [diff] [blame] | 1432 | // It is an error for alignas to decrease alignment, so we can | 
| John McCall | 9426870 | 2010-10-08 18:24:19 +0000 | [diff] [blame] | 1433 | // ignore that possibility;  Sema should diagnose it. | 
|  | 1434 | if (isa<FieldDecl>(D)) { | 
|  | 1435 | UseAlignAttrOnly = D->hasAttr<PackedAttr>() || | 
|  | 1436 | cast<FieldDecl>(D)->getParent()->hasAttr<PackedAttr>(); | 
|  | 1437 | } else { | 
|  | 1438 | UseAlignAttrOnly = true; | 
|  | 1439 | } | 
|  | 1440 | } | 
| Fariborz Jahanian | 9f10718 | 2011-05-05 21:19:14 +0000 | [diff] [blame] | 1441 | else if (isa<FieldDecl>(D)) | 
|  | 1442 | UseAlignAttrOnly = | 
|  | 1443 | D->hasAttr<PackedAttr>() || | 
|  | 1444 | cast<FieldDecl>(D)->getParent()->hasAttr<PackedAttr>(); | 
| John McCall | 9426870 | 2010-10-08 18:24:19 +0000 | [diff] [blame] | 1445 |  | 
| John McCall | 4e81961 | 2011-01-20 07:57:12 +0000 | [diff] [blame] | 1446 | // If we're using the align attribute only, just ignore everything | 
|  | 1447 | // else about the declaration and its type. | 
| John McCall | 9426870 | 2010-10-08 18:24:19 +0000 | [diff] [blame] | 1448 | if (UseAlignAttrOnly) { | 
| John McCall | 4e81961 | 2011-01-20 07:57:12 +0000 | [diff] [blame] | 1449 | // do nothing | 
|  | 1450 |  | 
| John McCall | 9426870 | 2010-10-08 18:24:19 +0000 | [diff] [blame] | 1451 | } else if (const ValueDecl *VD = dyn_cast<ValueDecl>(D)) { | 
| Chris Lattner | 6806131 | 2009-01-24 21:53:27 +0000 | [diff] [blame] | 1452 | QualType T = VD->getType(); | 
| Richard Smith | f6d7030 | 2014-06-10 23:34:28 +0000 | [diff] [blame] | 1453 | if (const ReferenceType *RT = T->getAs<ReferenceType>()) { | 
| Rafael Espindola | 71eccb3 | 2013-08-08 19:53:46 +0000 | [diff] [blame] | 1454 | if (ForAlignof) | 
| Sebastian Redl | 22e2e5c | 2009-11-23 17:18:46 +0000 | [diff] [blame] | 1455 | T = RT->getPointeeType(); | 
|  | 1456 | else | 
|  | 1457 | T = getPointerType(RT->getPointeeType()); | 
|  | 1458 | } | 
| Richard Smith | f6d7030 | 2014-06-10 23:34:28 +0000 | [diff] [blame] | 1459 | QualType BaseT = getBaseElementType(T); | 
|  | 1460 | if (!BaseT->isIncompleteType() && !T->isFunctionType()) { | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 1461 | // Adjust alignments of declarations with array type by the | 
|  | 1462 | // large-array alignment on the target. | 
| Rafael Espindola | 8857275 | 2013-08-07 18:08:19 +0000 | [diff] [blame] | 1463 | if (const ArrayType *arrayType = getAsArrayType(T)) { | 
| Rafael Espindola | 71eccb3 | 2013-08-08 19:53:46 +0000 | [diff] [blame] | 1464 | unsigned MinWidth = Target->getLargeArrayMinWidth(); | 
|  | 1465 | if (!ForAlignof && MinWidth) { | 
| Rafael Espindola | 8857275 | 2013-08-07 18:08:19 +0000 | [diff] [blame] | 1466 | if (isa<VariableArrayType>(arrayType)) | 
|  | 1467 | Align = std::max(Align, Target->getLargeArrayAlign()); | 
|  | 1468 | else if (isa<ConstantArrayType>(arrayType) && | 
|  | 1469 | MinWidth <= getTypeSize(cast<ConstantArrayType>(arrayType))) | 
|  | 1470 | Align = std::max(Align, Target->getLargeArrayAlign()); | 
|  | 1471 | } | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 1472 | } | 
| Chad Rosier | 99ee782 | 2011-07-26 07:03:04 +0000 | [diff] [blame] | 1473 | Align = std::max(Align, getPreferredTypeAlign(T.getTypePtr())); | 
| Ulrich Weigand | fa80642 | 2013-05-06 16:23:57 +0000 | [diff] [blame] | 1474 | if (const VarDecl *VD = dyn_cast<VarDecl>(D)) { | 
| Ulrich Weigand | b63f779 | 2015-04-21 17:26:18 +0000 | [diff] [blame] | 1475 | if (VD->hasGlobalStorage() && !ForAlignof) | 
| Ulrich Weigand | fa80642 | 2013-05-06 16:23:57 +0000 | [diff] [blame] | 1476 | Align = std::max(Align, getTargetInfo().getMinGlobalAlign()); | 
|  | 1477 | } | 
| Eli Friedman | 19a546c | 2009-02-22 02:56:25 +0000 | [diff] [blame] | 1478 | } | 
| John McCall | 4e81961 | 2011-01-20 07:57:12 +0000 | [diff] [blame] | 1479 |  | 
|  | 1480 | // Fields can be subject to extra alignment constraints, like if | 
|  | 1481 | // the field is packed, the struct is packed, or the struct has a | 
|  | 1482 | // a max-field-alignment constraint (#pragma pack).  So calculate | 
|  | 1483 | // the actual alignment of the field within the struct, and then | 
|  | 1484 | // (as we're expected to) constrain that by the alignment of the type. | 
| Matt Beaumont-Gay | 3577995 | 2013-06-25 22:19:15 +0000 | [diff] [blame] | 1485 | if (const FieldDecl *Field = dyn_cast<FieldDecl>(VD)) { | 
|  | 1486 | const RecordDecl *Parent = Field->getParent(); | 
|  | 1487 | // We can only produce a sensible answer if the record is valid. | 
|  | 1488 | if (!Parent->isInvalidDecl()) { | 
|  | 1489 | const ASTRecordLayout &Layout = getASTRecordLayout(Parent); | 
| John McCall | 4e81961 | 2011-01-20 07:57:12 +0000 | [diff] [blame] | 1490 |  | 
| Matt Beaumont-Gay | 3577995 | 2013-06-25 22:19:15 +0000 | [diff] [blame] | 1491 | // Start with the record's overall alignment. | 
|  | 1492 | unsigned FieldAlign = toBits(Layout.getAlignment()); | 
| John McCall | 4e81961 | 2011-01-20 07:57:12 +0000 | [diff] [blame] | 1493 |  | 
| Matt Beaumont-Gay | 3577995 | 2013-06-25 22:19:15 +0000 | [diff] [blame] | 1494 | // Use the GCD of that and the offset within the record. | 
|  | 1495 | uint64_t Offset = Layout.getFieldOffset(Field->getFieldIndex()); | 
|  | 1496 | if (Offset > 0) { | 
|  | 1497 | // Alignment is always a power of 2, so the GCD will be a power of 2, | 
|  | 1498 | // which means we get to do this crazy thing instead of Euclid's. | 
|  | 1499 | uint64_t LowBitOfOffset = Offset & (~Offset + 1); | 
|  | 1500 | if (LowBitOfOffset < FieldAlign) | 
|  | 1501 | FieldAlign = static_cast<unsigned>(LowBitOfOffset); | 
|  | 1502 | } | 
|  | 1503 |  | 
|  | 1504 | Align = std::min(Align, FieldAlign); | 
| John McCall | 4e81961 | 2011-01-20 07:57:12 +0000 | [diff] [blame] | 1505 | } | 
| Charles Davis | 3fc5107 | 2010-02-23 04:52:00 +0000 | [diff] [blame] | 1506 | } | 
| Chris Lattner | 6806131 | 2009-01-24 21:53:27 +0000 | [diff] [blame] | 1507 | } | 
| Eli Friedman | 19a546c | 2009-02-22 02:56:25 +0000 | [diff] [blame] | 1508 |  | 
| Ken Dyck | cc56c54 | 2011-01-15 18:38:59 +0000 | [diff] [blame] | 1509 | return toCharUnitsFromBits(Align); | 
| Chris Lattner | 6806131 | 2009-01-24 21:53:27 +0000 | [diff] [blame] | 1510 | } | 
| Chris Lattner | 9a8d1d9 | 2008-06-30 18:32:54 +0000 | [diff] [blame] | 1511 |  | 
| John McCall | f124992 | 2012-08-21 04:10:00 +0000 | [diff] [blame] | 1512 | // getTypeInfoDataSizeInChars - Return the size of a type, in | 
|  | 1513 | // chars. If the type is a record, its data size is returned.  This is | 
|  | 1514 | // the size of the memcpy that's performed when assigning this type | 
|  | 1515 | // using a trivial copy/move assignment operator. | 
|  | 1516 | std::pair<CharUnits, CharUnits> | 
|  | 1517 | ASTContext::getTypeInfoDataSizeInChars(QualType T) const { | 
|  | 1518 | std::pair<CharUnits, CharUnits> sizeAndAlign = getTypeInfoInChars(T); | 
|  | 1519 |  | 
|  | 1520 | // In C++, objects can sometimes be allocated into the tail padding | 
|  | 1521 | // of a base-class subobject.  We decide whether that's possible | 
|  | 1522 | // during class layout, so here we can just trust the layout results. | 
|  | 1523 | if (getLangOpts().CPlusPlus) { | 
|  | 1524 | if (const RecordType *RT = T->getAs<RecordType>()) { | 
|  | 1525 | const ASTRecordLayout &layout = getASTRecordLayout(RT->getDecl()); | 
|  | 1526 | sizeAndAlign.first = layout.getDataSize(); | 
|  | 1527 | } | 
|  | 1528 | } | 
|  | 1529 |  | 
|  | 1530 | return sizeAndAlign; | 
|  | 1531 | } | 
|  | 1532 |  | 
| Richard Trieu | 04d2d94 | 2013-05-14 21:59:17 +0000 | [diff] [blame] | 1533 | /// getConstantArrayInfoInChars - Performing the computation in CharUnits | 
|  | 1534 | /// instead of in bits prevents overflowing the uint64_t for some large arrays. | 
|  | 1535 | std::pair<CharUnits, CharUnits> | 
|  | 1536 | static getConstantArrayInfoInChars(const ASTContext &Context, | 
|  | 1537 | const ConstantArrayType *CAT) { | 
|  | 1538 | std::pair<CharUnits, CharUnits> EltInfo = | 
|  | 1539 | Context.getTypeInfoInChars(CAT->getElementType()); | 
|  | 1540 | uint64_t Size = CAT->getSize().getZExtValue(); | 
| Richard Trieu | c750907 | 2013-05-14 23:41:50 +0000 | [diff] [blame] | 1541 | assert((Size == 0 || static_cast<uint64_t>(EltInfo.first.getQuantity()) <= | 
|  | 1542 | (uint64_t)(-1)/Size) && | 
| Richard Trieu | 04d2d94 | 2013-05-14 21:59:17 +0000 | [diff] [blame] | 1543 | "Overflow in array type char size evaluation"); | 
|  | 1544 | uint64_t Width = EltInfo.first.getQuantity() * Size; | 
|  | 1545 | unsigned Align = EltInfo.second.getQuantity(); | 
| Warren Hunt | 5ae586a | 2013-11-01 23:59:41 +0000 | [diff] [blame] | 1546 | if (!Context.getTargetInfo().getCXXABI().isMicrosoft() || | 
|  | 1547 | Context.getTargetInfo().getPointerWidth(0) == 64) | 
| Rui Ueyama | 83aa979 | 2016-01-14 21:00:27 +0000 | [diff] [blame] | 1548 | Width = llvm::alignTo(Width, Align); | 
| Richard Trieu | 04d2d94 | 2013-05-14 21:59:17 +0000 | [diff] [blame] | 1549 | return std::make_pair(CharUnits::fromQuantity(Width), | 
|  | 1550 | CharUnits::fromQuantity(Align)); | 
|  | 1551 | } | 
|  | 1552 |  | 
| John McCall | 87fe5d5 | 2010-05-20 01:18:31 +0000 | [diff] [blame] | 1553 | std::pair<CharUnits, CharUnits> | 
| Ken Dyck | d24099d | 2011-02-20 01:55:18 +0000 | [diff] [blame] | 1554 | ASTContext::getTypeInfoInChars(const Type *T) const { | 
| Richard Trieu | 04d2d94 | 2013-05-14 21:59:17 +0000 | [diff] [blame] | 1555 | if (const ConstantArrayType *CAT = dyn_cast<ConstantArrayType>(T)) | 
|  | 1556 | return getConstantArrayInfoInChars(*this, CAT); | 
| David Majnemer | 34b5749 | 2014-07-30 01:30:47 +0000 | [diff] [blame] | 1557 | TypeInfo Info = getTypeInfo(T); | 
|  | 1558 | return std::make_pair(toCharUnitsFromBits(Info.Width), | 
|  | 1559 | toCharUnitsFromBits(Info.Align)); | 
| John McCall | 87fe5d5 | 2010-05-20 01:18:31 +0000 | [diff] [blame] | 1560 | } | 
|  | 1561 |  | 
|  | 1562 | std::pair<CharUnits, CharUnits> | 
| Ken Dyck | d24099d | 2011-02-20 01:55:18 +0000 | [diff] [blame] | 1563 | ASTContext::getTypeInfoInChars(QualType T) const { | 
| John McCall | 87fe5d5 | 2010-05-20 01:18:31 +0000 | [diff] [blame] | 1564 | return getTypeInfoInChars(T.getTypePtr()); | 
|  | 1565 | } | 
|  | 1566 |  | 
| David Majnemer | 34b5749 | 2014-07-30 01:30:47 +0000 | [diff] [blame] | 1567 | bool ASTContext::isAlignmentRequired(const Type *T) const { | 
|  | 1568 | return getTypeInfo(T).AlignIsRequired; | 
|  | 1569 | } | 
| Daniel Dunbar | c647587 | 2012-03-09 04:12:54 +0000 | [diff] [blame] | 1570 |  | 
| David Majnemer | 34b5749 | 2014-07-30 01:30:47 +0000 | [diff] [blame] | 1571 | bool ASTContext::isAlignmentRequired(QualType T) const { | 
|  | 1572 | return isAlignmentRequired(T.getTypePtr()); | 
|  | 1573 | } | 
|  | 1574 |  | 
|  | 1575 | TypeInfo ASTContext::getTypeInfo(const Type *T) const { | 
| David Majnemer | f8d3864 | 2014-07-30 08:42:33 +0000 | [diff] [blame] | 1576 | TypeInfoMap::iterator I = MemoizedTypeInfo.find(T); | 
|  | 1577 | if (I != MemoizedTypeInfo.end()) | 
|  | 1578 | return I->second; | 
|  | 1579 |  | 
|  | 1580 | // This call can invalidate MemoizedTypeInfo[T], so we need a second lookup. | 
|  | 1581 | TypeInfo TI = getTypeInfoImpl(T); | 
|  | 1582 | MemoizedTypeInfo[T] = TI; | 
| Rafael Espindola | eaa88c1 | 2014-07-30 04:40:23 +0000 | [diff] [blame] | 1583 | return TI; | 
| Daniel Dunbar | c647587 | 2012-03-09 04:12:54 +0000 | [diff] [blame] | 1584 | } | 
|  | 1585 |  | 
|  | 1586 | /// getTypeInfoImpl - Return the size of the specified type, in bits.  This | 
|  | 1587 | /// method does not work on incomplete types. | 
| John McCall | 8ccfcb5 | 2009-09-24 19:53:00 +0000 | [diff] [blame] | 1588 | /// | 
|  | 1589 | /// FIXME: Pointers into different addr spaces could have different sizes and | 
|  | 1590 | /// alignment requirements: getPointerInfo should take an AddrSpace, this | 
|  | 1591 | /// should take a QualType, &c. | 
| David Majnemer | 34b5749 | 2014-07-30 01:30:47 +0000 | [diff] [blame] | 1592 | TypeInfo ASTContext::getTypeInfoImpl(const Type *T) const { | 
|  | 1593 | uint64_t Width = 0; | 
|  | 1594 | unsigned Align = 8; | 
|  | 1595 | bool AlignIsRequired = false; | 
| Chris Lattner | 983a8bb | 2007-07-13 22:13:22 +0000 | [diff] [blame] | 1596 | switch (T->getTypeClass()) { | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 1597 | #define TYPE(Class, Base) | 
|  | 1598 | #define ABSTRACT_TYPE(Class, Base) | 
| Douglas Gregor | ef462e6 | 2009-04-30 17:32:17 +0000 | [diff] [blame] | 1599 | #define NON_CANONICAL_TYPE(Class, Base) | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 1600 | #define DEPENDENT_TYPE(Class, Base) case Type::Class: | 
| David Blaikie | ab277d6 | 2013-07-13 21:08:03 +0000 | [diff] [blame] | 1601 | #define NON_CANONICAL_UNLESS_DEPENDENT_TYPE(Class, Base)                       \ | 
|  | 1602 | case Type::Class:                                                            \ | 
|  | 1603 | assert(!T->isDependentType() && "should not see dependent types here");      \ | 
|  | 1604 | return getTypeInfo(cast<Class##Type>(T)->desugar().getTypePtr()); | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 1605 | #include "clang/AST/TypeNodes.def" | 
| John McCall | 15547bb | 2011-06-28 16:49:23 +0000 | [diff] [blame] | 1606 | llvm_unreachable("Should not see dependent types"); | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 1607 |  | 
| Chris Lattner | 355332d | 2007-07-13 22:27:08 +0000 | [diff] [blame] | 1608 | case Type::FunctionNoProto: | 
|  | 1609 | case Type::FunctionProto: | 
| Douglas Gregor | ef462e6 | 2009-04-30 17:32:17 +0000 | [diff] [blame] | 1610 | // GCC extension: alignof(function) = 32 bits | 
|  | 1611 | Width = 0; | 
|  | 1612 | Align = 32; | 
|  | 1613 | break; | 
|  | 1614 |  | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 1615 | case Type::IncompleteArray: | 
| Steve Naroff | 5c13180 | 2007-08-30 01:06:46 +0000 | [diff] [blame] | 1616 | case Type::VariableArray: | 
| Douglas Gregor | ef462e6 | 2009-04-30 17:32:17 +0000 | [diff] [blame] | 1617 | Width = 0; | 
|  | 1618 | Align = getTypeAlign(cast<ArrayType>(T)->getElementType()); | 
|  | 1619 | break; | 
|  | 1620 |  | 
| Steve Naroff | 5c13180 | 2007-08-30 01:06:46 +0000 | [diff] [blame] | 1621 | case Type::ConstantArray: { | 
| Daniel Dunbar | bbc0af7 | 2008-11-08 05:48:37 +0000 | [diff] [blame] | 1622 | const ConstantArrayType *CAT = cast<ConstantArrayType>(T); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 1623 |  | 
| David Majnemer | 34b5749 | 2014-07-30 01:30:47 +0000 | [diff] [blame] | 1624 | TypeInfo EltInfo = getTypeInfo(CAT->getElementType()); | 
| Abramo Bagnara | d541a4c | 2011-12-13 11:23:52 +0000 | [diff] [blame] | 1625 | uint64_t Size = CAT->getSize().getZExtValue(); | 
| David Majnemer | 34b5749 | 2014-07-30 01:30:47 +0000 | [diff] [blame] | 1626 | assert((Size == 0 || EltInfo.Width <= (uint64_t)(-1) / Size) && | 
| Daniel Dunbar | c647587 | 2012-03-09 04:12:54 +0000 | [diff] [blame] | 1627 | "Overflow in array type bit size evaluation"); | 
| David Majnemer | 34b5749 | 2014-07-30 01:30:47 +0000 | [diff] [blame] | 1628 | Width = EltInfo.Width * Size; | 
|  | 1629 | Align = EltInfo.Align; | 
| Warren Hunt | 5ae586a | 2013-11-01 23:59:41 +0000 | [diff] [blame] | 1630 | if (!getTargetInfo().getCXXABI().isMicrosoft() || | 
|  | 1631 | getTargetInfo().getPointerWidth(0) == 64) | 
| Rui Ueyama | 83aa979 | 2016-01-14 21:00:27 +0000 | [diff] [blame] | 1632 | Width = llvm::alignTo(Width, Align); | 
| Chris Lattner | f2e101f | 2007-07-19 22:06:24 +0000 | [diff] [blame] | 1633 | break; | 
| Christopher Lamb | c5fafa2 | 2007-12-29 05:10:55 +0000 | [diff] [blame] | 1634 | } | 
| Nate Begeman | ce4d7fc | 2008-04-18 23:10:10 +0000 | [diff] [blame] | 1635 | case Type::ExtVector: | 
| Chris Lattner | f2e101f | 2007-07-19 22:06:24 +0000 | [diff] [blame] | 1636 | case Type::Vector: { | 
| Chris Lattner | 63d2b36 | 2009-10-22 05:17:15 +0000 | [diff] [blame] | 1637 | const VectorType *VT = cast<VectorType>(T); | 
| David Majnemer | 34b5749 | 2014-07-30 01:30:47 +0000 | [diff] [blame] | 1638 | TypeInfo EltInfo = getTypeInfo(VT->getElementType()); | 
|  | 1639 | Width = EltInfo.Width * VT->getNumElements(); | 
| Eli Friedman | 3df5efe | 2008-05-30 09:31:38 +0000 | [diff] [blame] | 1640 | Align = Width; | 
| Nate Begeman | b699c9b | 2009-01-18 06:42:49 +0000 | [diff] [blame] | 1641 | // If the alignment is not a power of 2, round up to the next power of 2. | 
|  | 1642 | // This happens for non-power-of-2 length vectors. | 
| Dan Gohman | ef78c8e | 2010-04-21 23:32:43 +0000 | [diff] [blame] | 1643 | if (Align & (Align-1)) { | 
| Chris Lattner | 63d2b36 | 2009-10-22 05:17:15 +0000 | [diff] [blame] | 1644 | Align = llvm::NextPowerOf2(Align); | 
| Rui Ueyama | 83aa979 | 2016-01-14 21:00:27 +0000 | [diff] [blame] | 1645 | Width = llvm::alignTo(Width, Align); | 
| Chris Lattner | 63d2b36 | 2009-10-22 05:17:15 +0000 | [diff] [blame] | 1646 | } | 
| Chad Rosier | cc40ea7 | 2012-07-13 23:57:43 +0000 | [diff] [blame] | 1647 | // Adjust the alignment based on the target max. | 
|  | 1648 | uint64_t TargetVectorAlign = Target->getMaxVectorAlign(); | 
|  | 1649 | if (TargetVectorAlign && TargetVectorAlign < Align) | 
|  | 1650 | Align = TargetVectorAlign; | 
| Chris Lattner | f2e101f | 2007-07-19 22:06:24 +0000 | [diff] [blame] | 1651 | break; | 
|  | 1652 | } | 
| Chris Lattner | 647fb22 | 2007-07-18 18:26:58 +0000 | [diff] [blame] | 1653 |  | 
| Chris Lattner | 7570e9c | 2008-03-08 08:52:55 +0000 | [diff] [blame] | 1654 | case Type::Builtin: | 
| Chris Lattner | 983a8bb | 2007-07-13 22:13:22 +0000 | [diff] [blame] | 1655 | switch (cast<BuiltinType>(T)->getKind()) { | 
| David Blaikie | 83d382b | 2011-09-23 05:06:16 +0000 | [diff] [blame] | 1656 | default: llvm_unreachable("Unknown builtin type!"); | 
| Chris Lattner | 4481b42 | 2007-07-14 01:29:45 +0000 | [diff] [blame] | 1657 | case BuiltinType::Void: | 
| Douglas Gregor | ef462e6 | 2009-04-30 17:32:17 +0000 | [diff] [blame] | 1658 | // GCC extension: alignof(void) = 8 bits. | 
|  | 1659 | Width = 0; | 
|  | 1660 | Align = 8; | 
|  | 1661 | break; | 
|  | 1662 |  | 
| Chris Lattner | 6a4f745 | 2007-12-19 19:23:28 +0000 | [diff] [blame] | 1663 | case BuiltinType::Bool: | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 1664 | Width = Target->getBoolWidth(); | 
|  | 1665 | Align = Target->getBoolAlign(); | 
| Chris Lattner | 6a4f745 | 2007-12-19 19:23:28 +0000 | [diff] [blame] | 1666 | break; | 
| Chris Lattner | 355332d | 2007-07-13 22:27:08 +0000 | [diff] [blame] | 1667 | case BuiltinType::Char_S: | 
|  | 1668 | case BuiltinType::Char_U: | 
|  | 1669 | case BuiltinType::UChar: | 
| Chris Lattner | 6a4f745 | 2007-12-19 19:23:28 +0000 | [diff] [blame] | 1670 | case BuiltinType::SChar: | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 1671 | Width = Target->getCharWidth(); | 
|  | 1672 | Align = Target->getCharAlign(); | 
| Chris Lattner | 6a4f745 | 2007-12-19 19:23:28 +0000 | [diff] [blame] | 1673 | break; | 
| Chris Lattner | ad3467e | 2010-12-25 23:25:43 +0000 | [diff] [blame] | 1674 | case BuiltinType::WChar_S: | 
|  | 1675 | case BuiltinType::WChar_U: | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 1676 | Width = Target->getWCharWidth(); | 
|  | 1677 | Align = Target->getWCharAlign(); | 
| Argyrios Kyrtzidis | 40e9e48 | 2008-08-09 16:51:54 +0000 | [diff] [blame] | 1678 | break; | 
| Alisdair Meredith | a9ad47d | 2009-07-14 06:30:34 +0000 | [diff] [blame] | 1679 | case BuiltinType::Char16: | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 1680 | Width = Target->getChar16Width(); | 
|  | 1681 | Align = Target->getChar16Align(); | 
| Alisdair Meredith | a9ad47d | 2009-07-14 06:30:34 +0000 | [diff] [blame] | 1682 | break; | 
|  | 1683 | case BuiltinType::Char32: | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 1684 | Width = Target->getChar32Width(); | 
|  | 1685 | Align = Target->getChar32Align(); | 
| Alisdair Meredith | a9ad47d | 2009-07-14 06:30:34 +0000 | [diff] [blame] | 1686 | break; | 
| Chris Lattner | 355332d | 2007-07-13 22:27:08 +0000 | [diff] [blame] | 1687 | case BuiltinType::UShort: | 
| Chris Lattner | 6a4f745 | 2007-12-19 19:23:28 +0000 | [diff] [blame] | 1688 | case BuiltinType::Short: | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 1689 | Width = Target->getShortWidth(); | 
|  | 1690 | Align = Target->getShortAlign(); | 
| Chris Lattner | 6a4f745 | 2007-12-19 19:23:28 +0000 | [diff] [blame] | 1691 | break; | 
| Chris Lattner | 355332d | 2007-07-13 22:27:08 +0000 | [diff] [blame] | 1692 | case BuiltinType::UInt: | 
| Chris Lattner | 6a4f745 | 2007-12-19 19:23:28 +0000 | [diff] [blame] | 1693 | case BuiltinType::Int: | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 1694 | Width = Target->getIntWidth(); | 
|  | 1695 | Align = Target->getIntAlign(); | 
| Chris Lattner | 6a4f745 | 2007-12-19 19:23:28 +0000 | [diff] [blame] | 1696 | break; | 
| Chris Lattner | 355332d | 2007-07-13 22:27:08 +0000 | [diff] [blame] | 1697 | case BuiltinType::ULong: | 
| Chris Lattner | 6a4f745 | 2007-12-19 19:23:28 +0000 | [diff] [blame] | 1698 | case BuiltinType::Long: | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 1699 | Width = Target->getLongWidth(); | 
|  | 1700 | Align = Target->getLongAlign(); | 
| Chris Lattner | 6a4f745 | 2007-12-19 19:23:28 +0000 | [diff] [blame] | 1701 | break; | 
| Chris Lattner | 355332d | 2007-07-13 22:27:08 +0000 | [diff] [blame] | 1702 | case BuiltinType::ULongLong: | 
| Chris Lattner | 6a4f745 | 2007-12-19 19:23:28 +0000 | [diff] [blame] | 1703 | case BuiltinType::LongLong: | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 1704 | Width = Target->getLongLongWidth(); | 
|  | 1705 | Align = Target->getLongLongAlign(); | 
| Chris Lattner | 6a4f745 | 2007-12-19 19:23:28 +0000 | [diff] [blame] | 1706 | break; | 
| Chris Lattner | 0a415ec | 2009-04-30 02:55:13 +0000 | [diff] [blame] | 1707 | case BuiltinType::Int128: | 
|  | 1708 | case BuiltinType::UInt128: | 
|  | 1709 | Width = 128; | 
|  | 1710 | Align = 128; // int128_t is 128-bit aligned on all targets. | 
|  | 1711 | break; | 
| Anton Korobeynikov | f0c267e | 2011-10-14 23:23:15 +0000 | [diff] [blame] | 1712 | case BuiltinType::Half: | 
|  | 1713 | Width = Target->getHalfWidth(); | 
|  | 1714 | Align = Target->getHalfAlign(); | 
|  | 1715 | break; | 
| Chris Lattner | 6a4f745 | 2007-12-19 19:23:28 +0000 | [diff] [blame] | 1716 | case BuiltinType::Float: | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 1717 | Width = Target->getFloatWidth(); | 
|  | 1718 | Align = Target->getFloatAlign(); | 
| Chris Lattner | 6a4f745 | 2007-12-19 19:23:28 +0000 | [diff] [blame] | 1719 | break; | 
|  | 1720 | case BuiltinType::Double: | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 1721 | Width = Target->getDoubleWidth(); | 
|  | 1722 | Align = Target->getDoubleAlign(); | 
| Chris Lattner | 6a4f745 | 2007-12-19 19:23:28 +0000 | [diff] [blame] | 1723 | break; | 
|  | 1724 | case BuiltinType::LongDouble: | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 1725 | Width = Target->getLongDoubleWidth(); | 
|  | 1726 | Align = Target->getLongDoubleAlign(); | 
| Chris Lattner | 6a4f745 | 2007-12-19 19:23:28 +0000 | [diff] [blame] | 1727 | break; | 
| Nemanja Ivanovic | bb1ea2d | 2016-05-09 08:52:33 +0000 | [diff] [blame] | 1728 | case BuiltinType::Float128: | 
|  | 1729 | Width = Target->getFloat128Width(); | 
|  | 1730 | Align = Target->getFloat128Align(); | 
|  | 1731 | break; | 
| Sebastian Redl | 576fd42 | 2009-05-10 18:38:11 +0000 | [diff] [blame] | 1732 | case BuiltinType::NullPtr: | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 1733 | Width = Target->getPointerWidth(0); // C++ 3.9.1p11: sizeof(nullptr_t) | 
|  | 1734 | Align = Target->getPointerAlign(0); //   == sizeof(void*) | 
| Sebastian Redl | a81b0b7 | 2009-05-27 19:34:06 +0000 | [diff] [blame] | 1735 | break; | 
| Fariborz Jahanian | 9c6a39e | 2010-08-02 18:03:20 +0000 | [diff] [blame] | 1736 | case BuiltinType::ObjCId: | 
|  | 1737 | case BuiltinType::ObjCClass: | 
|  | 1738 | case BuiltinType::ObjCSel: | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 1739 | Width = Target->getPointerWidth(0); | 
|  | 1740 | Align = Target->getPointerAlign(0); | 
| Fariborz Jahanian | 9c6a39e | 2010-08-02 18:03:20 +0000 | [diff] [blame] | 1741 | break; | 
| Yaxun Liu | 0bc4b2d | 2016-07-28 19:26:30 +0000 | [diff] [blame] | 1742 | case BuiltinType::OCLSampler: { | 
|  | 1743 | auto AS = getTargetAddressSpace(LangAS::opencl_constant); | 
|  | 1744 | Width = Target->getPointerWidth(AS); | 
|  | 1745 | Align = Target->getPointerAlign(AS); | 
| Guy Benyei | 6105419 | 2013-02-07 10:55:47 +0000 | [diff] [blame] | 1746 | break; | 
| Yaxun Liu | 0bc4b2d | 2016-07-28 19:26:30 +0000 | [diff] [blame] | 1747 | } | 
| Guy Benyei | 1b4fb3e | 2013-01-20 12:31:11 +0000 | [diff] [blame] | 1748 | case BuiltinType::OCLEvent: | 
| Alexey Bader | 9c8453f | 2015-09-15 11:18:52 +0000 | [diff] [blame] | 1749 | case BuiltinType::OCLClkEvent: | 
|  | 1750 | case BuiltinType::OCLQueue: | 
|  | 1751 | case BuiltinType::OCLNDRange: | 
|  | 1752 | case BuiltinType::OCLReserveID: | 
| Guy Benyei | d8a08ea | 2012-12-18 14:38:23 +0000 | [diff] [blame] | 1753 | // Currently these types are pointers to opaque types. | 
|  | 1754 | Width = Target->getPointerWidth(0); | 
|  | 1755 | Align = Target->getPointerAlign(0); | 
|  | 1756 | break; | 
| Yaxun Liu | 99444cb | 2016-08-03 20:38:06 +0000 | [diff] [blame] | 1757 | #define IMAGE_TYPE(ImgType, Id, SingletonId, Access, Suffix) \ | 
|  | 1758 | case BuiltinType::Id: | 
|  | 1759 | #include "clang/Basic/OpenCLImageTypes.def" | 
|  | 1760 | { | 
|  | 1761 | auto AS = getTargetAddressSpace(Target->getOpenCLImageAddrSpace()); | 
|  | 1762 | Width = Target->getPointerWidth(AS); | 
|  | 1763 | Align = Target->getPointerAlign(AS); | 
|  | 1764 | } | 
| Chris Lattner | 983a8bb | 2007-07-13 22:13:22 +0000 | [diff] [blame] | 1765 | } | 
| Chris Lattner | 48f84b8 | 2007-07-15 23:46:53 +0000 | [diff] [blame] | 1766 | break; | 
| Steve Naroff | fb4330f | 2009-06-17 22:40:22 +0000 | [diff] [blame] | 1767 | case Type::ObjCObjectPointer: | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 1768 | Width = Target->getPointerWidth(0); | 
|  | 1769 | Align = Target->getPointerAlign(0); | 
| Chris Lattner | 6a4f745 | 2007-12-19 19:23:28 +0000 | [diff] [blame] | 1770 | break; | 
| Steve Naroff | 921a45c | 2008-09-24 15:05:44 +0000 | [diff] [blame] | 1771 | case Type::BlockPointer: { | 
| Peter Collingbourne | 599cb8e | 2011-03-18 22:38:29 +0000 | [diff] [blame] | 1772 | unsigned AS = getTargetAddressSpace( | 
|  | 1773 | cast<BlockPointerType>(T)->getPointeeType()); | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 1774 | Width = Target->getPointerWidth(AS); | 
|  | 1775 | Align = Target->getPointerAlign(AS); | 
| Steve Naroff | 921a45c | 2008-09-24 15:05:44 +0000 | [diff] [blame] | 1776 | break; | 
|  | 1777 | } | 
| Sebastian Redl | 22e2e5c | 2009-11-23 17:18:46 +0000 | [diff] [blame] | 1778 | case Type::LValueReference: | 
|  | 1779 | case Type::RValueReference: { | 
|  | 1780 | // alignof and sizeof should never enter this code path here, so we go | 
|  | 1781 | // the pointer route. | 
| Peter Collingbourne | 599cb8e | 2011-03-18 22:38:29 +0000 | [diff] [blame] | 1782 | unsigned AS = getTargetAddressSpace( | 
|  | 1783 | cast<ReferenceType>(T)->getPointeeType()); | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 1784 | Width = Target->getPointerWidth(AS); | 
|  | 1785 | Align = Target->getPointerAlign(AS); | 
| Sebastian Redl | 22e2e5c | 2009-11-23 17:18:46 +0000 | [diff] [blame] | 1786 | break; | 
|  | 1787 | } | 
| Chris Lattner | 2dca6ff | 2008-03-08 08:34:58 +0000 | [diff] [blame] | 1788 | case Type::Pointer: { | 
| Peter Collingbourne | 599cb8e | 2011-03-18 22:38:29 +0000 | [diff] [blame] | 1789 | unsigned AS = getTargetAddressSpace(cast<PointerType>(T)->getPointeeType()); | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 1790 | Width = Target->getPointerWidth(AS); | 
|  | 1791 | Align = Target->getPointerAlign(AS); | 
| Chris Lattner | 2dca6ff | 2008-03-08 08:34:58 +0000 | [diff] [blame] | 1792 | break; | 
|  | 1793 | } | 
| Sebastian Redl | 9ed6efd | 2009-01-24 21:16:55 +0000 | [diff] [blame] | 1794 | case Type::MemberPointer: { | 
| Charles Davis | 53c59df | 2010-08-16 03:33:14 +0000 | [diff] [blame] | 1795 | const MemberPointerType *MPT = cast<MemberPointerType>(T); | 
| Benjamin Kramer | 867ea1d | 2014-03-02 13:01:17 +0000 | [diff] [blame] | 1796 | std::tie(Width, Align) = ABI->getMemberPointerWidthAndAlign(MPT); | 
| Anders Carlsson | 32440a0 | 2009-05-17 02:06:04 +0000 | [diff] [blame] | 1797 | break; | 
| Sebastian Redl | 9ed6efd | 2009-01-24 21:16:55 +0000 | [diff] [blame] | 1798 | } | 
| Chris Lattner | 647fb22 | 2007-07-18 18:26:58 +0000 | [diff] [blame] | 1799 | case Type::Complex: { | 
|  | 1800 | // Complex types have the same alignment as their elements, but twice the | 
|  | 1801 | // size. | 
| David Majnemer | 34b5749 | 2014-07-30 01:30:47 +0000 | [diff] [blame] | 1802 | TypeInfo EltInfo = getTypeInfo(cast<ComplexType>(T)->getElementType()); | 
|  | 1803 | Width = EltInfo.Width * 2; | 
|  | 1804 | Align = EltInfo.Align; | 
| Chris Lattner | 647fb22 | 2007-07-18 18:26:58 +0000 | [diff] [blame] | 1805 | break; | 
|  | 1806 | } | 
| John McCall | 8b07ec2 | 2010-05-15 11:32:37 +0000 | [diff] [blame] | 1807 | case Type::ObjCObject: | 
|  | 1808 | return getTypeInfo(cast<ObjCObjectType>(T)->getBaseType().getTypePtr()); | 
| Reid Kleckner | 0503a87 | 2013-12-05 01:23:43 +0000 | [diff] [blame] | 1809 | case Type::Adjusted: | 
| Reid Kleckner | 8a36502 | 2013-06-24 17:51:48 +0000 | [diff] [blame] | 1810 | case Type::Decayed: | 
| Reid Kleckner | 0503a87 | 2013-12-05 01:23:43 +0000 | [diff] [blame] | 1811 | return getTypeInfo(cast<AdjustedType>(T)->getAdjustedType().getTypePtr()); | 
| Devang Patel | dbb7263 | 2008-06-04 21:54:36 +0000 | [diff] [blame] | 1812 | case Type::ObjCInterface: { | 
| Daniel Dunbar | bbc0af7 | 2008-11-08 05:48:37 +0000 | [diff] [blame] | 1813 | const ObjCInterfaceType *ObjCI = cast<ObjCInterfaceType>(T); | 
| Devang Patel | dbb7263 | 2008-06-04 21:54:36 +0000 | [diff] [blame] | 1814 | const ASTRecordLayout &Layout = getASTObjCInterfaceLayout(ObjCI->getDecl()); | 
| Ken Dyck | b0fcc59 | 2011-02-11 01:54:29 +0000 | [diff] [blame] | 1815 | Width = toBits(Layout.getSize()); | 
| Ken Dyck | 7ad11e7 | 2011-02-15 02:32:40 +0000 | [diff] [blame] | 1816 | Align = toBits(Layout.getAlignment()); | 
| Devang Patel | dbb7263 | 2008-06-04 21:54:36 +0000 | [diff] [blame] | 1817 | break; | 
|  | 1818 | } | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 1819 | case Type::Record: | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 1820 | case Type::Enum: { | 
| Daniel Dunbar | bbc0af7 | 2008-11-08 05:48:37 +0000 | [diff] [blame] | 1821 | const TagType *TT = cast<TagType>(T); | 
|  | 1822 |  | 
|  | 1823 | if (TT->getDecl()->isInvalidDecl()) { | 
| Douglas Gregor | 7f97189 | 2011-04-20 17:29:44 +0000 | [diff] [blame] | 1824 | Width = 8; | 
|  | 1825 | Align = 8; | 
| Chris Lattner | 572100b | 2008-08-09 21:35:13 +0000 | [diff] [blame] | 1826 | break; | 
|  | 1827 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 1828 |  | 
| David Majnemer | 475b25e | 2015-01-21 10:54:38 +0000 | [diff] [blame] | 1829 | if (const EnumType *ET = dyn_cast<EnumType>(TT)) { | 
|  | 1830 | const EnumDecl *ED = ET->getDecl(); | 
|  | 1831 | TypeInfo Info = | 
|  | 1832 | getTypeInfo(ED->getIntegerType()->getUnqualifiedDesugaredType()); | 
|  | 1833 | if (unsigned AttrAlign = ED->getMaxAlignment()) { | 
|  | 1834 | Info.Align = AttrAlign; | 
|  | 1835 | Info.AlignIsRequired = true; | 
|  | 1836 | } | 
|  | 1837 | return Info; | 
|  | 1838 | } | 
| Chris Lattner | 8b23c25 | 2008-04-06 22:05:18 +0000 | [diff] [blame] | 1839 |  | 
| Daniel Dunbar | bbc0af7 | 2008-11-08 05:48:37 +0000 | [diff] [blame] | 1840 | const RecordType *RT = cast<RecordType>(TT); | 
| David Majnemer | 5821ff7 | 2015-02-03 08:49:29 +0000 | [diff] [blame] | 1841 | const RecordDecl *RD = RT->getDecl(); | 
|  | 1842 | const ASTRecordLayout &Layout = getASTRecordLayout(RD); | 
| Ken Dyck | b0fcc59 | 2011-02-11 01:54:29 +0000 | [diff] [blame] | 1843 | Width = toBits(Layout.getSize()); | 
| Ken Dyck | 7ad11e7 | 2011-02-15 02:32:40 +0000 | [diff] [blame] | 1844 | Align = toBits(Layout.getAlignment()); | 
| David Majnemer | 5821ff7 | 2015-02-03 08:49:29 +0000 | [diff] [blame] | 1845 | AlignIsRequired = RD->hasAttr<AlignedAttr>(); | 
| Chris Lattner | 49a953a | 2007-07-23 22:46:22 +0000 | [diff] [blame] | 1846 | break; | 
| Chris Lattner | 983a8bb | 2007-07-13 22:13:22 +0000 | [diff] [blame] | 1847 | } | 
| Douglas Gregor | dc572a3 | 2009-03-30 22:58:21 +0000 | [diff] [blame] | 1848 |  | 
| Chris Lattner | 63d2b36 | 2009-10-22 05:17:15 +0000 | [diff] [blame] | 1849 | case Type::SubstTemplateTypeParm: | 
| John McCall | cebee16 | 2009-10-18 09:09:24 +0000 | [diff] [blame] | 1850 | return getTypeInfo(cast<SubstTemplateTypeParmType>(T)-> | 
|  | 1851 | getReplacementType().getTypePtr()); | 
| John McCall | cebee16 | 2009-10-18 09:09:24 +0000 | [diff] [blame] | 1852 |  | 
| Richard Smith | 30482bc | 2011-02-20 03:19:35 +0000 | [diff] [blame] | 1853 | case Type::Auto: { | 
|  | 1854 | const AutoType *A = cast<AutoType>(T); | 
| Richard Smith | 27d807c | 2013-04-30 13:56:41 +0000 | [diff] [blame] | 1855 | assert(!A->getDeducedType().isNull() && | 
|  | 1856 | "cannot request the size of an undeduced or dependent auto type"); | 
| Matt Beaumont-Gay | 7a24210e | 2011-02-22 20:00:16 +0000 | [diff] [blame] | 1857 | return getTypeInfo(A->getDeducedType().getTypePtr()); | 
| Richard Smith | 30482bc | 2011-02-20 03:19:35 +0000 | [diff] [blame] | 1858 | } | 
|  | 1859 |  | 
| Abramo Bagnara | 924a8f3 | 2010-12-10 16:29:40 +0000 | [diff] [blame] | 1860 | case Type::Paren: | 
|  | 1861 | return getTypeInfo(cast<ParenType>(T)->getInnerType().getTypePtr()); | 
|  | 1862 |  | 
| Douglas Gregor | ef462e6 | 2009-04-30 17:32:17 +0000 | [diff] [blame] | 1863 | case Type::Typedef: { | 
| Richard Smith | dda56e4 | 2011-04-15 14:24:37 +0000 | [diff] [blame] | 1864 | const TypedefNameDecl *Typedef = cast<TypedefType>(T)->getDecl(); | 
| David Majnemer | 34b5749 | 2014-07-30 01:30:47 +0000 | [diff] [blame] | 1865 | TypeInfo Info = getTypeInfo(Typedef->getUnderlyingType().getTypePtr()); | 
| Chris Lattner | 29eb47b | 2011-02-19 22:55:41 +0000 | [diff] [blame] | 1866 | // If the typedef has an aligned attribute on it, it overrides any computed | 
|  | 1867 | // alignment we have.  This violates the GCC documentation (which says that | 
|  | 1868 | // attribute(aligned) can only round up) but matches its implementation. | 
| David Majnemer | 34b5749 | 2014-07-30 01:30:47 +0000 | [diff] [blame] | 1869 | if (unsigned AttrAlign = Typedef->getMaxAlignment()) { | 
| Chris Lattner | 29eb47b | 2011-02-19 22:55:41 +0000 | [diff] [blame] | 1870 | Align = AttrAlign; | 
| David Majnemer | 34b5749 | 2014-07-30 01:30:47 +0000 | [diff] [blame] | 1871 | AlignIsRequired = true; | 
| David Majnemer | 37bffb6 | 2014-08-04 05:11:01 +0000 | [diff] [blame] | 1872 | } else { | 
| David Majnemer | 34b5749 | 2014-07-30 01:30:47 +0000 | [diff] [blame] | 1873 | Align = Info.Align; | 
| David Majnemer | 37bffb6 | 2014-08-04 05:11:01 +0000 | [diff] [blame] | 1874 | AlignIsRequired = Info.AlignIsRequired; | 
|  | 1875 | } | 
| David Majnemer | 34b5749 | 2014-07-30 01:30:47 +0000 | [diff] [blame] | 1876 | Width = Info.Width; | 
| Douglas Gregor | dc572a3 | 2009-03-30 22:58:21 +0000 | [diff] [blame] | 1877 | break; | 
| Chris Lattner | 8b23c25 | 2008-04-06 22:05:18 +0000 | [diff] [blame] | 1878 | } | 
| Douglas Gregor | ef462e6 | 2009-04-30 17:32:17 +0000 | [diff] [blame] | 1879 |  | 
| Abramo Bagnara | 6150c88 | 2010-05-11 21:36:43 +0000 | [diff] [blame] | 1880 | case Type::Elaborated: | 
|  | 1881 | return getTypeInfo(cast<ElaboratedType>(T)->getNamedType().getTypePtr()); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 1882 |  | 
| John McCall | 8190451 | 2011-01-06 01:58:22 +0000 | [diff] [blame] | 1883 | case Type::Attributed: | 
|  | 1884 | return getTypeInfo( | 
|  | 1885 | cast<AttributedType>(T)->getEquivalentType().getTypePtr()); | 
|  | 1886 |  | 
| Eli Friedman | 0dfb889 | 2011-10-06 23:00:33 +0000 | [diff] [blame] | 1887 | case Type::Atomic: { | 
| John McCall | a8ec7eb | 2013-03-07 21:37:17 +0000 | [diff] [blame] | 1888 | // Start with the base type information. | 
| David Majnemer | 34b5749 | 2014-07-30 01:30:47 +0000 | [diff] [blame] | 1889 | TypeInfo Info = getTypeInfo(cast<AtomicType>(T)->getValueType()); | 
|  | 1890 | Width = Info.Width; | 
|  | 1891 | Align = Info.Align; | 
| John McCall | a8ec7eb | 2013-03-07 21:37:17 +0000 | [diff] [blame] | 1892 |  | 
|  | 1893 | // If the size of the type doesn't exceed the platform's max | 
|  | 1894 | // atomic promotion width, make the size and alignment more | 
|  | 1895 | // favorable to atomic operations: | 
|  | 1896 | if (Width != 0 && Width <= Target->getMaxAtomicPromoteWidth()) { | 
|  | 1897 | // Round the size up to a power of 2. | 
|  | 1898 | if (!llvm::isPowerOf2_64(Width)) | 
|  | 1899 | Width = llvm::NextPowerOf2(Width); | 
|  | 1900 |  | 
|  | 1901 | // Set the alignment equal to the size. | 
| Eli Friedman | 4b72fdd | 2011-10-14 20:59:01 +0000 | [diff] [blame] | 1902 | Align = static_cast<unsigned>(Width); | 
|  | 1903 | } | 
| Eli Friedman | 0dfb889 | 2011-10-06 23:00:33 +0000 | [diff] [blame] | 1904 | } | 
| Xiuli Pan | 9c14e28 | 2016-01-09 12:53:17 +0000 | [diff] [blame] | 1905 | break; | 
|  | 1906 |  | 
|  | 1907 | case Type::Pipe: { | 
|  | 1908 | TypeInfo Info = getTypeInfo(cast<PipeType>(T)->getElementType()); | 
|  | 1909 | Width = Info.Width; | 
|  | 1910 | Align = Info.Align; | 
|  | 1911 | } | 
| Eli Friedman | 0dfb889 | 2011-10-06 23:00:33 +0000 | [diff] [blame] | 1912 |  | 
| Douglas Gregor | ef462e6 | 2009-04-30 17:32:17 +0000 | [diff] [blame] | 1913 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 1914 |  | 
| Eli Friedman | 4b72fdd | 2011-10-14 20:59:01 +0000 | [diff] [blame] | 1915 | assert(llvm::isPowerOf2_32(Align) && "Alignment must be power of 2"); | 
| David Majnemer | 34b5749 | 2014-07-30 01:30:47 +0000 | [diff] [blame] | 1916 | return TypeInfo(Width, Align, AlignIsRequired); | 
| Chris Lattner | 983a8bb | 2007-07-13 22:13:22 +0000 | [diff] [blame] | 1917 | } | 
|  | 1918 |  | 
| Alexey Bataev | 0039651 | 2015-07-02 03:40:19 +0000 | [diff] [blame] | 1919 | unsigned ASTContext::getOpenMPDefaultSimdAlign(QualType T) const { | 
|  | 1920 | unsigned SimdAlign = getTargetInfo().getSimdDefaultAlign(); | 
|  | 1921 | // Target ppc64 with QPX: simd default alignment for pointer to double is 32. | 
|  | 1922 | if ((getTargetInfo().getTriple().getArch() == llvm::Triple::ppc64 || | 
|  | 1923 | getTargetInfo().getTriple().getArch() == llvm::Triple::ppc64le) && | 
|  | 1924 | getTargetInfo().getABI() == "elfv1-qpx" && | 
|  | 1925 | T->isSpecificBuiltinType(BuiltinType::Double)) | 
|  | 1926 | SimdAlign = 256; | 
|  | 1927 | return SimdAlign; | 
|  | 1928 | } | 
|  | 1929 |  | 
| Ken Dyck | cc56c54 | 2011-01-15 18:38:59 +0000 | [diff] [blame] | 1930 | /// toCharUnitsFromBits - Convert a size in bits to a size in characters. | 
|  | 1931 | CharUnits ASTContext::toCharUnitsFromBits(int64_t BitSize) const { | 
|  | 1932 | return CharUnits::fromQuantity(BitSize / getCharWidth()); | 
|  | 1933 | } | 
|  | 1934 |  | 
| Ken Dyck | b0fcc59 | 2011-02-11 01:54:29 +0000 | [diff] [blame] | 1935 | /// toBits - Convert a size in characters to a size in characters. | 
|  | 1936 | int64_t ASTContext::toBits(CharUnits CharSize) const { | 
|  | 1937 | return CharSize.getQuantity() * getCharWidth(); | 
|  | 1938 | } | 
|  | 1939 |  | 
| Ken Dyck | 8c89d59 | 2009-12-22 14:23:30 +0000 | [diff] [blame] | 1940 | /// getTypeSizeInChars - Return the size of the specified type, in characters. | 
|  | 1941 | /// This method does not work on incomplete types. | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 1942 | CharUnits ASTContext::getTypeSizeInChars(QualType T) const { | 
| Richard Trieu | 04d2d94 | 2013-05-14 21:59:17 +0000 | [diff] [blame] | 1943 | return getTypeInfoInChars(T).first; | 
| Ken Dyck | 8c89d59 | 2009-12-22 14:23:30 +0000 | [diff] [blame] | 1944 | } | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 1945 | CharUnits ASTContext::getTypeSizeInChars(const Type *T) const { | 
| Richard Trieu | 04d2d94 | 2013-05-14 21:59:17 +0000 | [diff] [blame] | 1946 | return getTypeInfoInChars(T).first; | 
| Ken Dyck | 8c89d59 | 2009-12-22 14:23:30 +0000 | [diff] [blame] | 1947 | } | 
|  | 1948 |  | 
| Ken Dyck | a6046ab | 2010-01-26 17:25:18 +0000 | [diff] [blame] | 1949 | /// getTypeAlignInChars - Return the ABI-specified alignment of a type, in | 
| Ken Dyck | 24d28d6 | 2010-01-26 17:22:55 +0000 | [diff] [blame] | 1950 | /// characters. This method does not work on incomplete types. | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 1951 | CharUnits ASTContext::getTypeAlignInChars(QualType T) const { | 
| Ken Dyck | cc56c54 | 2011-01-15 18:38:59 +0000 | [diff] [blame] | 1952 | return toCharUnitsFromBits(getTypeAlign(T)); | 
| Ken Dyck | 24d28d6 | 2010-01-26 17:22:55 +0000 | [diff] [blame] | 1953 | } | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 1954 | CharUnits ASTContext::getTypeAlignInChars(const Type *T) const { | 
| Ken Dyck | cc56c54 | 2011-01-15 18:38:59 +0000 | [diff] [blame] | 1955 | return toCharUnitsFromBits(getTypeAlign(T)); | 
| Ken Dyck | 24d28d6 | 2010-01-26 17:22:55 +0000 | [diff] [blame] | 1956 | } | 
|  | 1957 |  | 
| Chris Lattner | a3402cd | 2009-01-27 18:08:34 +0000 | [diff] [blame] | 1958 | /// getPreferredTypeAlign - Return the "preferred" alignment of the specified | 
|  | 1959 | /// type for the current target in bits.  This can be different than the ABI | 
|  | 1960 | /// alignment in cases where it is beneficial for performance to overalign | 
|  | 1961 | /// a data type. | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 1962 | unsigned ASTContext::getPreferredTypeAlign(const Type *T) const { | 
| David Majnemer | 34b5749 | 2014-07-30 01:30:47 +0000 | [diff] [blame] | 1963 | TypeInfo TI = getTypeInfo(T); | 
|  | 1964 | unsigned ABIAlign = TI.Align; | 
| Eli Friedman | 7ab0957 | 2009-05-25 21:27:19 +0000 | [diff] [blame] | 1965 |  | 
| David Majnemer | e154456 | 2015-04-24 01:25:05 +0000 | [diff] [blame] | 1966 | T = T->getBaseElementTypeUnsafe(); | 
|  | 1967 |  | 
|  | 1968 | // The preferred alignment of member pointers is that of a pointer. | 
|  | 1969 | if (T->isMemberPointerType()) | 
|  | 1970 | return getPreferredTypeAlign(getPointerDiffType().getTypePtr()); | 
|  | 1971 |  | 
| Andrey Turetskiy | db6655f | 2016-02-10 11:58:46 +0000 | [diff] [blame] | 1972 | if (!Target->allowsLargerPreferedTypeAlignment()) | 
|  | 1973 | return ABIAlign; | 
| Robert Lytton | eaf6f36 | 2013-11-12 10:09:34 +0000 | [diff] [blame] | 1974 |  | 
| Eli Friedman | 7ab0957 | 2009-05-25 21:27:19 +0000 | [diff] [blame] | 1975 | // Double and long long should be naturally aligned if possible. | 
| David Majnemer | 1d4db8f | 2014-02-24 23:43:27 +0000 | [diff] [blame] | 1976 | if (const ComplexType *CT = T->getAs<ComplexType>()) | 
| Eli Friedman | 7ab0957 | 2009-05-25 21:27:19 +0000 | [diff] [blame] | 1977 | T = CT->getElementType().getTypePtr(); | 
| David Majnemer | 475b25e | 2015-01-21 10:54:38 +0000 | [diff] [blame] | 1978 | if (const EnumType *ET = T->getAs<EnumType>()) | 
|  | 1979 | T = ET->getDecl()->getIntegerType().getTypePtr(); | 
| Eli Friedman | 7ab0957 | 2009-05-25 21:27:19 +0000 | [diff] [blame] | 1980 | if (T->isSpecificBuiltinType(BuiltinType::Double) || | 
| Chad Rosier | b57321a | 2012-03-21 20:20:47 +0000 | [diff] [blame] | 1981 | T->isSpecificBuiltinType(BuiltinType::LongLong) || | 
|  | 1982 | T->isSpecificBuiltinType(BuiltinType::ULongLong)) | 
| David Majnemer | 8b6bd57 | 2014-02-24 23:34:17 +0000 | [diff] [blame] | 1983 | // Don't increase the alignment if an alignment attribute was specified on a | 
|  | 1984 | // typedef declaration. | 
| David Majnemer | 34b5749 | 2014-07-30 01:30:47 +0000 | [diff] [blame] | 1985 | if (!TI.AlignIsRequired) | 
| David Majnemer | 8b6bd57 | 2014-02-24 23:34:17 +0000 | [diff] [blame] | 1986 | return std::max(ABIAlign, (unsigned)getTypeSize(T)); | 
| Eli Friedman | 7ab0957 | 2009-05-25 21:27:19 +0000 | [diff] [blame] | 1987 |  | 
| Chris Lattner | a3402cd | 2009-01-27 18:08:34 +0000 | [diff] [blame] | 1988 | return ABIAlign; | 
|  | 1989 | } | 
|  | 1990 |  | 
| Ulrich Weigand | ca3cb7f | 2015-04-21 17:29:35 +0000 | [diff] [blame] | 1991 | /// getTargetDefaultAlignForAttributeAligned - Return the default alignment | 
|  | 1992 | /// for __attribute__((aligned)) on this target, to be used if no alignment | 
|  | 1993 | /// value is specified. | 
| Eugene Zelenko | d4304d2 | 2015-11-04 21:37:17 +0000 | [diff] [blame] | 1994 | unsigned ASTContext::getTargetDefaultAlignForAttributeAligned() const { | 
| Ulrich Weigand | ca3cb7f | 2015-04-21 17:29:35 +0000 | [diff] [blame] | 1995 | return getTargetInfo().getDefaultAlignForAttributeAligned(); | 
|  | 1996 | } | 
|  | 1997 |  | 
| Ulrich Weigand | fa80642 | 2013-05-06 16:23:57 +0000 | [diff] [blame] | 1998 | /// getAlignOfGlobalVar - Return the alignment in bits that should be given | 
|  | 1999 | /// to a global variable of the specified type. | 
|  | 2000 | unsigned ASTContext::getAlignOfGlobalVar(QualType T) const { | 
|  | 2001 | return std::max(getTypeAlign(T), getTargetInfo().getMinGlobalAlign()); | 
|  | 2002 | } | 
|  | 2003 |  | 
|  | 2004 | /// getAlignOfGlobalVarInChars - Return the alignment in characters that | 
|  | 2005 | /// should be given to a global variable of the specified type. | 
|  | 2006 | CharUnits ASTContext::getAlignOfGlobalVarInChars(QualType T) const { | 
|  | 2007 | return toCharUnitsFromBits(getAlignOfGlobalVar(T)); | 
|  | 2008 | } | 
|  | 2009 |  | 
| David Majnemer | 08ef2ba | 2015-06-23 20:34:18 +0000 | [diff] [blame] | 2010 | CharUnits ASTContext::getOffsetOfBaseWithVBPtr(const CXXRecordDecl *RD) const { | 
|  | 2011 | CharUnits Offset = CharUnits::Zero(); | 
|  | 2012 | const ASTRecordLayout *Layout = &getASTRecordLayout(RD); | 
|  | 2013 | while (const CXXRecordDecl *Base = Layout->getBaseSharingVBPtr()) { | 
|  | 2014 | Offset += Layout->getBaseClassOffset(Base); | 
|  | 2015 | Layout = &getASTRecordLayout(Base); | 
|  | 2016 | } | 
|  | 2017 | return Offset; | 
|  | 2018 | } | 
|  | 2019 |  | 
| Fariborz Jahanian | a50b3a2 | 2010-08-20 21:21:08 +0000 | [diff] [blame] | 2020 | /// DeepCollectObjCIvars - | 
|  | 2021 | /// This routine first collects all declared, but not synthesized, ivars in | 
|  | 2022 | /// super class and then collects all ivars, including those synthesized for | 
|  | 2023 | /// current class. This routine is used for implementation of current class | 
|  | 2024 | /// when all ivars, declared and synthesized are known. | 
| Fariborz Jahanian | 0f44d81 | 2009-05-12 18:14:29 +0000 | [diff] [blame] | 2025 | /// | 
| Fariborz Jahanian | a50b3a2 | 2010-08-20 21:21:08 +0000 | [diff] [blame] | 2026 | void ASTContext::DeepCollectObjCIvars(const ObjCInterfaceDecl *OI, | 
|  | 2027 | bool leafClass, | 
| Jordy Rose | a91768e | 2011-07-22 02:08:32 +0000 | [diff] [blame] | 2028 | SmallVectorImpl<const ObjCIvarDecl*> &Ivars) const { | 
| Fariborz Jahanian | a50b3a2 | 2010-08-20 21:21:08 +0000 | [diff] [blame] | 2029 | if (const ObjCInterfaceDecl *SuperClass = OI->getSuperClass()) | 
|  | 2030 | DeepCollectObjCIvars(SuperClass, false, Ivars); | 
|  | 2031 | if (!leafClass) { | 
| Aaron Ballman | 59abbd4 | 2014-03-13 21:09:43 +0000 | [diff] [blame] | 2032 | for (const auto *I : OI->ivars()) | 
|  | 2033 | Ivars.push_back(I); | 
| Chad Rosier | 6fdf38b | 2011-08-17 23:08:45 +0000 | [diff] [blame] | 2034 | } else { | 
| Fariborz Jahanian | b26d578 | 2011-06-28 18:05:25 +0000 | [diff] [blame] | 2035 | ObjCInterfaceDecl *IDecl = const_cast<ObjCInterfaceDecl *>(OI); | 
| Jordy Rose | a91768e | 2011-07-22 02:08:32 +0000 | [diff] [blame] | 2036 | for (const ObjCIvarDecl *Iv = IDecl->all_declared_ivar_begin(); Iv; | 
| Fariborz Jahanian | b26d578 | 2011-06-28 18:05:25 +0000 | [diff] [blame] | 2037 | Iv= Iv->getNextIvar()) | 
|  | 2038 | Ivars.push_back(Iv); | 
|  | 2039 | } | 
| Fariborz Jahanian | 0f44d81 | 2009-05-12 18:14:29 +0000 | [diff] [blame] | 2040 | } | 
|  | 2041 |  | 
| Fariborz Jahanian | 6c5a8e2 | 2009-10-30 01:13:23 +0000 | [diff] [blame] | 2042 | /// CollectInheritedProtocols - Collect all protocols in current class and | 
|  | 2043 | /// those inherited by it. | 
|  | 2044 | void ASTContext::CollectInheritedProtocols(const Decl *CDecl, | 
| Fariborz Jahanian | dc68f95 | 2010-02-12 19:27:33 +0000 | [diff] [blame] | 2045 | llvm::SmallPtrSet<ObjCProtocolDecl*, 8> &Protocols) { | 
| Fariborz Jahanian | 6c5a8e2 | 2009-10-30 01:13:23 +0000 | [diff] [blame] | 2046 | if (const ObjCInterfaceDecl *OI = dyn_cast<ObjCInterfaceDecl>(CDecl)) { | 
| Ted Kremenek | 0ef508d | 2010-09-01 01:21:15 +0000 | [diff] [blame] | 2047 | // We can use protocol_iterator here instead of | 
|  | 2048 | // all_referenced_protocol_iterator since we are walking all categories. | 
| Aaron Ballman | a9f49e3 | 2014-03-13 20:55:22 +0000 | [diff] [blame] | 2049 | for (auto *Proto : OI->all_referenced_protocols()) { | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 2050 | CollectInheritedProtocols(Proto, Protocols); | 
| Fariborz Jahanian | 8e3b9db | 2010-02-25 18:24:33 +0000 | [diff] [blame] | 2051 | } | 
| Fariborz Jahanian | 6c5a8e2 | 2009-10-30 01:13:23 +0000 | [diff] [blame] | 2052 |  | 
|  | 2053 | // Categories of this Interface. | 
| Aaron Ballman | 3fe486a | 2014-03-13 21:23:55 +0000 | [diff] [blame] | 2054 | for (const auto *Cat : OI->visible_categories()) | 
|  | 2055 | CollectInheritedProtocols(Cat, Protocols); | 
| Douglas Gregor | 048fbfa | 2013-01-16 23:00:23 +0000 | [diff] [blame] | 2056 |  | 
| Fariborz Jahanian | 6c5a8e2 | 2009-10-30 01:13:23 +0000 | [diff] [blame] | 2057 | if (ObjCInterfaceDecl *SD = OI->getSuperClass()) | 
|  | 2058 | while (SD) { | 
|  | 2059 | CollectInheritedProtocols(SD, Protocols); | 
|  | 2060 | SD = SD->getSuperClass(); | 
|  | 2061 | } | 
| Benjamin Kramer | 0a3fe04 | 2010-04-27 17:47:25 +0000 | [diff] [blame] | 2062 | } else if (const ObjCCategoryDecl *OC = dyn_cast<ObjCCategoryDecl>(CDecl)) { | 
| Aaron Ballman | 19a4176 | 2014-03-14 12:55:57 +0000 | [diff] [blame] | 2063 | for (auto *Proto : OC->protocols()) { | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 2064 | CollectInheritedProtocols(Proto, Protocols); | 
| Fariborz Jahanian | 6c5a8e2 | 2009-10-30 01:13:23 +0000 | [diff] [blame] | 2065 | } | 
| Benjamin Kramer | 0a3fe04 | 2010-04-27 17:47:25 +0000 | [diff] [blame] | 2066 | } else if (const ObjCProtocolDecl *OP = dyn_cast<ObjCProtocolDecl>(CDecl)) { | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 2067 | // Insert the protocol. | 
|  | 2068 | if (!Protocols.insert( | 
|  | 2069 | const_cast<ObjCProtocolDecl *>(OP->getCanonicalDecl())).second) | 
|  | 2070 | return; | 
|  | 2071 |  | 
|  | 2072 | for (auto *Proto : OP->protocols()) | 
|  | 2073 | CollectInheritedProtocols(Proto, Protocols); | 
| Fariborz Jahanian | 6c5a8e2 | 2009-10-30 01:13:23 +0000 | [diff] [blame] | 2074 | } | 
|  | 2075 | } | 
|  | 2076 |  | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 2077 | unsigned ASTContext::CountNonClassIvars(const ObjCInterfaceDecl *OI) const { | 
| Fariborz Jahanian | d2ae2d0 | 2010-03-22 18:25:57 +0000 | [diff] [blame] | 2078 | unsigned count = 0; | 
|  | 2079 | // Count ivars declared in class extension. | 
| Aaron Ballman | b4a5345 | 2014-03-13 21:57:01 +0000 | [diff] [blame] | 2080 | for (const auto *Ext : OI->known_extensions()) | 
| Douglas Gregor | 048fbfa | 2013-01-16 23:00:23 +0000 | [diff] [blame] | 2081 | count += Ext->ivar_size(); | 
| Douglas Gregor | 048fbfa | 2013-01-16 23:00:23 +0000 | [diff] [blame] | 2082 |  | 
| Fariborz Jahanian | d2ae2d0 | 2010-03-22 18:25:57 +0000 | [diff] [blame] | 2083 | // Count ivar defined in this class's implementation.  This | 
|  | 2084 | // includes synthesized ivars. | 
|  | 2085 | if (ObjCImplementationDecl *ImplDecl = OI->getImplementation()) | 
| Benjamin Kramer | 0a3fe04 | 2010-04-27 17:47:25 +0000 | [diff] [blame] | 2086 | count += ImplDecl->ivar_size(); | 
|  | 2087 |  | 
| Fariborz Jahanian | 7c80959 | 2009-06-04 01:19:09 +0000 | [diff] [blame] | 2088 | return count; | 
|  | 2089 | } | 
|  | 2090 |  | 
| Argyrios Kyrtzidis | 2e809ce | 2012-02-03 05:58:16 +0000 | [diff] [blame] | 2091 | bool ASTContext::isSentinelNullExpr(const Expr *E) { | 
|  | 2092 | if (!E) | 
|  | 2093 | return false; | 
|  | 2094 |  | 
|  | 2095 | // nullptr_t is always treated as null. | 
|  | 2096 | if (E->getType()->isNullPtrType()) return true; | 
|  | 2097 |  | 
|  | 2098 | if (E->getType()->isAnyPointerType() && | 
|  | 2099 | E->IgnoreParenCasts()->isNullPointerConstant(*this, | 
|  | 2100 | Expr::NPC_ValueDependentIsNull)) | 
|  | 2101 | return true; | 
|  | 2102 |  | 
|  | 2103 | // Unfortunately, __null has type 'int'. | 
|  | 2104 | if (isa<GNUNullExpr>(E)) return true; | 
|  | 2105 |  | 
|  | 2106 | return false; | 
|  | 2107 | } | 
|  | 2108 |  | 
| Argyrios Kyrtzidis | 6d9fab7 | 2009-07-21 00:05:53 +0000 | [diff] [blame] | 2109 | /// \brief Get the implementation of ObjCInterfaceDecl,or NULL if none exists. | 
|  | 2110 | ObjCImplementationDecl *ASTContext::getObjCImplementation(ObjCInterfaceDecl *D) { | 
|  | 2111 | llvm::DenseMap<ObjCContainerDecl*, ObjCImplDecl*>::iterator | 
|  | 2112 | I = ObjCImpls.find(D); | 
|  | 2113 | if (I != ObjCImpls.end()) | 
|  | 2114 | return cast<ObjCImplementationDecl>(I->second); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2115 | return nullptr; | 
| Argyrios Kyrtzidis | 6d9fab7 | 2009-07-21 00:05:53 +0000 | [diff] [blame] | 2116 | } | 
|  | 2117 | /// \brief Get the implementation of ObjCCategoryDecl, or NULL if none exists. | 
|  | 2118 | ObjCCategoryImplDecl *ASTContext::getObjCImplementation(ObjCCategoryDecl *D) { | 
|  | 2119 | llvm::DenseMap<ObjCContainerDecl*, ObjCImplDecl*>::iterator | 
|  | 2120 | I = ObjCImpls.find(D); | 
|  | 2121 | if (I != ObjCImpls.end()) | 
|  | 2122 | return cast<ObjCCategoryImplDecl>(I->second); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2123 | return nullptr; | 
| Argyrios Kyrtzidis | 6d9fab7 | 2009-07-21 00:05:53 +0000 | [diff] [blame] | 2124 | } | 
|  | 2125 |  | 
|  | 2126 | /// \brief Set the implementation of ObjCInterfaceDecl. | 
|  | 2127 | void ASTContext::setObjCImplementation(ObjCInterfaceDecl *IFaceD, | 
|  | 2128 | ObjCImplementationDecl *ImplD) { | 
|  | 2129 | assert(IFaceD && ImplD && "Passed null params"); | 
|  | 2130 | ObjCImpls[IFaceD] = ImplD; | 
|  | 2131 | } | 
|  | 2132 | /// \brief Set the implementation of ObjCCategoryDecl. | 
|  | 2133 | void ASTContext::setObjCImplementation(ObjCCategoryDecl *CatD, | 
|  | 2134 | ObjCCategoryImplDecl *ImplD) { | 
|  | 2135 | assert(CatD && ImplD && "Passed null params"); | 
|  | 2136 | ObjCImpls[CatD] = ImplD; | 
|  | 2137 | } | 
|  | 2138 |  | 
| Chandler Carruth | 21c9060 | 2015-12-30 03:24:14 +0000 | [diff] [blame] | 2139 | const ObjCMethodDecl * | 
|  | 2140 | ASTContext::getObjCMethodRedeclaration(const ObjCMethodDecl *MD) const { | 
|  | 2141 | return ObjCMethodRedecls.lookup(MD); | 
|  | 2142 | } | 
|  | 2143 |  | 
|  | 2144 | void ASTContext::setObjCMethodRedeclaration(const ObjCMethodDecl *MD, | 
|  | 2145 | const ObjCMethodDecl *Redecl) { | 
|  | 2146 | assert(!getObjCMethodRedeclaration(MD) && "MD already has a redeclaration"); | 
|  | 2147 | ObjCMethodRedecls[MD] = Redecl; | 
|  | 2148 | } | 
|  | 2149 |  | 
| Dmitri Gribenko | 37527c2 | 2013-02-03 13:23:21 +0000 | [diff] [blame] | 2150 | const ObjCInterfaceDecl *ASTContext::getObjContainingInterface( | 
|  | 2151 | const NamedDecl *ND) const { | 
|  | 2152 | if (const ObjCInterfaceDecl *ID = | 
|  | 2153 | dyn_cast<ObjCInterfaceDecl>(ND->getDeclContext())) | 
| Argyrios Kyrtzidis | b9689bb | 2011-11-01 17:14:12 +0000 | [diff] [blame] | 2154 | return ID; | 
| Dmitri Gribenko | 37527c2 | 2013-02-03 13:23:21 +0000 | [diff] [blame] | 2155 | if (const ObjCCategoryDecl *CD = | 
|  | 2156 | dyn_cast<ObjCCategoryDecl>(ND->getDeclContext())) | 
| Argyrios Kyrtzidis | b9689bb | 2011-11-01 17:14:12 +0000 | [diff] [blame] | 2157 | return CD->getClassInterface(); | 
| Dmitri Gribenko | 37527c2 | 2013-02-03 13:23:21 +0000 | [diff] [blame] | 2158 | if (const ObjCImplDecl *IMD = | 
|  | 2159 | dyn_cast<ObjCImplDecl>(ND->getDeclContext())) | 
| Argyrios Kyrtzidis | b9689bb | 2011-11-01 17:14:12 +0000 | [diff] [blame] | 2160 | return IMD->getClassInterface(); | 
|  | 2161 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2162 | return nullptr; | 
| Argyrios Kyrtzidis | b9689bb | 2011-11-01 17:14:12 +0000 | [diff] [blame] | 2163 | } | 
|  | 2164 |  | 
| Fariborz Jahanian | 7cfe767 | 2010-12-01 22:29:46 +0000 | [diff] [blame] | 2165 | /// \brief Get the copy initialization expression of VarDecl,or NULL if | 
|  | 2166 | /// none exists. | 
| Fariborz Jahanian | 5019809 | 2010-12-02 17:02:11 +0000 | [diff] [blame] | 2167 | Expr *ASTContext::getBlockVarCopyInits(const VarDecl*VD) { | 
| Fariborz Jahanian | 1321cbb | 2010-12-06 17:28:17 +0000 | [diff] [blame] | 2168 | assert(VD && "Passed null params"); | 
|  | 2169 | assert(VD->hasAttr<BlocksAttr>() && | 
|  | 2170 | "getBlockVarCopyInits - not __block var"); | 
| Fariborz Jahanian | 5019809 | 2010-12-02 17:02:11 +0000 | [diff] [blame] | 2171 | llvm::DenseMap<const VarDecl*, Expr*>::iterator | 
| Fariborz Jahanian | 1321cbb | 2010-12-06 17:28:17 +0000 | [diff] [blame] | 2172 | I = BlockVarCopyInits.find(VD); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2173 | return (I != BlockVarCopyInits.end()) ? cast<Expr>(I->second) : nullptr; | 
| Fariborz Jahanian | 7cfe767 | 2010-12-01 22:29:46 +0000 | [diff] [blame] | 2174 | } | 
|  | 2175 |  | 
|  | 2176 | /// \brief Set the copy inialization expression of a block var decl. | 
|  | 2177 | void ASTContext::setBlockVarCopyInits(VarDecl*VD, Expr* Init) { | 
|  | 2178 | assert(VD && Init && "Passed null params"); | 
| Fariborz Jahanian | 1321cbb | 2010-12-06 17:28:17 +0000 | [diff] [blame] | 2179 | assert(VD->hasAttr<BlocksAttr>() && | 
|  | 2180 | "setBlockVarCopyInits - not __block var"); | 
| Fariborz Jahanian | 7cfe767 | 2010-12-01 22:29:46 +0000 | [diff] [blame] | 2181 | BlockVarCopyInits[VD] = Init; | 
|  | 2182 | } | 
|  | 2183 |  | 
| John McCall | bcd0350 | 2009-12-07 02:54:59 +0000 | [diff] [blame] | 2184 | TypeSourceInfo *ASTContext::CreateTypeSourceInfo(QualType T, | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 2185 | unsigned DataSize) const { | 
| John McCall | 26fe7e0 | 2009-10-21 00:23:54 +0000 | [diff] [blame] | 2186 | if (!DataSize) | 
|  | 2187 | DataSize = TypeLoc::getFullDataSizeForType(T); | 
|  | 2188 | else | 
|  | 2189 | assert(DataSize == TypeLoc::getFullDataSizeForType(T) && | 
| John McCall | bcd0350 | 2009-12-07 02:54:59 +0000 | [diff] [blame] | 2190 | "incorrect data size provided to CreateTypeSourceInfo!"); | 
| John McCall | 26fe7e0 | 2009-10-21 00:23:54 +0000 | [diff] [blame] | 2191 |  | 
| John McCall | bcd0350 | 2009-12-07 02:54:59 +0000 | [diff] [blame] | 2192 | TypeSourceInfo *TInfo = | 
|  | 2193 | (TypeSourceInfo*)BumpAlloc.Allocate(sizeof(TypeSourceInfo) + DataSize, 8); | 
|  | 2194 | new (TInfo) TypeSourceInfo(T); | 
|  | 2195 | return TInfo; | 
| Argyrios Kyrtzidis | 3f79ad7 | 2009-08-19 01:27:32 +0000 | [diff] [blame] | 2196 | } | 
|  | 2197 |  | 
| John McCall | bcd0350 | 2009-12-07 02:54:59 +0000 | [diff] [blame] | 2198 | TypeSourceInfo *ASTContext::getTrivialTypeSourceInfo(QualType T, | 
| Douglas Gregor | 0231d8d | 2011-01-19 20:10:05 +0000 | [diff] [blame] | 2199 | SourceLocation L) const { | 
| John McCall | bcd0350 | 2009-12-07 02:54:59 +0000 | [diff] [blame] | 2200 | TypeSourceInfo *DI = CreateTypeSourceInfo(T); | 
| Douglas Gregor | 2d525f0 | 2011-01-25 19:13:18 +0000 | [diff] [blame] | 2201 | DI->getTypeLoc().initialize(const_cast<ASTContext &>(*this), L); | 
| John McCall | 3665e00 | 2009-10-23 21:14:09 +0000 | [diff] [blame] | 2202 | return DI; | 
|  | 2203 | } | 
|  | 2204 |  | 
| Daniel Dunbar | 02f7f5f | 2009-05-03 10:38:35 +0000 | [diff] [blame] | 2205 | const ASTRecordLayout & | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 2206 | ASTContext::getASTObjCInterfaceLayout(const ObjCInterfaceDecl *D) const { | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2207 | return getObjCLayout(D, nullptr); | 
| Daniel Dunbar | 02f7f5f | 2009-05-03 10:38:35 +0000 | [diff] [blame] | 2208 | } | 
|  | 2209 |  | 
|  | 2210 | const ASTRecordLayout & | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 2211 | ASTContext::getASTObjCImplementationLayout( | 
|  | 2212 | const ObjCImplementationDecl *D) const { | 
| Daniel Dunbar | 02f7f5f | 2009-05-03 10:38:35 +0000 | [diff] [blame] | 2213 | return getObjCLayout(D->getClassInterface(), D); | 
|  | 2214 | } | 
|  | 2215 |  | 
| Chris Lattner | 983a8bb | 2007-07-13 22:13:22 +0000 | [diff] [blame] | 2216 | //===----------------------------------------------------------------------===// | 
|  | 2217 | //                   Type creation/memoization methods | 
|  | 2218 | //===----------------------------------------------------------------------===// | 
|  | 2219 |  | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 2220 | QualType | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2221 | ASTContext::getExtQualType(const Type *baseType, Qualifiers quals) const { | 
|  | 2222 | unsigned fastQuals = quals.getFastQualifiers(); | 
|  | 2223 | quals.removeFastQualifiers(); | 
| John McCall | 8ccfcb5 | 2009-09-24 19:53:00 +0000 | [diff] [blame] | 2224 |  | 
|  | 2225 | // Check if we've already instantiated this type. | 
|  | 2226 | llvm::FoldingSetNodeID ID; | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2227 | ExtQuals::Profile(ID, baseType, quals); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2228 | void *insertPos = nullptr; | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2229 | if (ExtQuals *eq = ExtQualNodes.FindNodeOrInsertPos(ID, insertPos)) { | 
|  | 2230 | assert(eq->getQualifiers() == quals); | 
|  | 2231 | return QualType(eq, fastQuals); | 
| John McCall | 8ccfcb5 | 2009-09-24 19:53:00 +0000 | [diff] [blame] | 2232 | } | 
|  | 2233 |  | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2234 | // If the base type is not canonical, make the appropriate canonical type. | 
|  | 2235 | QualType canon; | 
|  | 2236 | if (!baseType->isCanonicalUnqualified()) { | 
|  | 2237 | SplitQualType canonSplit = baseType->getCanonicalTypeInternal().split(); | 
| John McCall | 18ce25e | 2012-02-08 00:46:36 +0000 | [diff] [blame] | 2238 | canonSplit.Quals.addConsistentQualifiers(quals); | 
|  | 2239 | canon = getExtQualType(canonSplit.Ty, canonSplit.Quals); | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2240 |  | 
|  | 2241 | // Re-find the insert position. | 
|  | 2242 | (void) ExtQualNodes.FindNodeOrInsertPos(ID, insertPos); | 
|  | 2243 | } | 
|  | 2244 |  | 
|  | 2245 | ExtQuals *eq = new (*this, TypeAlignment) ExtQuals(baseType, canon, quals); | 
|  | 2246 | ExtQualNodes.InsertNode(eq, insertPos); | 
|  | 2247 | return QualType(eq, fastQuals); | 
| John McCall | 8ccfcb5 | 2009-09-24 19:53:00 +0000 | [diff] [blame] | 2248 | } | 
|  | 2249 |  | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 2250 | QualType | 
|  | 2251 | ASTContext::getAddrSpaceQualType(QualType T, unsigned AddressSpace) const { | 
| Chris Lattner | 76a00cf | 2008-04-06 22:59:24 +0000 | [diff] [blame] | 2252 | QualType CanT = getCanonicalType(T); | 
|  | 2253 | if (CanT.getAddressSpace() == AddressSpace) | 
| Chris Lattner | 445fcab | 2008-02-20 20:55:12 +0000 | [diff] [blame] | 2254 | return T; | 
| Chris Lattner | d60183d | 2009-02-18 22:53:11 +0000 | [diff] [blame] | 2255 |  | 
| John McCall | 8ccfcb5 | 2009-09-24 19:53:00 +0000 | [diff] [blame] | 2256 | // If we are composing extended qualifiers together, merge together | 
|  | 2257 | // into one ExtQuals node. | 
|  | 2258 | QualifierCollector Quals; | 
|  | 2259 | const Type *TypeNode = Quals.strip(T); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2260 |  | 
| John McCall | 8ccfcb5 | 2009-09-24 19:53:00 +0000 | [diff] [blame] | 2261 | // If this type already has an address space specified, it cannot get | 
|  | 2262 | // another one. | 
|  | 2263 | assert(!Quals.hasAddressSpace() && | 
|  | 2264 | "Type cannot be in multiple addr spaces!"); | 
|  | 2265 | Quals.addAddressSpace(AddressSpace); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2266 |  | 
| John McCall | 8ccfcb5 | 2009-09-24 19:53:00 +0000 | [diff] [blame] | 2267 | return getExtQualType(TypeNode, Quals); | 
| Christopher Lamb | 025b5fb | 2008-02-04 02:31:56 +0000 | [diff] [blame] | 2268 | } | 
|  | 2269 |  | 
| Chris Lattner | d60183d | 2009-02-18 22:53:11 +0000 | [diff] [blame] | 2270 | QualType ASTContext::getObjCGCQualType(QualType T, | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 2271 | Qualifiers::GC GCAttr) const { | 
| Fariborz Jahanian | e27e934 | 2009-02-18 05:09:49 +0000 | [diff] [blame] | 2272 | QualType CanT = getCanonicalType(T); | 
| Chris Lattner | d60183d | 2009-02-18 22:53:11 +0000 | [diff] [blame] | 2273 | if (CanT.getObjCGCAttr() == GCAttr) | 
| Fariborz Jahanian | e27e934 | 2009-02-18 05:09:49 +0000 | [diff] [blame] | 2274 | return T; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2275 |  | 
| John McCall | 53fa714 | 2010-12-24 02:08:15 +0000 | [diff] [blame] | 2276 | if (const PointerType *ptr = T->getAs<PointerType>()) { | 
|  | 2277 | QualType Pointee = ptr->getPointeeType(); | 
| Steve Naroff | 6b712a7 | 2009-07-14 18:25:06 +0000 | [diff] [blame] | 2278 | if (Pointee->isAnyPointerType()) { | 
| Fariborz Jahanian | b68215c | 2009-06-03 17:15:17 +0000 | [diff] [blame] | 2279 | QualType ResultType = getObjCGCQualType(Pointee, GCAttr); | 
|  | 2280 | return getPointerType(ResultType); | 
|  | 2281 | } | 
|  | 2282 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2283 |  | 
| John McCall | 8ccfcb5 | 2009-09-24 19:53:00 +0000 | [diff] [blame] | 2284 | // If we are composing extended qualifiers together, merge together | 
|  | 2285 | // into one ExtQuals node. | 
|  | 2286 | QualifierCollector Quals; | 
|  | 2287 | const Type *TypeNode = Quals.strip(T); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2288 |  | 
| John McCall | 8ccfcb5 | 2009-09-24 19:53:00 +0000 | [diff] [blame] | 2289 | // If this type already has an ObjCGC specified, it cannot get | 
|  | 2290 | // another one. | 
|  | 2291 | assert(!Quals.hasObjCGCAttr() && | 
|  | 2292 | "Type cannot have multiple ObjCGCs!"); | 
|  | 2293 | Quals.addObjCGCAttr(GCAttr); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2294 |  | 
| John McCall | 8ccfcb5 | 2009-09-24 19:53:00 +0000 | [diff] [blame] | 2295 | return getExtQualType(TypeNode, Quals); | 
| Fariborz Jahanian | e27e934 | 2009-02-18 05:09:49 +0000 | [diff] [blame] | 2296 | } | 
| Chris Lattner | 983a8bb | 2007-07-13 22:13:22 +0000 | [diff] [blame] | 2297 |  | 
| John McCall | 4f5019e | 2010-12-19 02:44:49 +0000 | [diff] [blame] | 2298 | const FunctionType *ASTContext::adjustFunctionType(const FunctionType *T, | 
|  | 2299 | FunctionType::ExtInfo Info) { | 
|  | 2300 | if (T->getExtInfo() == Info) | 
|  | 2301 | return T; | 
|  | 2302 |  | 
|  | 2303 | QualType Result; | 
|  | 2304 | if (const FunctionNoProtoType *FNPT = dyn_cast<FunctionNoProtoType>(T)) { | 
| Alp Toker | 314cc81 | 2014-01-25 16:55:45 +0000 | [diff] [blame] | 2305 | Result = getFunctionNoProtoType(FNPT->getReturnType(), Info); | 
| John McCall | 4f5019e | 2010-12-19 02:44:49 +0000 | [diff] [blame] | 2306 | } else { | 
|  | 2307 | const FunctionProtoType *FPT = cast<FunctionProtoType>(T); | 
|  | 2308 | FunctionProtoType::ExtProtoInfo EPI = FPT->getExtProtoInfo(); | 
|  | 2309 | EPI.ExtInfo = Info; | 
| Alp Toker | 314cc81 | 2014-01-25 16:55:45 +0000 | [diff] [blame] | 2310 | Result = getFunctionType(FPT->getReturnType(), FPT->getParamTypes(), EPI); | 
| John McCall | 4f5019e | 2010-12-19 02:44:49 +0000 | [diff] [blame] | 2311 | } | 
|  | 2312 |  | 
|  | 2313 | return cast<FunctionType>(Result.getTypePtr()); | 
|  | 2314 | } | 
|  | 2315 |  | 
| Richard Smith | 2a7d481 | 2013-05-04 07:00:32 +0000 | [diff] [blame] | 2316 | void ASTContext::adjustDeducedFunctionResultType(FunctionDecl *FD, | 
|  | 2317 | QualType ResultType) { | 
| Richard Smith | 1fa5d64 | 2013-05-11 05:45:24 +0000 | [diff] [blame] | 2318 | FD = FD->getMostRecentDecl(); | 
|  | 2319 | while (true) { | 
| Richard Smith | 2a7d481 | 2013-05-04 07:00:32 +0000 | [diff] [blame] | 2320 | const FunctionProtoType *FPT = FD->getType()->castAs<FunctionProtoType>(); | 
|  | 2321 | FunctionProtoType::ExtProtoInfo EPI = FPT->getExtProtoInfo(); | 
| Alp Toker | 9cacbab | 2014-01-20 20:26:09 +0000 | [diff] [blame] | 2322 | FD->setType(getFunctionType(ResultType, FPT->getParamTypes(), EPI)); | 
| Richard Smith | 1fa5d64 | 2013-05-11 05:45:24 +0000 | [diff] [blame] | 2323 | if (FunctionDecl *Next = FD->getPreviousDecl()) | 
|  | 2324 | FD = Next; | 
|  | 2325 | else | 
|  | 2326 | break; | 
| Richard Smith | 2a7d481 | 2013-05-04 07:00:32 +0000 | [diff] [blame] | 2327 | } | 
| Richard Smith | 1fa5d64 | 2013-05-11 05:45:24 +0000 | [diff] [blame] | 2328 | if (ASTMutationListener *L = getASTMutationListener()) | 
|  | 2329 | L->DeducedReturnType(FD, ResultType); | 
| Richard Smith | 2a7d481 | 2013-05-04 07:00:32 +0000 | [diff] [blame] | 2330 | } | 
|  | 2331 |  | 
| Richard Smith | 0b3a462 | 2014-11-13 20:01:57 +0000 | [diff] [blame] | 2332 | /// Get a function type and produce the equivalent function type with the | 
|  | 2333 | /// specified exception specification. Type sugar that can be present on a | 
|  | 2334 | /// declaration of a function with an exception specification is permitted | 
|  | 2335 | /// and preserved. Other type sugar (for instance, typedefs) is not. | 
|  | 2336 | static QualType getFunctionTypeWithExceptionSpec( | 
|  | 2337 | ASTContext &Context, QualType Orig, | 
|  | 2338 | const FunctionProtoType::ExceptionSpecInfo &ESI) { | 
|  | 2339 | // Might have some parens. | 
|  | 2340 | if (auto *PT = dyn_cast<ParenType>(Orig)) | 
|  | 2341 | return Context.getParenType( | 
|  | 2342 | getFunctionTypeWithExceptionSpec(Context, PT->getInnerType(), ESI)); | 
|  | 2343 |  | 
|  | 2344 | // Might have a calling-convention attribute. | 
|  | 2345 | if (auto *AT = dyn_cast<AttributedType>(Orig)) | 
|  | 2346 | return Context.getAttributedType( | 
|  | 2347 | AT->getAttrKind(), | 
|  | 2348 | getFunctionTypeWithExceptionSpec(Context, AT->getModifiedType(), ESI), | 
|  | 2349 | getFunctionTypeWithExceptionSpec(Context, AT->getEquivalentType(), | 
|  | 2350 | ESI)); | 
|  | 2351 |  | 
|  | 2352 | // Anything else must be a function type. Rebuild it with the new exception | 
|  | 2353 | // specification. | 
|  | 2354 | const FunctionProtoType *Proto = cast<FunctionProtoType>(Orig); | 
|  | 2355 | return Context.getFunctionType( | 
|  | 2356 | Proto->getReturnType(), Proto->getParamTypes(), | 
|  | 2357 | Proto->getExtProtoInfo().withExceptionSpec(ESI)); | 
|  | 2358 | } | 
|  | 2359 |  | 
|  | 2360 | void ASTContext::adjustExceptionSpec( | 
|  | 2361 | FunctionDecl *FD, const FunctionProtoType::ExceptionSpecInfo &ESI, | 
|  | 2362 | bool AsWritten) { | 
|  | 2363 | // Update the type. | 
|  | 2364 | QualType Updated = | 
|  | 2365 | getFunctionTypeWithExceptionSpec(*this, FD->getType(), ESI); | 
|  | 2366 | FD->setType(Updated); | 
|  | 2367 |  | 
|  | 2368 | if (!AsWritten) | 
|  | 2369 | return; | 
|  | 2370 |  | 
|  | 2371 | // Update the type in the type source information too. | 
|  | 2372 | if (TypeSourceInfo *TSInfo = FD->getTypeSourceInfo()) { | 
|  | 2373 | // If the type and the type-as-written differ, we may need to update | 
|  | 2374 | // the type-as-written too. | 
|  | 2375 | if (TSInfo->getType() != FD->getType()) | 
|  | 2376 | Updated = getFunctionTypeWithExceptionSpec(*this, TSInfo->getType(), ESI); | 
|  | 2377 |  | 
|  | 2378 | // FIXME: When we get proper type location information for exceptions, | 
|  | 2379 | // we'll also have to rebuild the TypeSourceInfo. For now, we just patch | 
|  | 2380 | // up the TypeSourceInfo; | 
|  | 2381 | assert(TypeLoc::getFullDataSizeForType(Updated) == | 
|  | 2382 | TypeLoc::getFullDataSizeForType(TSInfo->getType()) && | 
|  | 2383 | "TypeLoc size mismatch from updating exception specification"); | 
|  | 2384 | TSInfo->overrideType(Updated); | 
|  | 2385 | } | 
|  | 2386 | } | 
|  | 2387 |  | 
| Chris Lattner | c639593 | 2007-06-22 20:56:16 +0000 | [diff] [blame] | 2388 | /// getComplexType - Return the uniqued reference to the type for a complex | 
|  | 2389 | /// number with the specified element type. | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 2390 | QualType ASTContext::getComplexType(QualType T) const { | 
| Chris Lattner | c639593 | 2007-06-22 20:56:16 +0000 | [diff] [blame] | 2391 | // Unique pointers, to guarantee there is only one pointer of a particular | 
|  | 2392 | // structure. | 
|  | 2393 | llvm::FoldingSetNodeID ID; | 
|  | 2394 | ComplexType::Profile(ID, T); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2395 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2396 | void *InsertPos = nullptr; | 
| Chris Lattner | c639593 | 2007-06-22 20:56:16 +0000 | [diff] [blame] | 2397 | if (ComplexType *CT = ComplexTypes.FindNodeOrInsertPos(ID, InsertPos)) | 
|  | 2398 | return QualType(CT, 0); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2399 |  | 
| Chris Lattner | c639593 | 2007-06-22 20:56:16 +0000 | [diff] [blame] | 2400 | // If the pointee type isn't canonical, this won't be a canonical type either, | 
|  | 2401 | // so fill in the canonical type field. | 
|  | 2402 | QualType Canonical; | 
| John McCall | b692a09 | 2009-10-22 20:10:53 +0000 | [diff] [blame] | 2403 | if (!T.isCanonical()) { | 
| Chris Lattner | 76a00cf | 2008-04-06 22:59:24 +0000 | [diff] [blame] | 2404 | Canonical = getComplexType(getCanonicalType(T)); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2405 |  | 
| Chris Lattner | c639593 | 2007-06-22 20:56:16 +0000 | [diff] [blame] | 2406 | // Get the new insert position for the node we care about. | 
|  | 2407 | ComplexType *NewIP = ComplexTypes.FindNodeOrInsertPos(ID, InsertPos); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2408 | assert(!NewIP && "Shouldn't be in the map!"); (void)NewIP; | 
| Chris Lattner | c639593 | 2007-06-22 20:56:16 +0000 | [diff] [blame] | 2409 | } | 
| John McCall | 90d1c2d | 2009-09-24 23:30:46 +0000 | [diff] [blame] | 2410 | ComplexType *New = new (*this, TypeAlignment) ComplexType(T, Canonical); | 
| Chris Lattner | c639593 | 2007-06-22 20:56:16 +0000 | [diff] [blame] | 2411 | Types.push_back(New); | 
|  | 2412 | ComplexTypes.InsertNode(New, InsertPos); | 
|  | 2413 | return QualType(New, 0); | 
|  | 2414 | } | 
|  | 2415 |  | 
| Chris Lattner | 970e54e | 2006-11-12 00:37:36 +0000 | [diff] [blame] | 2416 | /// getPointerType - Return the uniqued reference to the type for a pointer to | 
|  | 2417 | /// the specified type. | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 2418 | QualType ASTContext::getPointerType(QualType T) const { | 
| Chris Lattner | d5973eb | 2006-11-12 00:53:46 +0000 | [diff] [blame] | 2419 | // Unique pointers, to guarantee there is only one pointer of a particular | 
|  | 2420 | // structure. | 
| Chris Lattner | 23b7eb6 | 2007-06-15 23:05:46 +0000 | [diff] [blame] | 2421 | llvm::FoldingSetNodeID ID; | 
| Chris Lattner | 67521df | 2007-01-27 01:29:36 +0000 | [diff] [blame] | 2422 | PointerType::Profile(ID, T); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2423 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2424 | void *InsertPos = nullptr; | 
| Chris Lattner | 67521df | 2007-01-27 01:29:36 +0000 | [diff] [blame] | 2425 | if (PointerType *PT = PointerTypes.FindNodeOrInsertPos(ID, InsertPos)) | 
| Steve Naroff | e5aa9be | 2007-04-05 22:36:20 +0000 | [diff] [blame] | 2426 | return QualType(PT, 0); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2427 |  | 
| Chris Lattner | 7ccecb9 | 2006-11-12 08:50:50 +0000 | [diff] [blame] | 2428 | // If the pointee type isn't canonical, this won't be a canonical type either, | 
|  | 2429 | // so fill in the canonical type field. | 
| Steve Naroff | e5aa9be | 2007-04-05 22:36:20 +0000 | [diff] [blame] | 2430 | QualType Canonical; | 
| Bob Wilson | c8541f2 | 2013-03-15 17:12:43 +0000 | [diff] [blame] | 2431 | if (!T.isCanonical()) { | 
| Chris Lattner | 76a00cf | 2008-04-06 22:59:24 +0000 | [diff] [blame] | 2432 | Canonical = getPointerType(getCanonicalType(T)); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2433 |  | 
| Bob Wilson | c8541f2 | 2013-03-15 17:12:43 +0000 | [diff] [blame] | 2434 | // Get the new insert position for the node we care about. | 
|  | 2435 | PointerType *NewIP = PointerTypes.FindNodeOrInsertPos(ID, InsertPos); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2436 | assert(!NewIP && "Shouldn't be in the map!"); (void)NewIP; | 
| Bob Wilson | c8541f2 | 2013-03-15 17:12:43 +0000 | [diff] [blame] | 2437 | } | 
| John McCall | 90d1c2d | 2009-09-24 23:30:46 +0000 | [diff] [blame] | 2438 | PointerType *New = new (*this, TypeAlignment) PointerType(T, Canonical); | 
| Chris Lattner | 67521df | 2007-01-27 01:29:36 +0000 | [diff] [blame] | 2439 | Types.push_back(New); | 
|  | 2440 | PointerTypes.InsertNode(New, InsertPos); | 
| Steve Naroff | e5aa9be | 2007-04-05 22:36:20 +0000 | [diff] [blame] | 2441 | return QualType(New, 0); | 
| Chris Lattner | ddc135e | 2006-11-10 06:34:16 +0000 | [diff] [blame] | 2442 | } | 
|  | 2443 |  | 
| Reid Kleckner | 0503a87 | 2013-12-05 01:23:43 +0000 | [diff] [blame] | 2444 | QualType ASTContext::getAdjustedType(QualType Orig, QualType New) const { | 
|  | 2445 | llvm::FoldingSetNodeID ID; | 
|  | 2446 | AdjustedType::Profile(ID, Orig, New); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2447 | void *InsertPos = nullptr; | 
| Reid Kleckner | 0503a87 | 2013-12-05 01:23:43 +0000 | [diff] [blame] | 2448 | AdjustedType *AT = AdjustedTypes.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 2449 | if (AT) | 
|  | 2450 | return QualType(AT, 0); | 
|  | 2451 |  | 
|  | 2452 | QualType Canonical = getCanonicalType(New); | 
|  | 2453 |  | 
|  | 2454 | // Get the new insert position for the node we care about. | 
|  | 2455 | AT = AdjustedTypes.FindNodeOrInsertPos(ID, InsertPos); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2456 | assert(!AT && "Shouldn't be in the map!"); | 
| Reid Kleckner | 0503a87 | 2013-12-05 01:23:43 +0000 | [diff] [blame] | 2457 |  | 
|  | 2458 | AT = new (*this, TypeAlignment) | 
|  | 2459 | AdjustedType(Type::Adjusted, Orig, New, Canonical); | 
|  | 2460 | Types.push_back(AT); | 
|  | 2461 | AdjustedTypes.InsertNode(AT, InsertPos); | 
|  | 2462 | return QualType(AT, 0); | 
|  | 2463 | } | 
|  | 2464 |  | 
| Reid Kleckner | 8a36502 | 2013-06-24 17:51:48 +0000 | [diff] [blame] | 2465 | QualType ASTContext::getDecayedType(QualType T) const { | 
|  | 2466 | assert((T->isArrayType() || T->isFunctionType()) && "T does not decay"); | 
|  | 2467 |  | 
| Reid Kleckner | 8a36502 | 2013-06-24 17:51:48 +0000 | [diff] [blame] | 2468 | QualType Decayed; | 
|  | 2469 |  | 
|  | 2470 | // C99 6.7.5.3p7: | 
|  | 2471 | //   A declaration of a parameter as "array of type" shall be | 
|  | 2472 | //   adjusted to "qualified pointer to type", where the type | 
|  | 2473 | //   qualifiers (if any) are those specified within the [ and ] of | 
|  | 2474 | //   the array type derivation. | 
|  | 2475 | if (T->isArrayType()) | 
|  | 2476 | Decayed = getArrayDecayedType(T); | 
|  | 2477 |  | 
|  | 2478 | // C99 6.7.5.3p8: | 
|  | 2479 | //   A declaration of a parameter as "function returning type" | 
|  | 2480 | //   shall be adjusted to "pointer to function returning type", as | 
|  | 2481 | //   in 6.3.2.1. | 
|  | 2482 | if (T->isFunctionType()) | 
|  | 2483 | Decayed = getPointerType(T); | 
|  | 2484 |  | 
| Reid Kleckner | 0503a87 | 2013-12-05 01:23:43 +0000 | [diff] [blame] | 2485 | llvm::FoldingSetNodeID ID; | 
|  | 2486 | AdjustedType::Profile(ID, T, Decayed); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2487 | void *InsertPos = nullptr; | 
| Reid Kleckner | 0503a87 | 2013-12-05 01:23:43 +0000 | [diff] [blame] | 2488 | AdjustedType *AT = AdjustedTypes.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 2489 | if (AT) | 
|  | 2490 | return QualType(AT, 0); | 
|  | 2491 |  | 
| Reid Kleckner | 8a36502 | 2013-06-24 17:51:48 +0000 | [diff] [blame] | 2492 | QualType Canonical = getCanonicalType(Decayed); | 
|  | 2493 |  | 
|  | 2494 | // Get the new insert position for the node we care about. | 
| Reid Kleckner | 0503a87 | 2013-12-05 01:23:43 +0000 | [diff] [blame] | 2495 | AT = AdjustedTypes.FindNodeOrInsertPos(ID, InsertPos); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2496 | assert(!AT && "Shouldn't be in the map!"); | 
| Reid Kleckner | 8a36502 | 2013-06-24 17:51:48 +0000 | [diff] [blame] | 2497 |  | 
| Reid Kleckner | 0503a87 | 2013-12-05 01:23:43 +0000 | [diff] [blame] | 2498 | AT = new (*this, TypeAlignment) DecayedType(T, Decayed, Canonical); | 
|  | 2499 | Types.push_back(AT); | 
|  | 2500 | AdjustedTypes.InsertNode(AT, InsertPos); | 
|  | 2501 | return QualType(AT, 0); | 
| Reid Kleckner | 8a36502 | 2013-06-24 17:51:48 +0000 | [diff] [blame] | 2502 | } | 
|  | 2503 |  | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2504 | /// getBlockPointerType - Return the uniqued reference to the type for | 
| Steve Naroff | ec33ed9 | 2008-08-27 16:04:49 +0000 | [diff] [blame] | 2505 | /// a pointer to the specified block. | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 2506 | QualType ASTContext::getBlockPointerType(QualType T) const { | 
| Steve Naroff | 0ac01283 | 2008-08-28 19:20:44 +0000 | [diff] [blame] | 2507 | assert(T->isFunctionType() && "block of function types only"); | 
|  | 2508 | // Unique pointers, to guarantee there is only one block of a particular | 
| Steve Naroff | ec33ed9 | 2008-08-27 16:04:49 +0000 | [diff] [blame] | 2509 | // structure. | 
|  | 2510 | llvm::FoldingSetNodeID ID; | 
|  | 2511 | BlockPointerType::Profile(ID, T); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2512 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2513 | void *InsertPos = nullptr; | 
| Steve Naroff | ec33ed9 | 2008-08-27 16:04:49 +0000 | [diff] [blame] | 2514 | if (BlockPointerType *PT = | 
|  | 2515 | BlockPointerTypes.FindNodeOrInsertPos(ID, InsertPos)) | 
|  | 2516 | return QualType(PT, 0); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2517 |  | 
|  | 2518 | // If the block pointee type isn't canonical, this won't be a canonical | 
| Steve Naroff | ec33ed9 | 2008-08-27 16:04:49 +0000 | [diff] [blame] | 2519 | // type either so fill in the canonical type field. | 
|  | 2520 | QualType Canonical; | 
| John McCall | b692a09 | 2009-10-22 20:10:53 +0000 | [diff] [blame] | 2521 | if (!T.isCanonical()) { | 
| Steve Naroff | ec33ed9 | 2008-08-27 16:04:49 +0000 | [diff] [blame] | 2522 | Canonical = getBlockPointerType(getCanonicalType(T)); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2523 |  | 
| Steve Naroff | ec33ed9 | 2008-08-27 16:04:49 +0000 | [diff] [blame] | 2524 | // Get the new insert position for the node we care about. | 
|  | 2525 | BlockPointerType *NewIP = | 
|  | 2526 | BlockPointerTypes.FindNodeOrInsertPos(ID, InsertPos); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2527 | assert(!NewIP && "Shouldn't be in the map!"); (void)NewIP; | 
| Steve Naroff | ec33ed9 | 2008-08-27 16:04:49 +0000 | [diff] [blame] | 2528 | } | 
| John McCall | 90d1c2d | 2009-09-24 23:30:46 +0000 | [diff] [blame] | 2529 | BlockPointerType *New | 
|  | 2530 | = new (*this, TypeAlignment) BlockPointerType(T, Canonical); | 
| Steve Naroff | ec33ed9 | 2008-08-27 16:04:49 +0000 | [diff] [blame] | 2531 | Types.push_back(New); | 
|  | 2532 | BlockPointerTypes.InsertNode(New, InsertPos); | 
|  | 2533 | return QualType(New, 0); | 
|  | 2534 | } | 
|  | 2535 |  | 
| Sebastian Redl | 0f8b23f | 2009-03-16 23:22:08 +0000 | [diff] [blame] | 2536 | /// getLValueReferenceType - Return the uniqued reference to the type for an | 
|  | 2537 | /// lvalue reference to the specified type. | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 2538 | QualType | 
|  | 2539 | ASTContext::getLValueReferenceType(QualType T, bool SpelledAsLValue) const { | 
| Douglas Gregor | 291e8ee | 2011-05-21 22:16:50 +0000 | [diff] [blame] | 2540 | assert(getCanonicalType(T) != OverloadTy && | 
|  | 2541 | "Unresolved overloaded function type"); | 
|  | 2542 |  | 
| Bill Wendling | 3708c18 | 2007-05-27 10:15:43 +0000 | [diff] [blame] | 2543 | // Unique pointers, to guarantee there is only one pointer of a particular | 
|  | 2544 | // structure. | 
| Chris Lattner | 23b7eb6 | 2007-06-15 23:05:46 +0000 | [diff] [blame] | 2545 | llvm::FoldingSetNodeID ID; | 
| John McCall | fc93cf9 | 2009-10-22 22:37:11 +0000 | [diff] [blame] | 2546 | ReferenceType::Profile(ID, T, SpelledAsLValue); | 
| Bill Wendling | 3708c18 | 2007-05-27 10:15:43 +0000 | [diff] [blame] | 2547 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2548 | void *InsertPos = nullptr; | 
| Sebastian Redl | 0f8b23f | 2009-03-16 23:22:08 +0000 | [diff] [blame] | 2549 | if (LValueReferenceType *RT = | 
|  | 2550 | LValueReferenceTypes.FindNodeOrInsertPos(ID, InsertPos)) | 
| Bill Wendling | 3708c18 | 2007-05-27 10:15:43 +0000 | [diff] [blame] | 2551 | return QualType(RT, 0); | 
| Sebastian Redl | 0f8b23f | 2009-03-16 23:22:08 +0000 | [diff] [blame] | 2552 |  | 
| John McCall | fc93cf9 | 2009-10-22 22:37:11 +0000 | [diff] [blame] | 2553 | const ReferenceType *InnerRef = T->getAs<ReferenceType>(); | 
|  | 2554 |  | 
| Bill Wendling | 3708c18 | 2007-05-27 10:15:43 +0000 | [diff] [blame] | 2555 | // If the referencee type isn't canonical, this won't be a canonical type | 
|  | 2556 | // either, so fill in the canonical type field. | 
|  | 2557 | QualType Canonical; | 
| John McCall | fc93cf9 | 2009-10-22 22:37:11 +0000 | [diff] [blame] | 2558 | if (!SpelledAsLValue || InnerRef || !T.isCanonical()) { | 
|  | 2559 | QualType PointeeType = (InnerRef ? InnerRef->getPointeeType() : T); | 
|  | 2560 | Canonical = getLValueReferenceType(getCanonicalType(PointeeType)); | 
| Sebastian Redl | 0f8b23f | 2009-03-16 23:22:08 +0000 | [diff] [blame] | 2561 |  | 
| Bill Wendling | 3708c18 | 2007-05-27 10:15:43 +0000 | [diff] [blame] | 2562 | // Get the new insert position for the node we care about. | 
| Sebastian Redl | 0f8b23f | 2009-03-16 23:22:08 +0000 | [diff] [blame] | 2563 | LValueReferenceType *NewIP = | 
|  | 2564 | LValueReferenceTypes.FindNodeOrInsertPos(ID, InsertPos); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2565 | assert(!NewIP && "Shouldn't be in the map!"); (void)NewIP; | 
| Bill Wendling | 3708c18 | 2007-05-27 10:15:43 +0000 | [diff] [blame] | 2566 | } | 
|  | 2567 |  | 
| John McCall | 90d1c2d | 2009-09-24 23:30:46 +0000 | [diff] [blame] | 2568 | LValueReferenceType *New | 
| John McCall | fc93cf9 | 2009-10-22 22:37:11 +0000 | [diff] [blame] | 2569 | = new (*this, TypeAlignment) LValueReferenceType(T, Canonical, | 
|  | 2570 | SpelledAsLValue); | 
| Bill Wendling | 3708c18 | 2007-05-27 10:15:43 +0000 | [diff] [blame] | 2571 | Types.push_back(New); | 
| Sebastian Redl | 0f8b23f | 2009-03-16 23:22:08 +0000 | [diff] [blame] | 2572 | LValueReferenceTypes.InsertNode(New, InsertPos); | 
| John McCall | fc93cf9 | 2009-10-22 22:37:11 +0000 | [diff] [blame] | 2573 |  | 
| Sebastian Redl | 0f8b23f | 2009-03-16 23:22:08 +0000 | [diff] [blame] | 2574 | return QualType(New, 0); | 
|  | 2575 | } | 
|  | 2576 |  | 
|  | 2577 | /// getRValueReferenceType - Return the uniqued reference to the type for an | 
|  | 2578 | /// rvalue reference to the specified type. | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 2579 | QualType ASTContext::getRValueReferenceType(QualType T) const { | 
| Sebastian Redl | 0f8b23f | 2009-03-16 23:22:08 +0000 | [diff] [blame] | 2580 | // Unique pointers, to guarantee there is only one pointer of a particular | 
|  | 2581 | // structure. | 
|  | 2582 | llvm::FoldingSetNodeID ID; | 
| John McCall | fc93cf9 | 2009-10-22 22:37:11 +0000 | [diff] [blame] | 2583 | ReferenceType::Profile(ID, T, false); | 
| Sebastian Redl | 0f8b23f | 2009-03-16 23:22:08 +0000 | [diff] [blame] | 2584 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2585 | void *InsertPos = nullptr; | 
| Sebastian Redl | 0f8b23f | 2009-03-16 23:22:08 +0000 | [diff] [blame] | 2586 | if (RValueReferenceType *RT = | 
|  | 2587 | RValueReferenceTypes.FindNodeOrInsertPos(ID, InsertPos)) | 
|  | 2588 | return QualType(RT, 0); | 
|  | 2589 |  | 
| John McCall | fc93cf9 | 2009-10-22 22:37:11 +0000 | [diff] [blame] | 2590 | const ReferenceType *InnerRef = T->getAs<ReferenceType>(); | 
|  | 2591 |  | 
| Sebastian Redl | 0f8b23f | 2009-03-16 23:22:08 +0000 | [diff] [blame] | 2592 | // If the referencee type isn't canonical, this won't be a canonical type | 
|  | 2593 | // either, so fill in the canonical type field. | 
|  | 2594 | QualType Canonical; | 
| John McCall | fc93cf9 | 2009-10-22 22:37:11 +0000 | [diff] [blame] | 2595 | if (InnerRef || !T.isCanonical()) { | 
|  | 2596 | QualType PointeeType = (InnerRef ? InnerRef->getPointeeType() : T); | 
|  | 2597 | Canonical = getRValueReferenceType(getCanonicalType(PointeeType)); | 
| Sebastian Redl | 0f8b23f | 2009-03-16 23:22:08 +0000 | [diff] [blame] | 2598 |  | 
|  | 2599 | // Get the new insert position for the node we care about. | 
|  | 2600 | RValueReferenceType *NewIP = | 
|  | 2601 | RValueReferenceTypes.FindNodeOrInsertPos(ID, InsertPos); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2602 | assert(!NewIP && "Shouldn't be in the map!"); (void)NewIP; | 
| Sebastian Redl | 0f8b23f | 2009-03-16 23:22:08 +0000 | [diff] [blame] | 2603 | } | 
|  | 2604 |  | 
| John McCall | 90d1c2d | 2009-09-24 23:30:46 +0000 | [diff] [blame] | 2605 | RValueReferenceType *New | 
|  | 2606 | = new (*this, TypeAlignment) RValueReferenceType(T, Canonical); | 
| Sebastian Redl | 0f8b23f | 2009-03-16 23:22:08 +0000 | [diff] [blame] | 2607 | Types.push_back(New); | 
|  | 2608 | RValueReferenceTypes.InsertNode(New, InsertPos); | 
| Bill Wendling | 3708c18 | 2007-05-27 10:15:43 +0000 | [diff] [blame] | 2609 | return QualType(New, 0); | 
|  | 2610 | } | 
|  | 2611 |  | 
| Sebastian Redl | 9ed6efd | 2009-01-24 21:16:55 +0000 | [diff] [blame] | 2612 | /// getMemberPointerType - Return the uniqued reference to the type for a | 
|  | 2613 | /// member pointer to the specified type, in the specified class. | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 2614 | QualType ASTContext::getMemberPointerType(QualType T, const Type *Cls) const { | 
| Sebastian Redl | 9ed6efd | 2009-01-24 21:16:55 +0000 | [diff] [blame] | 2615 | // Unique pointers, to guarantee there is only one pointer of a particular | 
|  | 2616 | // structure. | 
|  | 2617 | llvm::FoldingSetNodeID ID; | 
|  | 2618 | MemberPointerType::Profile(ID, T, Cls); | 
|  | 2619 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2620 | void *InsertPos = nullptr; | 
| Sebastian Redl | 9ed6efd | 2009-01-24 21:16:55 +0000 | [diff] [blame] | 2621 | if (MemberPointerType *PT = | 
|  | 2622 | MemberPointerTypes.FindNodeOrInsertPos(ID, InsertPos)) | 
|  | 2623 | return QualType(PT, 0); | 
|  | 2624 |  | 
|  | 2625 | // If the pointee or class type isn't canonical, this won't be a canonical | 
|  | 2626 | // type either, so fill in the canonical type field. | 
|  | 2627 | QualType Canonical; | 
| Douglas Gregor | 615ac67 | 2009-11-04 16:49:01 +0000 | [diff] [blame] | 2628 | if (!T.isCanonical() || !Cls->isCanonicalUnqualified()) { | 
| Sebastian Redl | 9ed6efd | 2009-01-24 21:16:55 +0000 | [diff] [blame] | 2629 | Canonical = getMemberPointerType(getCanonicalType(T),getCanonicalType(Cls)); | 
|  | 2630 |  | 
|  | 2631 | // Get the new insert position for the node we care about. | 
|  | 2632 | MemberPointerType *NewIP = | 
|  | 2633 | MemberPointerTypes.FindNodeOrInsertPos(ID, InsertPos); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2634 | assert(!NewIP && "Shouldn't be in the map!"); (void)NewIP; | 
| Sebastian Redl | 9ed6efd | 2009-01-24 21:16:55 +0000 | [diff] [blame] | 2635 | } | 
| John McCall | 90d1c2d | 2009-09-24 23:30:46 +0000 | [diff] [blame] | 2636 | MemberPointerType *New | 
|  | 2637 | = new (*this, TypeAlignment) MemberPointerType(T, Cls, Canonical); | 
| Sebastian Redl | 9ed6efd | 2009-01-24 21:16:55 +0000 | [diff] [blame] | 2638 | Types.push_back(New); | 
|  | 2639 | MemberPointerTypes.InsertNode(New, InsertPos); | 
|  | 2640 | return QualType(New, 0); | 
|  | 2641 | } | 
|  | 2642 |  | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2643 | /// getConstantArrayType - Return the unique reference to the type for an | 
| Steve Naroff | 5c13180 | 2007-08-30 01:06:46 +0000 | [diff] [blame] | 2644 | /// array of the specified element type. | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2645 | QualType ASTContext::getConstantArrayType(QualType EltTy, | 
| Chris Lattner | e2df3f9 | 2009-05-13 04:12:56 +0000 | [diff] [blame] | 2646 | const llvm::APInt &ArySizeIn, | 
| Steve Naroff | 90dfdd5 | 2007-08-30 18:10:14 +0000 | [diff] [blame] | 2647 | ArrayType::ArraySizeModifier ASM, | 
| Abramo Bagnara | 92141d2 | 2011-01-27 19:55:10 +0000 | [diff] [blame] | 2648 | unsigned IndexTypeQuals) const { | 
| Sebastian Redl | 2dfdb82 | 2009-11-05 15:52:31 +0000 | [diff] [blame] | 2649 | assert((EltTy->isDependentType() || | 
|  | 2650 | EltTy->isIncompleteType() || EltTy->isConstantSizeType()) && | 
| Eli Friedman | be7e42b | 2009-05-29 20:17:55 +0000 | [diff] [blame] | 2651 | "Constant array of VLAs is illegal!"); | 
|  | 2652 |  | 
| Chris Lattner | e2df3f9 | 2009-05-13 04:12:56 +0000 | [diff] [blame] | 2653 | // Convert the array size into a canonical width matching the pointer size for | 
|  | 2654 | // the target. | 
|  | 2655 | llvm::APInt ArySize(ArySizeIn); | 
| Jay Foad | 6d4db0c | 2010-12-07 08:25:34 +0000 | [diff] [blame] | 2656 | ArySize = | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 2657 | ArySize.zextOrTrunc(Target->getPointerWidth(getTargetAddressSpace(EltTy))); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2658 |  | 
| Chris Lattner | 23b7eb6 | 2007-06-15 23:05:46 +0000 | [diff] [blame] | 2659 | llvm::FoldingSetNodeID ID; | 
| Abramo Bagnara | 92141d2 | 2011-01-27 19:55:10 +0000 | [diff] [blame] | 2660 | ConstantArrayType::Profile(ID, EltTy, ArySize, ASM, IndexTypeQuals); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2661 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2662 | void *InsertPos = nullptr; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2663 | if (ConstantArrayType *ATP = | 
| Ted Kremenek | fc581a9 | 2007-10-31 17:10:13 +0000 | [diff] [blame] | 2664 | ConstantArrayTypes.FindNodeOrInsertPos(ID, InsertPos)) | 
| Steve Naroff | e5aa9be | 2007-04-05 22:36:20 +0000 | [diff] [blame] | 2665 | return QualType(ATP, 0); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2666 |  | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2667 | // If the element type isn't canonical or has qualifiers, this won't | 
|  | 2668 | // be a canonical type either, so fill in the canonical type field. | 
|  | 2669 | QualType Canon; | 
|  | 2670 | if (!EltTy.isCanonical() || EltTy.hasLocalQualifiers()) { | 
|  | 2671 | SplitQualType canonSplit = getCanonicalType(EltTy).split(); | 
| John McCall | 18ce25e | 2012-02-08 00:46:36 +0000 | [diff] [blame] | 2672 | Canon = getConstantArrayType(QualType(canonSplit.Ty, 0), ArySize, | 
| Abramo Bagnara | 92141d2 | 2011-01-27 19:55:10 +0000 | [diff] [blame] | 2673 | ASM, IndexTypeQuals); | 
| John McCall | 18ce25e | 2012-02-08 00:46:36 +0000 | [diff] [blame] | 2674 | Canon = getQualifiedType(Canon, canonSplit.Quals); | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2675 |  | 
| Chris Lattner | 36f8e65 | 2007-01-27 08:31:04 +0000 | [diff] [blame] | 2676 | // Get the new insert position for the node we care about. | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2677 | ConstantArrayType *NewIP = | 
| Ted Kremenek | fc581a9 | 2007-10-31 17:10:13 +0000 | [diff] [blame] | 2678 | ConstantArrayTypes.FindNodeOrInsertPos(ID, InsertPos); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2679 | assert(!NewIP && "Shouldn't be in the map!"); (void)NewIP; | 
| Chris Lattner | 36f8e65 | 2007-01-27 08:31:04 +0000 | [diff] [blame] | 2680 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2681 |  | 
| John McCall | 90d1c2d | 2009-09-24 23:30:46 +0000 | [diff] [blame] | 2682 | ConstantArrayType *New = new(*this,TypeAlignment) | 
| Abramo Bagnara | 92141d2 | 2011-01-27 19:55:10 +0000 | [diff] [blame] | 2683 | ConstantArrayType(EltTy, Canon, ArySize, ASM, IndexTypeQuals); | 
| Ted Kremenek | fc581a9 | 2007-10-31 17:10:13 +0000 | [diff] [blame] | 2684 | ConstantArrayTypes.InsertNode(New, InsertPos); | 
| Chris Lattner | 36f8e65 | 2007-01-27 08:31:04 +0000 | [diff] [blame] | 2685 | Types.push_back(New); | 
| Steve Naroff | e5aa9be | 2007-04-05 22:36:20 +0000 | [diff] [blame] | 2686 | return QualType(New, 0); | 
| Chris Lattner | 7ccecb9 | 2006-11-12 08:50:50 +0000 | [diff] [blame] | 2687 | } | 
|  | 2688 |  | 
| John McCall | 0654946 | 2011-01-18 08:40:38 +0000 | [diff] [blame] | 2689 | /// getVariableArrayDecayedType - Turns the given type, which may be | 
|  | 2690 | /// variably-modified, into the corresponding type with all the known | 
|  | 2691 | /// sizes replaced with [*]. | 
|  | 2692 | QualType ASTContext::getVariableArrayDecayedType(QualType type) const { | 
|  | 2693 | // Vastly most common case. | 
|  | 2694 | if (!type->isVariablyModifiedType()) return type; | 
| Fariborz Jahanian | 8fb87ae | 2010-09-24 17:30:16 +0000 | [diff] [blame] | 2695 |  | 
| John McCall | 0654946 | 2011-01-18 08:40:38 +0000 | [diff] [blame] | 2696 | QualType result; | 
| Fariborz Jahanian | 8fb87ae | 2010-09-24 17:30:16 +0000 | [diff] [blame] | 2697 |  | 
| John McCall | 0654946 | 2011-01-18 08:40:38 +0000 | [diff] [blame] | 2698 | SplitQualType split = type.getSplitDesugaredType(); | 
| John McCall | 18ce25e | 2012-02-08 00:46:36 +0000 | [diff] [blame] | 2699 | const Type *ty = split.Ty; | 
| John McCall | 0654946 | 2011-01-18 08:40:38 +0000 | [diff] [blame] | 2700 | switch (ty->getTypeClass()) { | 
|  | 2701 | #define TYPE(Class, Base) | 
|  | 2702 | #define ABSTRACT_TYPE(Class, Base) | 
|  | 2703 | #define NON_CANONICAL_TYPE(Class, Base) case Type::Class: | 
|  | 2704 | #include "clang/AST/TypeNodes.def" | 
|  | 2705 | llvm_unreachable("didn't desugar past all non-canonical types?"); | 
|  | 2706 |  | 
|  | 2707 | // These types should never be variably-modified. | 
|  | 2708 | case Type::Builtin: | 
|  | 2709 | case Type::Complex: | 
|  | 2710 | case Type::Vector: | 
|  | 2711 | case Type::ExtVector: | 
|  | 2712 | case Type::DependentSizedExtVector: | 
|  | 2713 | case Type::ObjCObject: | 
|  | 2714 | case Type::ObjCInterface: | 
|  | 2715 | case Type::ObjCObjectPointer: | 
|  | 2716 | case Type::Record: | 
|  | 2717 | case Type::Enum: | 
|  | 2718 | case Type::UnresolvedUsing: | 
|  | 2719 | case Type::TypeOfExpr: | 
|  | 2720 | case Type::TypeOf: | 
|  | 2721 | case Type::Decltype: | 
| Alexis Hunt | e852b10 | 2011-05-24 22:41:36 +0000 | [diff] [blame] | 2722 | case Type::UnaryTransform: | 
| John McCall | 0654946 | 2011-01-18 08:40:38 +0000 | [diff] [blame] | 2723 | case Type::DependentName: | 
|  | 2724 | case Type::InjectedClassName: | 
|  | 2725 | case Type::TemplateSpecialization: | 
|  | 2726 | case Type::DependentTemplateSpecialization: | 
|  | 2727 | case Type::TemplateTypeParm: | 
|  | 2728 | case Type::SubstTemplateTypeParmPack: | 
| Richard Smith | 30482bc | 2011-02-20 03:19:35 +0000 | [diff] [blame] | 2729 | case Type::Auto: | 
| John McCall | 0654946 | 2011-01-18 08:40:38 +0000 | [diff] [blame] | 2730 | case Type::PackExpansion: | 
|  | 2731 | llvm_unreachable("type should never be variably-modified"); | 
|  | 2732 |  | 
|  | 2733 | // These types can be variably-modified but should never need to | 
|  | 2734 | // further decay. | 
|  | 2735 | case Type::FunctionNoProto: | 
|  | 2736 | case Type::FunctionProto: | 
|  | 2737 | case Type::BlockPointer: | 
|  | 2738 | case Type::MemberPointer: | 
| Xiuli Pan | 9c14e28 | 2016-01-09 12:53:17 +0000 | [diff] [blame] | 2739 | case Type::Pipe: | 
| John McCall | 0654946 | 2011-01-18 08:40:38 +0000 | [diff] [blame] | 2740 | return type; | 
|  | 2741 |  | 
|  | 2742 | // These types can be variably-modified.  All these modifications | 
|  | 2743 | // preserve structure except as noted by comments. | 
|  | 2744 | // TODO: if we ever care about optimizing VLAs, there are no-op | 
|  | 2745 | // optimizations available here. | 
|  | 2746 | case Type::Pointer: | 
|  | 2747 | result = getPointerType(getVariableArrayDecayedType( | 
|  | 2748 | cast<PointerType>(ty)->getPointeeType())); | 
|  | 2749 | break; | 
|  | 2750 |  | 
|  | 2751 | case Type::LValueReference: { | 
|  | 2752 | const LValueReferenceType *lv = cast<LValueReferenceType>(ty); | 
|  | 2753 | result = getLValueReferenceType( | 
|  | 2754 | getVariableArrayDecayedType(lv->getPointeeType()), | 
|  | 2755 | lv->isSpelledAsLValue()); | 
|  | 2756 | break; | 
|  | 2757 | } | 
|  | 2758 |  | 
|  | 2759 | case Type::RValueReference: { | 
|  | 2760 | const RValueReferenceType *lv = cast<RValueReferenceType>(ty); | 
|  | 2761 | result = getRValueReferenceType( | 
|  | 2762 | getVariableArrayDecayedType(lv->getPointeeType())); | 
|  | 2763 | break; | 
|  | 2764 | } | 
|  | 2765 |  | 
| Eli Friedman | 0dfb889 | 2011-10-06 23:00:33 +0000 | [diff] [blame] | 2766 | case Type::Atomic: { | 
|  | 2767 | const AtomicType *at = cast<AtomicType>(ty); | 
|  | 2768 | result = getAtomicType(getVariableArrayDecayedType(at->getValueType())); | 
|  | 2769 | break; | 
|  | 2770 | } | 
|  | 2771 |  | 
| John McCall | 0654946 | 2011-01-18 08:40:38 +0000 | [diff] [blame] | 2772 | case Type::ConstantArray: { | 
|  | 2773 | const ConstantArrayType *cat = cast<ConstantArrayType>(ty); | 
|  | 2774 | result = getConstantArrayType( | 
|  | 2775 | getVariableArrayDecayedType(cat->getElementType()), | 
|  | 2776 | cat->getSize(), | 
|  | 2777 | cat->getSizeModifier(), | 
|  | 2778 | cat->getIndexTypeCVRQualifiers()); | 
|  | 2779 | break; | 
|  | 2780 | } | 
|  | 2781 |  | 
|  | 2782 | case Type::DependentSizedArray: { | 
|  | 2783 | const DependentSizedArrayType *dat = cast<DependentSizedArrayType>(ty); | 
|  | 2784 | result = getDependentSizedArrayType( | 
|  | 2785 | getVariableArrayDecayedType(dat->getElementType()), | 
|  | 2786 | dat->getSizeExpr(), | 
|  | 2787 | dat->getSizeModifier(), | 
|  | 2788 | dat->getIndexTypeCVRQualifiers(), | 
|  | 2789 | dat->getBracketsRange()); | 
|  | 2790 | break; | 
|  | 2791 | } | 
|  | 2792 |  | 
|  | 2793 | // Turn incomplete types into [*] types. | 
|  | 2794 | case Type::IncompleteArray: { | 
|  | 2795 | const IncompleteArrayType *iat = cast<IncompleteArrayType>(ty); | 
|  | 2796 | result = getVariableArrayType( | 
|  | 2797 | getVariableArrayDecayedType(iat->getElementType()), | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2798 | /*size*/ nullptr, | 
| John McCall | 0654946 | 2011-01-18 08:40:38 +0000 | [diff] [blame] | 2799 | ArrayType::Normal, | 
|  | 2800 | iat->getIndexTypeCVRQualifiers(), | 
|  | 2801 | SourceRange()); | 
|  | 2802 | break; | 
|  | 2803 | } | 
|  | 2804 |  | 
|  | 2805 | // Turn VLA types into [*] types. | 
|  | 2806 | case Type::VariableArray: { | 
|  | 2807 | const VariableArrayType *vat = cast<VariableArrayType>(ty); | 
|  | 2808 | result = getVariableArrayType( | 
|  | 2809 | getVariableArrayDecayedType(vat->getElementType()), | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2810 | /*size*/ nullptr, | 
| John McCall | 0654946 | 2011-01-18 08:40:38 +0000 | [diff] [blame] | 2811 | ArrayType::Star, | 
|  | 2812 | vat->getIndexTypeCVRQualifiers(), | 
|  | 2813 | vat->getBracketsRange()); | 
|  | 2814 | break; | 
|  | 2815 | } | 
|  | 2816 | } | 
|  | 2817 |  | 
|  | 2818 | // Apply the top-level qualifiers from the original. | 
| John McCall | 18ce25e | 2012-02-08 00:46:36 +0000 | [diff] [blame] | 2819 | return getQualifiedType(result, split.Quals); | 
| John McCall | 0654946 | 2011-01-18 08:40:38 +0000 | [diff] [blame] | 2820 | } | 
| Fariborz Jahanian | 8fb87ae | 2010-09-24 17:30:16 +0000 | [diff] [blame] | 2821 |  | 
| Steve Naroff | cadebd0 | 2007-08-30 18:14:25 +0000 | [diff] [blame] | 2822 | /// getVariableArrayType - Returns a non-unique reference to the type for a | 
|  | 2823 | /// variable array of the specified element type. | 
| Douglas Gregor | 0431825 | 2009-07-06 15:59:29 +0000 | [diff] [blame] | 2824 | QualType ASTContext::getVariableArrayType(QualType EltTy, | 
|  | 2825 | Expr *NumElts, | 
| Steve Naroff | 90dfdd5 | 2007-08-30 18:10:14 +0000 | [diff] [blame] | 2826 | ArrayType::ArraySizeModifier ASM, | 
| Abramo Bagnara | 92141d2 | 2011-01-27 19:55:10 +0000 | [diff] [blame] | 2827 | unsigned IndexTypeQuals, | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 2828 | SourceRange Brackets) const { | 
| Eli Friedman | bd25828 | 2008-02-15 18:16:39 +0000 | [diff] [blame] | 2829 | // Since we don't unique expressions, it isn't possible to unique VLA's | 
|  | 2830 | // that have an expression provided for their size. | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2831 | QualType Canon; | 
| Douglas Gregor | 5e8c8c0 | 2010-05-23 16:10:32 +0000 | [diff] [blame] | 2832 |  | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2833 | // Be sure to pull qualifiers off the element type. | 
|  | 2834 | if (!EltTy.isCanonical() || EltTy.hasLocalQualifiers()) { | 
|  | 2835 | SplitQualType canonSplit = getCanonicalType(EltTy).split(); | 
| John McCall | 18ce25e | 2012-02-08 00:46:36 +0000 | [diff] [blame] | 2836 | Canon = getVariableArrayType(QualType(canonSplit.Ty, 0), NumElts, ASM, | 
| Abramo Bagnara | 92141d2 | 2011-01-27 19:55:10 +0000 | [diff] [blame] | 2837 | IndexTypeQuals, Brackets); | 
| John McCall | 18ce25e | 2012-02-08 00:46:36 +0000 | [diff] [blame] | 2838 | Canon = getQualifiedType(Canon, canonSplit.Quals); | 
| Douglas Gregor | 5e8c8c0 | 2010-05-23 16:10:32 +0000 | [diff] [blame] | 2839 | } | 
|  | 2840 |  | 
| John McCall | 90d1c2d | 2009-09-24 23:30:46 +0000 | [diff] [blame] | 2841 | VariableArrayType *New = new(*this, TypeAlignment) | 
| Abramo Bagnara | 92141d2 | 2011-01-27 19:55:10 +0000 | [diff] [blame] | 2842 | VariableArrayType(EltTy, Canon, NumElts, ASM, IndexTypeQuals, Brackets); | 
| Eli Friedman | bd25828 | 2008-02-15 18:16:39 +0000 | [diff] [blame] | 2843 |  | 
|  | 2844 | VariableArrayTypes.push_back(New); | 
|  | 2845 | Types.push_back(New); | 
|  | 2846 | return QualType(New, 0); | 
|  | 2847 | } | 
|  | 2848 |  | 
| Douglas Gregor | 4619e43 | 2008-12-05 23:32:09 +0000 | [diff] [blame] | 2849 | /// getDependentSizedArrayType - Returns a non-unique reference to | 
|  | 2850 | /// the type for a dependently-sized array of the specified element | 
| Douglas Gregor | f3f9552 | 2009-07-31 00:23:35 +0000 | [diff] [blame] | 2851 | /// type. | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2852 | QualType ASTContext::getDependentSizedArrayType(QualType elementType, | 
|  | 2853 | Expr *numElements, | 
| Douglas Gregor | 4619e43 | 2008-12-05 23:32:09 +0000 | [diff] [blame] | 2854 | ArrayType::ArraySizeModifier ASM, | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2855 | unsigned elementTypeQuals, | 
|  | 2856 | SourceRange brackets) const { | 
|  | 2857 | assert((!numElements || numElements->isTypeDependent() || | 
|  | 2858 | numElements->isValueDependent()) && | 
| Douglas Gregor | 4619e43 | 2008-12-05 23:32:09 +0000 | [diff] [blame] | 2859 | "Size must be type- or value-dependent!"); | 
|  | 2860 |  | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2861 | // Dependently-sized array types that do not have a specified number | 
|  | 2862 | // of elements will have their sizes deduced from a dependent | 
|  | 2863 | // initializer.  We do no canonicalization here at all, which is okay | 
|  | 2864 | // because they can't be used in most locations. | 
|  | 2865 | if (!numElements) { | 
|  | 2866 | DependentSizedArrayType *newType | 
|  | 2867 | = new (*this, TypeAlignment) | 
|  | 2868 | DependentSizedArrayType(*this, elementType, QualType(), | 
|  | 2869 | numElements, ASM, elementTypeQuals, | 
|  | 2870 | brackets); | 
|  | 2871 | Types.push_back(newType); | 
|  | 2872 | return QualType(newType, 0); | 
|  | 2873 | } | 
|  | 2874 |  | 
|  | 2875 | // Otherwise, we actually build a new type every time, but we | 
|  | 2876 | // also build a canonical type. | 
|  | 2877 |  | 
|  | 2878 | SplitQualType canonElementType = getCanonicalType(elementType).split(); | 
|  | 2879 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2880 | void *insertPos = nullptr; | 
| Douglas Gregor | c42075a | 2010-02-04 18:10:26 +0000 | [diff] [blame] | 2881 | llvm::FoldingSetNodeID ID; | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2882 | DependentSizedArrayType::Profile(ID, *this, | 
| John McCall | 18ce25e | 2012-02-08 00:46:36 +0000 | [diff] [blame] | 2883 | QualType(canonElementType.Ty, 0), | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2884 | ASM, elementTypeQuals, numElements); | 
| Douglas Gregor | ad2956c | 2009-11-19 18:03:26 +0000 | [diff] [blame] | 2885 |  | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2886 | // Look for an existing type with these properties. | 
|  | 2887 | DependentSizedArrayType *canonTy = | 
|  | 2888 | DependentSizedArrayTypes.FindNodeOrInsertPos(ID, insertPos); | 
| Douglas Gregor | ad2956c | 2009-11-19 18:03:26 +0000 | [diff] [blame] | 2889 |  | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2890 | // If we don't have one, build one. | 
|  | 2891 | if (!canonTy) { | 
|  | 2892 | canonTy = new (*this, TypeAlignment) | 
| John McCall | 18ce25e | 2012-02-08 00:46:36 +0000 | [diff] [blame] | 2893 | DependentSizedArrayType(*this, QualType(canonElementType.Ty, 0), | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2894 | QualType(), numElements, ASM, elementTypeQuals, | 
|  | 2895 | brackets); | 
|  | 2896 | DependentSizedArrayTypes.InsertNode(canonTy, insertPos); | 
|  | 2897 | Types.push_back(canonTy); | 
| Douglas Gregor | ad2956c | 2009-11-19 18:03:26 +0000 | [diff] [blame] | 2898 | } | 
|  | 2899 |  | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2900 | // Apply qualifiers from the element type to the array. | 
|  | 2901 | QualType canon = getQualifiedType(QualType(canonTy,0), | 
| John McCall | 18ce25e | 2012-02-08 00:46:36 +0000 | [diff] [blame] | 2902 | canonElementType.Quals); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2903 |  | 
| David Majnemer | 16a7470 | 2015-07-24 05:54:19 +0000 | [diff] [blame] | 2904 | // If we didn't need extra canonicalization for the element type or the size | 
|  | 2905 | // expression, then just use that as our result. | 
|  | 2906 | if (QualType(canonElementType.Ty, 0) == elementType && | 
|  | 2907 | canonTy->getSizeExpr() == numElements) | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2908 | return canon; | 
|  | 2909 |  | 
|  | 2910 | // Otherwise, we need to build a type which follows the spelling | 
|  | 2911 | // of the element type. | 
|  | 2912 | DependentSizedArrayType *sugaredType | 
|  | 2913 | = new (*this, TypeAlignment) | 
|  | 2914 | DependentSizedArrayType(*this, elementType, canon, numElements, | 
|  | 2915 | ASM, elementTypeQuals, brackets); | 
|  | 2916 | Types.push_back(sugaredType); | 
|  | 2917 | return QualType(sugaredType, 0); | 
| Douglas Gregor | 4619e43 | 2008-12-05 23:32:09 +0000 | [diff] [blame] | 2918 | } | 
|  | 2919 |  | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2920 | QualType ASTContext::getIncompleteArrayType(QualType elementType, | 
| Eli Friedman | bd25828 | 2008-02-15 18:16:39 +0000 | [diff] [blame] | 2921 | ArrayType::ArraySizeModifier ASM, | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2922 | unsigned elementTypeQuals) const { | 
| Eli Friedman | bd25828 | 2008-02-15 18:16:39 +0000 | [diff] [blame] | 2923 | llvm::FoldingSetNodeID ID; | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2924 | IncompleteArrayType::Profile(ID, elementType, ASM, elementTypeQuals); | 
| Eli Friedman | bd25828 | 2008-02-15 18:16:39 +0000 | [diff] [blame] | 2925 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2926 | void *insertPos = nullptr; | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2927 | if (IncompleteArrayType *iat = | 
|  | 2928 | IncompleteArrayTypes.FindNodeOrInsertPos(ID, insertPos)) | 
|  | 2929 | return QualType(iat, 0); | 
| Eli Friedman | bd25828 | 2008-02-15 18:16:39 +0000 | [diff] [blame] | 2930 |  | 
|  | 2931 | // If the element type isn't canonical, this won't be a canonical type | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2932 | // either, so fill in the canonical type field.  We also have to pull | 
|  | 2933 | // qualifiers off the element type. | 
|  | 2934 | QualType canon; | 
| Eli Friedman | bd25828 | 2008-02-15 18:16:39 +0000 | [diff] [blame] | 2935 |  | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2936 | if (!elementType.isCanonical() || elementType.hasLocalQualifiers()) { | 
|  | 2937 | SplitQualType canonSplit = getCanonicalType(elementType).split(); | 
| John McCall | 18ce25e | 2012-02-08 00:46:36 +0000 | [diff] [blame] | 2938 | canon = getIncompleteArrayType(QualType(canonSplit.Ty, 0), | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2939 | ASM, elementTypeQuals); | 
| John McCall | 18ce25e | 2012-02-08 00:46:36 +0000 | [diff] [blame] | 2940 | canon = getQualifiedType(canon, canonSplit.Quals); | 
| Eli Friedman | bd25828 | 2008-02-15 18:16:39 +0000 | [diff] [blame] | 2941 |  | 
|  | 2942 | // Get the new insert position for the node we care about. | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2943 | IncompleteArrayType *existing = | 
|  | 2944 | IncompleteArrayTypes.FindNodeOrInsertPos(ID, insertPos); | 
|  | 2945 | assert(!existing && "Shouldn't be in the map!"); (void) existing; | 
| Ted Kremenek | 843ebedd | 2007-10-29 23:37:31 +0000 | [diff] [blame] | 2946 | } | 
| Eli Friedman | bd25828 | 2008-02-15 18:16:39 +0000 | [diff] [blame] | 2947 |  | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2948 | IncompleteArrayType *newType = new (*this, TypeAlignment) | 
|  | 2949 | IncompleteArrayType(elementType, canon, ASM, elementTypeQuals); | 
| Eli Friedman | bd25828 | 2008-02-15 18:16:39 +0000 | [diff] [blame] | 2950 |  | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2951 | IncompleteArrayTypes.InsertNode(newType, insertPos); | 
|  | 2952 | Types.push_back(newType); | 
|  | 2953 | return QualType(newType, 0); | 
| Steve Naroff | 5c13180 | 2007-08-30 01:06:46 +0000 | [diff] [blame] | 2954 | } | 
|  | 2955 |  | 
| Steve Naroff | 91fcddb | 2007-07-18 18:00:27 +0000 | [diff] [blame] | 2956 | /// getVectorType - Return the unique reference to a vector type of | 
|  | 2957 | /// the specified element type and size. VectorType must be a built-in type. | 
| John Thompson | 2233460 | 2010-02-05 00:12:22 +0000 | [diff] [blame] | 2958 | QualType ASTContext::getVectorType(QualType vecType, unsigned NumElts, | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 2959 | VectorType::VectorKind VecKind) const { | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 2960 | assert(vecType->isBuiltinType()); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2961 |  | 
| Steve Naroff | 4ae0ac6 | 2007-07-06 23:09:18 +0000 | [diff] [blame] | 2962 | // Check if we've already instantiated a vector of this type. | 
|  | 2963 | llvm::FoldingSetNodeID ID; | 
| Bob Wilson | aeb5644 | 2010-11-10 21:56:12 +0000 | [diff] [blame] | 2964 | VectorType::Profile(ID, vecType, NumElts, Type::Vector, VecKind); | 
| Chris Lattner | 37141f4 | 2010-06-23 06:00:24 +0000 | [diff] [blame] | 2965 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2966 | void *InsertPos = nullptr; | 
| Steve Naroff | 4ae0ac6 | 2007-07-06 23:09:18 +0000 | [diff] [blame] | 2967 | if (VectorType *VTP = VectorTypes.FindNodeOrInsertPos(ID, InsertPos)) | 
|  | 2968 | return QualType(VTP, 0); | 
|  | 2969 |  | 
|  | 2970 | // If the element type isn't canonical, this won't be a canonical type either, | 
|  | 2971 | // so fill in the canonical type field. | 
|  | 2972 | QualType Canonical; | 
| Douglas Gregor | 59e8b3b | 2010-08-06 10:14:59 +0000 | [diff] [blame] | 2973 | if (!vecType.isCanonical()) { | 
| Bob Wilson | 7795480 | 2010-11-16 00:32:20 +0000 | [diff] [blame] | 2974 | Canonical = getVectorType(getCanonicalType(vecType), NumElts, VecKind); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2975 |  | 
| Steve Naroff | 4ae0ac6 | 2007-07-06 23:09:18 +0000 | [diff] [blame] | 2976 | // Get the new insert position for the node we care about. | 
|  | 2977 | VectorType *NewIP = VectorTypes.FindNodeOrInsertPos(ID, InsertPos); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2978 | assert(!NewIP && "Shouldn't be in the map!"); (void)NewIP; | 
| Steve Naroff | 4ae0ac6 | 2007-07-06 23:09:18 +0000 | [diff] [blame] | 2979 | } | 
| John McCall | 90d1c2d | 2009-09-24 23:30:46 +0000 | [diff] [blame] | 2980 | VectorType *New = new (*this, TypeAlignment) | 
| Bob Wilson | aeb5644 | 2010-11-10 21:56:12 +0000 | [diff] [blame] | 2981 | VectorType(vecType, NumElts, Canonical, VecKind); | 
| Steve Naroff | 4ae0ac6 | 2007-07-06 23:09:18 +0000 | [diff] [blame] | 2982 | VectorTypes.InsertNode(New, InsertPos); | 
|  | 2983 | Types.push_back(New); | 
|  | 2984 | return QualType(New, 0); | 
|  | 2985 | } | 
|  | 2986 |  | 
| Nate Begeman | ce4d7fc | 2008-04-18 23:10:10 +0000 | [diff] [blame] | 2987 | /// getExtVectorType - Return the unique reference to an extended vector type of | 
| Steve Naroff | 91fcddb | 2007-07-18 18:00:27 +0000 | [diff] [blame] | 2988 | /// the specified element type and size. VectorType must be a built-in type. | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 2989 | QualType | 
|  | 2990 | ASTContext::getExtVectorType(QualType vecType, unsigned NumElts) const { | 
| Douglas Gregor | 39c0272 | 2011-06-15 16:02:29 +0000 | [diff] [blame] | 2991 | assert(vecType->isBuiltinType() || vecType->isDependentType()); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 2992 |  | 
| Steve Naroff | 91fcddb | 2007-07-18 18:00:27 +0000 | [diff] [blame] | 2993 | // Check if we've already instantiated a vector of this type. | 
|  | 2994 | llvm::FoldingSetNodeID ID; | 
| Chris Lattner | 37141f4 | 2010-06-23 06:00:24 +0000 | [diff] [blame] | 2995 | VectorType::Profile(ID, vecType, NumElts, Type::ExtVector, | 
| Bob Wilson | aeb5644 | 2010-11-10 21:56:12 +0000 | [diff] [blame] | 2996 | VectorType::GenericVector); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 2997 | void *InsertPos = nullptr; | 
| Steve Naroff | 91fcddb | 2007-07-18 18:00:27 +0000 | [diff] [blame] | 2998 | if (VectorType *VTP = VectorTypes.FindNodeOrInsertPos(ID, InsertPos)) | 
|  | 2999 | return QualType(VTP, 0); | 
|  | 3000 |  | 
|  | 3001 | // If the element type isn't canonical, this won't be a canonical type either, | 
|  | 3002 | // so fill in the canonical type field. | 
|  | 3003 | QualType Canonical; | 
| John McCall | b692a09 | 2009-10-22 20:10:53 +0000 | [diff] [blame] | 3004 | if (!vecType.isCanonical()) { | 
| Nate Begeman | ce4d7fc | 2008-04-18 23:10:10 +0000 | [diff] [blame] | 3005 | Canonical = getExtVectorType(getCanonicalType(vecType), NumElts); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3006 |  | 
| Steve Naroff | 91fcddb | 2007-07-18 18:00:27 +0000 | [diff] [blame] | 3007 | // Get the new insert position for the node we care about. | 
|  | 3008 | VectorType *NewIP = VectorTypes.FindNodeOrInsertPos(ID, InsertPos); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 3009 | assert(!NewIP && "Shouldn't be in the map!"); (void)NewIP; | 
| Steve Naroff | 91fcddb | 2007-07-18 18:00:27 +0000 | [diff] [blame] | 3010 | } | 
| John McCall | 90d1c2d | 2009-09-24 23:30:46 +0000 | [diff] [blame] | 3011 | ExtVectorType *New = new (*this, TypeAlignment) | 
|  | 3012 | ExtVectorType(vecType, NumElts, Canonical); | 
| Steve Naroff | 91fcddb | 2007-07-18 18:00:27 +0000 | [diff] [blame] | 3013 | VectorTypes.InsertNode(New, InsertPos); | 
|  | 3014 | Types.push_back(New); | 
|  | 3015 | return QualType(New, 0); | 
|  | 3016 | } | 
|  | 3017 |  | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 3018 | QualType | 
|  | 3019 | ASTContext::getDependentSizedExtVectorType(QualType vecType, | 
|  | 3020 | Expr *SizeExpr, | 
|  | 3021 | SourceLocation AttrLoc) const { | 
| Douglas Gregor | 352169a | 2009-07-31 03:54:25 +0000 | [diff] [blame] | 3022 | llvm::FoldingSetNodeID ID; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3023 | DependentSizedExtVectorType::Profile(ID, *this, getCanonicalType(vecType), | 
| Douglas Gregor | 352169a | 2009-07-31 03:54:25 +0000 | [diff] [blame] | 3024 | SizeExpr); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3025 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 3026 | void *InsertPos = nullptr; | 
| Douglas Gregor | 352169a | 2009-07-31 03:54:25 +0000 | [diff] [blame] | 3027 | DependentSizedExtVectorType *Canon | 
|  | 3028 | = DependentSizedExtVectorTypes.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 3029 | DependentSizedExtVectorType *New; | 
|  | 3030 | if (Canon) { | 
|  | 3031 | // We already have a canonical version of this array type; use it as | 
|  | 3032 | // the canonical type for a newly-built type. | 
| John McCall | 90d1c2d | 2009-09-24 23:30:46 +0000 | [diff] [blame] | 3033 | New = new (*this, TypeAlignment) | 
|  | 3034 | DependentSizedExtVectorType(*this, vecType, QualType(Canon, 0), | 
|  | 3035 | SizeExpr, AttrLoc); | 
| Douglas Gregor | 352169a | 2009-07-31 03:54:25 +0000 | [diff] [blame] | 3036 | } else { | 
|  | 3037 | QualType CanonVecTy = getCanonicalType(vecType); | 
|  | 3038 | if (CanonVecTy == vecType) { | 
| John McCall | 90d1c2d | 2009-09-24 23:30:46 +0000 | [diff] [blame] | 3039 | New = new (*this, TypeAlignment) | 
|  | 3040 | DependentSizedExtVectorType(*this, vecType, QualType(), SizeExpr, | 
|  | 3041 | AttrLoc); | 
| Douglas Gregor | c42075a | 2010-02-04 18:10:26 +0000 | [diff] [blame] | 3042 |  | 
|  | 3043 | DependentSizedExtVectorType *CanonCheck | 
|  | 3044 | = DependentSizedExtVectorTypes.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 3045 | assert(!CanonCheck && "Dependent-sized ext_vector canonical type broken"); | 
|  | 3046 | (void)CanonCheck; | 
| Douglas Gregor | 352169a | 2009-07-31 03:54:25 +0000 | [diff] [blame] | 3047 | DependentSizedExtVectorTypes.InsertNode(New, InsertPos); | 
|  | 3048 | } else { | 
|  | 3049 | QualType Canon = getDependentSizedExtVectorType(CanonVecTy, SizeExpr, | 
|  | 3050 | SourceLocation()); | 
| John McCall | 90d1c2d | 2009-09-24 23:30:46 +0000 | [diff] [blame] | 3051 | New = new (*this, TypeAlignment) | 
|  | 3052 | DependentSizedExtVectorType(*this, vecType, Canon, SizeExpr, AttrLoc); | 
| Douglas Gregor | 352169a | 2009-07-31 03:54:25 +0000 | [diff] [blame] | 3053 | } | 
|  | 3054 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3055 |  | 
| Douglas Gregor | 758a869 | 2009-06-17 21:51:59 +0000 | [diff] [blame] | 3056 | Types.push_back(New); | 
|  | 3057 | return QualType(New, 0); | 
|  | 3058 | } | 
|  | 3059 |  | 
| John McCall | 18afab7 | 2016-03-01 00:49:02 +0000 | [diff] [blame] | 3060 | /// \brief Determine whether \p T is canonical as the result type of a function. | 
|  | 3061 | static bool isCanonicalResultType(QualType T) { | 
|  | 3062 | return T.isCanonical() && | 
|  | 3063 | (T.getObjCLifetime() == Qualifiers::OCL_None || | 
|  | 3064 | T.getObjCLifetime() == Qualifiers::OCL_ExplicitNone); | 
|  | 3065 | } | 
|  | 3066 |  | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 3067 | /// getFunctionNoProtoType - Return a K&R style C function type like 'int()'. | 
| Chris Lattner | c6ad813 | 2006-12-02 07:52:18 +0000 | [diff] [blame] | 3068 | /// | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 3069 | QualType | 
|  | 3070 | ASTContext::getFunctionNoProtoType(QualType ResultTy, | 
|  | 3071 | const FunctionType::ExtInfo &Info) const { | 
| Chris Lattner | c6ad813 | 2006-12-02 07:52:18 +0000 | [diff] [blame] | 3072 | // Unique functions, to guarantee there is only one function of a particular | 
|  | 3073 | // structure. | 
| Chris Lattner | 23b7eb6 | 2007-06-15 23:05:46 +0000 | [diff] [blame] | 3074 | llvm::FoldingSetNodeID ID; | 
| Rafael Espindola | c50c27c | 2010-03-30 20:24:48 +0000 | [diff] [blame] | 3075 | FunctionNoProtoType::Profile(ID, ResultTy, Info); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3076 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 3077 | void *InsertPos = nullptr; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3078 | if (FunctionNoProtoType *FT = | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 3079 | FunctionNoProtoTypes.FindNodeOrInsertPos(ID, InsertPos)) | 
| Steve Naroff | e5aa9be | 2007-04-05 22:36:20 +0000 | [diff] [blame] | 3080 | return QualType(FT, 0); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3081 |  | 
| Steve Naroff | e5aa9be | 2007-04-05 22:36:20 +0000 | [diff] [blame] | 3082 | QualType Canonical; | 
| John McCall | 18afab7 | 2016-03-01 00:49:02 +0000 | [diff] [blame] | 3083 | if (!isCanonicalResultType(ResultTy)) { | 
|  | 3084 | Canonical = | 
|  | 3085 | getFunctionNoProtoType(getCanonicalFunctionResultType(ResultTy), Info); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3086 |  | 
| Chris Lattner | 47955de | 2007-01-27 08:37:20 +0000 | [diff] [blame] | 3087 | // Get the new insert position for the node we care about. | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 3088 | FunctionNoProtoType *NewIP = | 
|  | 3089 | FunctionNoProtoTypes.FindNodeOrInsertPos(ID, InsertPos); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 3090 | assert(!NewIP && "Shouldn't be in the map!"); (void)NewIP; | 
| Chris Lattner | 47955de | 2007-01-27 08:37:20 +0000 | [diff] [blame] | 3091 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3092 |  | 
| John McCall | 90d1c2d | 2009-09-24 23:30:46 +0000 | [diff] [blame] | 3093 | FunctionNoProtoType *New = new (*this, TypeAlignment) | 
| John McCall | 18afab7 | 2016-03-01 00:49:02 +0000 | [diff] [blame] | 3094 | FunctionNoProtoType(ResultTy, Canonical, Info); | 
| Chris Lattner | 47955de | 2007-01-27 08:37:20 +0000 | [diff] [blame] | 3095 | Types.push_back(New); | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 3096 | FunctionNoProtoTypes.InsertNode(New, InsertPos); | 
| Steve Naroff | e5aa9be | 2007-04-05 22:36:20 +0000 | [diff] [blame] | 3097 | return QualType(New, 0); | 
| Chris Lattner | c6ad813 | 2006-12-02 07:52:18 +0000 | [diff] [blame] | 3098 | } | 
|  | 3099 |  | 
| Douglas Gregor | a602a15 | 2015-10-01 20:20:47 +0000 | [diff] [blame] | 3100 | CanQualType | 
|  | 3101 | ASTContext::getCanonicalFunctionResultType(QualType ResultType) const { | 
|  | 3102 | CanQualType CanResultType = getCanonicalType(ResultType); | 
|  | 3103 |  | 
|  | 3104 | // Canonical result types do not have ARC lifetime qualifiers. | 
|  | 3105 | if (CanResultType.getQualifiers().hasObjCLifetime()) { | 
|  | 3106 | Qualifiers Qs = CanResultType.getQualifiers(); | 
|  | 3107 | Qs.removeObjCLifetime(); | 
|  | 3108 | return CanQualType::CreateUnsafe( | 
|  | 3109 | getQualifiedType(CanResultType.getUnqualifiedType(), Qs)); | 
|  | 3110 | } | 
|  | 3111 |  | 
|  | 3112 | return CanResultType; | 
|  | 3113 | } | 
|  | 3114 |  | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 3115 | QualType | 
| Jordan Rose | 5c38272 | 2013-03-08 21:51:21 +0000 | [diff] [blame] | 3116 | ASTContext::getFunctionType(QualType ResultTy, ArrayRef<QualType> ArgArray, | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 3117 | const FunctionProtoType::ExtProtoInfo &EPI) const { | 
| Jordan Rose | 5c38272 | 2013-03-08 21:51:21 +0000 | [diff] [blame] | 3118 | size_t NumArgs = ArgArray.size(); | 
|  | 3119 |  | 
| Chris Lattner | c6ad813 | 2006-12-02 07:52:18 +0000 | [diff] [blame] | 3120 | // Unique functions, to guarantee there is only one function of a particular | 
|  | 3121 | // structure. | 
| Chris Lattner | 23b7eb6 | 2007-06-15 23:05:46 +0000 | [diff] [blame] | 3122 | llvm::FoldingSetNodeID ID; | 
| Jordan Rose | 5c38272 | 2013-03-08 21:51:21 +0000 | [diff] [blame] | 3123 | FunctionProtoType::Profile(ID, ResultTy, ArgArray.begin(), NumArgs, EPI, | 
|  | 3124 | *this); | 
| Chris Lattner | fd4de79 | 2007-01-27 01:15:32 +0000 | [diff] [blame] | 3125 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 3126 | void *InsertPos = nullptr; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3127 | if (FunctionProtoType *FTP = | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 3128 | FunctionProtoTypes.FindNodeOrInsertPos(ID, InsertPos)) | 
| Steve Naroff | e5aa9be | 2007-04-05 22:36:20 +0000 | [diff] [blame] | 3129 | return QualType(FTP, 0); | 
| Sebastian Redl | 5068f77ac | 2009-05-27 22:11:52 +0000 | [diff] [blame] | 3130 |  | 
|  | 3131 | // Determine whether the type being created is already canonical or not. | 
| Richard Smith | 5e58029 | 2012-02-10 09:58:53 +0000 | [diff] [blame] | 3132 | bool isCanonical = | 
| Richard Smith | 8acb428 | 2014-07-31 21:57:55 +0000 | [diff] [blame] | 3133 | EPI.ExceptionSpec.Type == EST_None && isCanonicalResultType(ResultTy) && | 
| Richard Smith | 5e58029 | 2012-02-10 09:58:53 +0000 | [diff] [blame] | 3134 | !EPI.HasTrailingReturn; | 
| Chris Lattner | c6ad813 | 2006-12-02 07:52:18 +0000 | [diff] [blame] | 3135 | for (unsigned i = 0; i != NumArgs && isCanonical; ++i) | 
| John McCall | fc93cf9 | 2009-10-22 22:37:11 +0000 | [diff] [blame] | 3136 | if (!ArgArray[i].isCanonicalAsParam()) | 
| Chris Lattner | c6ad813 | 2006-12-02 07:52:18 +0000 | [diff] [blame] | 3137 | isCanonical = false; | 
|  | 3138 |  | 
|  | 3139 | // If this type isn't canonical, get the canonical version of it. | 
| Sebastian Redl | 5068f77ac | 2009-05-27 22:11:52 +0000 | [diff] [blame] | 3140 | // The exception spec is not part of the canonical type. | 
| Steve Naroff | e5aa9be | 2007-04-05 22:36:20 +0000 | [diff] [blame] | 3141 | QualType Canonical; | 
| Reid Kleckner | 78af070 | 2013-08-27 23:08:25 +0000 | [diff] [blame] | 3142 | if (!isCanonical) { | 
| Chris Lattner | 0e62c1c | 2011-07-23 10:55:15 +0000 | [diff] [blame] | 3143 | SmallVector<QualType, 16> CanonicalArgs; | 
| Chris Lattner | c6ad813 | 2006-12-02 07:52:18 +0000 | [diff] [blame] | 3144 | CanonicalArgs.reserve(NumArgs); | 
|  | 3145 | for (unsigned i = 0; i != NumArgs; ++i) | 
| John McCall | fc93cf9 | 2009-10-22 22:37:11 +0000 | [diff] [blame] | 3146 | CanonicalArgs.push_back(getCanonicalParamType(ArgArray[i])); | 
| Sebastian Redl | 5068f77ac | 2009-05-27 22:11:52 +0000 | [diff] [blame] | 3147 |  | 
| John McCall | db40c7f | 2010-12-14 08:05:40 +0000 | [diff] [blame] | 3148 | FunctionProtoType::ExtProtoInfo CanonicalEPI = EPI; | 
| Richard Smith | 5e58029 | 2012-02-10 09:58:53 +0000 | [diff] [blame] | 3149 | CanonicalEPI.HasTrailingReturn = false; | 
| Richard Smith | 8acb428 | 2014-07-31 21:57:55 +0000 | [diff] [blame] | 3150 | CanonicalEPI.ExceptionSpec = FunctionProtoType::ExceptionSpecInfo(); | 
| John McCall | db40c7f | 2010-12-14 08:05:40 +0000 | [diff] [blame] | 3151 |  | 
| Douglas Gregor | a602a15 | 2015-10-01 20:20:47 +0000 | [diff] [blame] | 3152 | // Adjust the canonical function result type. | 
|  | 3153 | CanQualType CanResultTy = getCanonicalFunctionResultType(ResultTy); | 
| Jordan Rose | 5c38272 | 2013-03-08 21:51:21 +0000 | [diff] [blame] | 3154 | Canonical = getFunctionType(CanResultTy, CanonicalArgs, CanonicalEPI); | 
| Sebastian Redl | 5068f77ac | 2009-05-27 22:11:52 +0000 | [diff] [blame] | 3155 |  | 
| Chris Lattner | fd4de79 | 2007-01-27 01:15:32 +0000 | [diff] [blame] | 3156 | // Get the new insert position for the node we care about. | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 3157 | FunctionProtoType *NewIP = | 
|  | 3158 | FunctionProtoTypes.FindNodeOrInsertPos(ID, InsertPos); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 3159 | assert(!NewIP && "Shouldn't be in the map!"); (void)NewIP; | 
| Chris Lattner | c6ad813 | 2006-12-02 07:52:18 +0000 | [diff] [blame] | 3160 | } | 
| Sebastian Redl | 5068f77ac | 2009-05-27 22:11:52 +0000 | [diff] [blame] | 3161 |  | 
| John McCall | 31168b0 | 2011-06-15 23:02:42 +0000 | [diff] [blame] | 3162 | // FunctionProtoType objects are allocated with extra bytes after | 
|  | 3163 | // them for three variable size arrays at the end: | 
|  | 3164 | //  - parameter types | 
|  | 3165 | //  - exception types | 
| John McCall | 18afab7 | 2016-03-01 00:49:02 +0000 | [diff] [blame] | 3166 | //  - extended parameter information | 
| John McCall | 31168b0 | 2011-06-15 23:02:42 +0000 | [diff] [blame] | 3167 | // Instead of the exception types, there could be a noexcept | 
| Richard Smith | d3b5c908 | 2012-07-27 04:22:15 +0000 | [diff] [blame] | 3168 | // expression, or information used to resolve the exception | 
|  | 3169 | // specification. | 
| John McCall | db40c7f | 2010-12-14 08:05:40 +0000 | [diff] [blame] | 3170 | size_t Size = sizeof(FunctionProtoType) + | 
| Sebastian Redl | fa453cf | 2011-03-12 11:50:43 +0000 | [diff] [blame] | 3171 | NumArgs * sizeof(QualType); | 
| John McCall | 18afab7 | 2016-03-01 00:49:02 +0000 | [diff] [blame] | 3172 |  | 
| Richard Smith | 8acb428 | 2014-07-31 21:57:55 +0000 | [diff] [blame] | 3173 | if (EPI.ExceptionSpec.Type == EST_Dynamic) { | 
|  | 3174 | Size += EPI.ExceptionSpec.Exceptions.size() * sizeof(QualType); | 
|  | 3175 | } else if (EPI.ExceptionSpec.Type == EST_ComputedNoexcept) { | 
| Sebastian Redl | 31ad754 | 2011-03-13 17:09:40 +0000 | [diff] [blame] | 3176 | Size += sizeof(Expr*); | 
| Richard Smith | 8acb428 | 2014-07-31 21:57:55 +0000 | [diff] [blame] | 3177 | } else if (EPI.ExceptionSpec.Type == EST_Uninstantiated) { | 
| Richard Smith | d372942 | 2012-04-19 00:08:28 +0000 | [diff] [blame] | 3178 | Size += 2 * sizeof(FunctionDecl*); | 
| Richard Smith | 8acb428 | 2014-07-31 21:57:55 +0000 | [diff] [blame] | 3179 | } else if (EPI.ExceptionSpec.Type == EST_Unevaluated) { | 
| Richard Smith | d3b5c908 | 2012-07-27 04:22:15 +0000 | [diff] [blame] | 3180 | Size += sizeof(FunctionDecl*); | 
| Sebastian Redl | fa453cf | 2011-03-12 11:50:43 +0000 | [diff] [blame] | 3181 | } | 
| John McCall | 18afab7 | 2016-03-01 00:49:02 +0000 | [diff] [blame] | 3182 |  | 
|  | 3183 | // Put the ExtParameterInfos last.  If all were equal, it would make | 
|  | 3184 | // more sense to put these before the exception specification, because | 
|  | 3185 | // it's much easier to skip past them compared to the elaborate switch | 
|  | 3186 | // required to skip the exception specification.  However, all is not | 
|  | 3187 | // equal; ExtParameterInfos are used to model very uncommon features, | 
|  | 3188 | // and it's better not to burden the more common paths. | 
|  | 3189 | if (EPI.ExtParameterInfos) { | 
|  | 3190 | Size += NumArgs * sizeof(FunctionProtoType::ExtParameterInfo); | 
|  | 3191 | } | 
| John McCall | 31168b0 | 2011-06-15 23:02:42 +0000 | [diff] [blame] | 3192 |  | 
| John McCall | db40c7f | 2010-12-14 08:05:40 +0000 | [diff] [blame] | 3193 | FunctionProtoType *FTP = (FunctionProtoType*) Allocate(Size, TypeAlignment); | 
| Roman Divacky | 65b88cd | 2011-03-01 17:40:53 +0000 | [diff] [blame] | 3194 | FunctionProtoType::ExtProtoInfo newEPI = EPI; | 
| Jordan Rose | 5c38272 | 2013-03-08 21:51:21 +0000 | [diff] [blame] | 3195 | new (FTP) FunctionProtoType(ResultTy, ArgArray, Canonical, newEPI); | 
| Chris Lattner | c6ad813 | 2006-12-02 07:52:18 +0000 | [diff] [blame] | 3196 | Types.push_back(FTP); | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 3197 | FunctionProtoTypes.InsertNode(FTP, InsertPos); | 
| Steve Naroff | e5aa9be | 2007-04-05 22:36:20 +0000 | [diff] [blame] | 3198 | return QualType(FTP, 0); | 
| Chris Lattner | c6ad813 | 2006-12-02 07:52:18 +0000 | [diff] [blame] | 3199 | } | 
| Chris Lattner | ef51c20 | 2006-11-10 07:17:23 +0000 | [diff] [blame] | 3200 |  | 
| Xiuli Pan | 9c14e28 | 2016-01-09 12:53:17 +0000 | [diff] [blame] | 3201 | /// Return pipe type for the specified type. | 
|  | 3202 | QualType ASTContext::getPipeType(QualType T) const { | 
|  | 3203 | llvm::FoldingSetNodeID ID; | 
|  | 3204 | PipeType::Profile(ID, T); | 
|  | 3205 |  | 
|  | 3206 | void *InsertPos = 0; | 
|  | 3207 | if (PipeType *PT = PipeTypes.FindNodeOrInsertPos(ID, InsertPos)) | 
|  | 3208 | return QualType(PT, 0); | 
|  | 3209 |  | 
|  | 3210 | // If the pipe element type isn't canonical, this won't be a canonical type | 
|  | 3211 | // either, so fill in the canonical type field. | 
|  | 3212 | QualType Canonical; | 
|  | 3213 | if (!T.isCanonical()) { | 
|  | 3214 | Canonical = getPipeType(getCanonicalType(T)); | 
|  | 3215 |  | 
|  | 3216 | // Get the new insert position for the node we care about. | 
|  | 3217 | PipeType *NewIP = PipeTypes.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 3218 | assert(!NewIP && "Shouldn't be in the map!"); | 
|  | 3219 | (void)NewIP; | 
|  | 3220 | } | 
|  | 3221 | PipeType *New = new (*this, TypeAlignment) PipeType(T, Canonical); | 
|  | 3222 | Types.push_back(New); | 
|  | 3223 | PipeTypes.InsertNode(New, InsertPos); | 
|  | 3224 | return QualType(New, 0); | 
|  | 3225 | } | 
|  | 3226 |  | 
| John McCall | e78aac4 | 2010-03-10 03:28:59 +0000 | [diff] [blame] | 3227 | #ifndef NDEBUG | 
|  | 3228 | static bool NeedsInjectedClassNameType(const RecordDecl *D) { | 
|  | 3229 | if (!isa<CXXRecordDecl>(D)) return false; | 
|  | 3230 | const CXXRecordDecl *RD = cast<CXXRecordDecl>(D); | 
|  | 3231 | if (isa<ClassTemplatePartialSpecializationDecl>(RD)) | 
|  | 3232 | return true; | 
|  | 3233 | if (RD->getDescribedClassTemplate() && | 
|  | 3234 | !isa<ClassTemplateSpecializationDecl>(RD)) | 
|  | 3235 | return true; | 
|  | 3236 | return false; | 
|  | 3237 | } | 
|  | 3238 | #endif | 
|  | 3239 |  | 
|  | 3240 | /// getInjectedClassNameType - Return the unique reference to the | 
|  | 3241 | /// injected class name type for the specified templated declaration. | 
|  | 3242 | QualType ASTContext::getInjectedClassNameType(CXXRecordDecl *Decl, | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 3243 | QualType TST) const { | 
| John McCall | e78aac4 | 2010-03-10 03:28:59 +0000 | [diff] [blame] | 3244 | assert(NeedsInjectedClassNameType(Decl)); | 
|  | 3245 | if (Decl->TypeForDecl) { | 
|  | 3246 | assert(isa<InjectedClassNameType>(Decl->TypeForDecl)); | 
| Douglas Gregor | ec9fd13 | 2012-01-14 16:38:05 +0000 | [diff] [blame] | 3247 | } else if (CXXRecordDecl *PrevDecl = Decl->getPreviousDecl()) { | 
| John McCall | e78aac4 | 2010-03-10 03:28:59 +0000 | [diff] [blame] | 3248 | assert(PrevDecl->TypeForDecl && "previous declaration has no type"); | 
|  | 3249 | Decl->TypeForDecl = PrevDecl->TypeForDecl; | 
|  | 3250 | assert(isa<InjectedClassNameType>(Decl->TypeForDecl)); | 
|  | 3251 | } else { | 
| John McCall | 424cec9 | 2011-01-19 06:33:43 +0000 | [diff] [blame] | 3252 | Type *newType = | 
| John McCall | 2408e32 | 2010-04-27 00:57:59 +0000 | [diff] [blame] | 3253 | new (*this, TypeAlignment) InjectedClassNameType(Decl, TST); | 
| John McCall | 424cec9 | 2011-01-19 06:33:43 +0000 | [diff] [blame] | 3254 | Decl->TypeForDecl = newType; | 
|  | 3255 | Types.push_back(newType); | 
| John McCall | e78aac4 | 2010-03-10 03:28:59 +0000 | [diff] [blame] | 3256 | } | 
|  | 3257 | return QualType(Decl->TypeForDecl, 0); | 
|  | 3258 | } | 
|  | 3259 |  | 
| Douglas Gregor | 83a586e | 2008-04-13 21:07:44 +0000 | [diff] [blame] | 3260 | /// getTypeDeclType - Return the unique reference to the type for the | 
|  | 3261 | /// specified type declaration. | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 3262 | QualType ASTContext::getTypeDeclTypeSlow(const TypeDecl *Decl) const { | 
| Argyrios Kyrtzidis | 89656d2 | 2008-10-16 16:50:47 +0000 | [diff] [blame] | 3263 | assert(Decl && "Passed null for Decl param"); | 
| John McCall | 96f0b5f | 2010-03-10 06:48:02 +0000 | [diff] [blame] | 3264 | assert(!Decl->TypeForDecl && "TypeForDecl present in slow case"); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3265 |  | 
| Richard Smith | dda56e4 | 2011-04-15 14:24:37 +0000 | [diff] [blame] | 3266 | if (const TypedefNameDecl *Typedef = dyn_cast<TypedefNameDecl>(Decl)) | 
| Douglas Gregor | 83a586e | 2008-04-13 21:07:44 +0000 | [diff] [blame] | 3267 | return getTypedefType(Typedef); | 
| John McCall | 96f0b5f | 2010-03-10 06:48:02 +0000 | [diff] [blame] | 3268 |  | 
| John McCall | 96f0b5f | 2010-03-10 06:48:02 +0000 | [diff] [blame] | 3269 | assert(!isa<TemplateTypeParmDecl>(Decl) && | 
|  | 3270 | "Template type parameter types are always available."); | 
|  | 3271 |  | 
| John McCall | 81e3850 | 2010-02-16 03:57:14 +0000 | [diff] [blame] | 3272 | if (const RecordDecl *Record = dyn_cast<RecordDecl>(Decl)) { | 
| Rafael Espindola | 3f9e444 | 2013-10-19 02:13:21 +0000 | [diff] [blame] | 3273 | assert(Record->isFirstDecl() && "struct/union has previous declaration"); | 
| John McCall | 96f0b5f | 2010-03-10 06:48:02 +0000 | [diff] [blame] | 3274 | assert(!NeedsInjectedClassNameType(Record)); | 
| Argyrios Kyrtzidis | b5fcdc2 | 2010-07-04 21:44:47 +0000 | [diff] [blame] | 3275 | return getRecordType(Record); | 
| John McCall | 81e3850 | 2010-02-16 03:57:14 +0000 | [diff] [blame] | 3276 | } else if (const EnumDecl *Enum = dyn_cast<EnumDecl>(Decl)) { | 
| Rafael Espindola | 3f9e444 | 2013-10-19 02:13:21 +0000 | [diff] [blame] | 3277 | assert(Enum->isFirstDecl() && "enum has previous declaration"); | 
| Argyrios Kyrtzidis | b5fcdc2 | 2010-07-04 21:44:47 +0000 | [diff] [blame] | 3278 | return getEnumType(Enum); | 
| John McCall | 81e3850 | 2010-02-16 03:57:14 +0000 | [diff] [blame] | 3279 | } else if (const UnresolvedUsingTypenameDecl *Using = | 
| John McCall | b96ec56 | 2009-12-04 22:46:56 +0000 | [diff] [blame] | 3280 | dyn_cast<UnresolvedUsingTypenameDecl>(Decl)) { | 
| John McCall | 424cec9 | 2011-01-19 06:33:43 +0000 | [diff] [blame] | 3281 | Type *newType = new (*this, TypeAlignment) UnresolvedUsingType(Using); | 
|  | 3282 | Decl->TypeForDecl = newType; | 
|  | 3283 | Types.push_back(newType); | 
| Mike Stump | e9c6ffc | 2009-07-31 02:02:20 +0000 | [diff] [blame] | 3284 | } else | 
| John McCall | 96f0b5f | 2010-03-10 06:48:02 +0000 | [diff] [blame] | 3285 | llvm_unreachable("TypeDecl without a type?"); | 
| Argyrios Kyrtzidis | faf0876 | 2008-08-07 20:55:28 +0000 | [diff] [blame] | 3286 |  | 
| Argyrios Kyrtzidis | faf0876 | 2008-08-07 20:55:28 +0000 | [diff] [blame] | 3287 | return QualType(Decl->TypeForDecl, 0); | 
| Douglas Gregor | 83a586e | 2008-04-13 21:07:44 +0000 | [diff] [blame] | 3288 | } | 
|  | 3289 |  | 
| Chris Lattner | 32d920b | 2007-01-26 02:01:53 +0000 | [diff] [blame] | 3290 | /// getTypedefType - Return the unique reference to the type for the | 
| Richard Smith | dda56e4 | 2011-04-15 14:24:37 +0000 | [diff] [blame] | 3291 | /// specified typedef name decl. | 
| Argyrios Kyrtzidis | 45a83f9 | 2010-07-02 11:55:11 +0000 | [diff] [blame] | 3292 | QualType | 
| Richard Smith | dda56e4 | 2011-04-15 14:24:37 +0000 | [diff] [blame] | 3293 | ASTContext::getTypedefType(const TypedefNameDecl *Decl, | 
|  | 3294 | QualType Canonical) const { | 
| Steve Naroff | e5aa9be | 2007-04-05 22:36:20 +0000 | [diff] [blame] | 3295 | if (Decl->TypeForDecl) return QualType(Decl->TypeForDecl, 0); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3296 |  | 
| Argyrios Kyrtzidis | 45a83f9 | 2010-07-02 11:55:11 +0000 | [diff] [blame] | 3297 | if (Canonical.isNull()) | 
|  | 3298 | Canonical = getCanonicalType(Decl->getUnderlyingType()); | 
| John McCall | 424cec9 | 2011-01-19 06:33:43 +0000 | [diff] [blame] | 3299 | TypedefType *newType = new(*this, TypeAlignment) | 
| John McCall | 90d1c2d | 2009-09-24 23:30:46 +0000 | [diff] [blame] | 3300 | TypedefType(Type::Typedef, Decl, Canonical); | 
| John McCall | 424cec9 | 2011-01-19 06:33:43 +0000 | [diff] [blame] | 3301 | Decl->TypeForDecl = newType; | 
|  | 3302 | Types.push_back(newType); | 
|  | 3303 | return QualType(newType, 0); | 
| Chris Lattner | d0342e5 | 2006-11-20 04:02:15 +0000 | [diff] [blame] | 3304 | } | 
|  | 3305 |  | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 3306 | QualType ASTContext::getRecordType(const RecordDecl *Decl) const { | 
| Argyrios Kyrtzidis | b5fcdc2 | 2010-07-04 21:44:47 +0000 | [diff] [blame] | 3307 | if (Decl->TypeForDecl) return QualType(Decl->TypeForDecl, 0); | 
|  | 3308 |  | 
| Douglas Gregor | ec9fd13 | 2012-01-14 16:38:05 +0000 | [diff] [blame] | 3309 | if (const RecordDecl *PrevDecl = Decl->getPreviousDecl()) | 
| Argyrios Kyrtzidis | b5fcdc2 | 2010-07-04 21:44:47 +0000 | [diff] [blame] | 3310 | if (PrevDecl->TypeForDecl) | 
|  | 3311 | return QualType(Decl->TypeForDecl = PrevDecl->TypeForDecl, 0); | 
|  | 3312 |  | 
| John McCall | 424cec9 | 2011-01-19 06:33:43 +0000 | [diff] [blame] | 3313 | RecordType *newType = new (*this, TypeAlignment) RecordType(Decl); | 
|  | 3314 | Decl->TypeForDecl = newType; | 
|  | 3315 | Types.push_back(newType); | 
|  | 3316 | return QualType(newType, 0); | 
| Argyrios Kyrtzidis | b5fcdc2 | 2010-07-04 21:44:47 +0000 | [diff] [blame] | 3317 | } | 
|  | 3318 |  | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 3319 | QualType ASTContext::getEnumType(const EnumDecl *Decl) const { | 
| Argyrios Kyrtzidis | b5fcdc2 | 2010-07-04 21:44:47 +0000 | [diff] [blame] | 3320 | if (Decl->TypeForDecl) return QualType(Decl->TypeForDecl, 0); | 
|  | 3321 |  | 
| Douglas Gregor | ec9fd13 | 2012-01-14 16:38:05 +0000 | [diff] [blame] | 3322 | if (const EnumDecl *PrevDecl = Decl->getPreviousDecl()) | 
| Argyrios Kyrtzidis | b5fcdc2 | 2010-07-04 21:44:47 +0000 | [diff] [blame] | 3323 | if (PrevDecl->TypeForDecl) | 
|  | 3324 | return QualType(Decl->TypeForDecl = PrevDecl->TypeForDecl, 0); | 
|  | 3325 |  | 
| John McCall | 424cec9 | 2011-01-19 06:33:43 +0000 | [diff] [blame] | 3326 | EnumType *newType = new (*this, TypeAlignment) EnumType(Decl); | 
|  | 3327 | Decl->TypeForDecl = newType; | 
|  | 3328 | Types.push_back(newType); | 
|  | 3329 | return QualType(newType, 0); | 
| Argyrios Kyrtzidis | b5fcdc2 | 2010-07-04 21:44:47 +0000 | [diff] [blame] | 3330 | } | 
|  | 3331 |  | 
| John McCall | 8190451 | 2011-01-06 01:58:22 +0000 | [diff] [blame] | 3332 | QualType ASTContext::getAttributedType(AttributedType::Kind attrKind, | 
|  | 3333 | QualType modifiedType, | 
|  | 3334 | QualType equivalentType) { | 
|  | 3335 | llvm::FoldingSetNodeID id; | 
|  | 3336 | AttributedType::Profile(id, attrKind, modifiedType, equivalentType); | 
|  | 3337 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 3338 | void *insertPos = nullptr; | 
| John McCall | 8190451 | 2011-01-06 01:58:22 +0000 | [diff] [blame] | 3339 | AttributedType *type = AttributedTypes.FindNodeOrInsertPos(id, insertPos); | 
|  | 3340 | if (type) return QualType(type, 0); | 
|  | 3341 |  | 
|  | 3342 | QualType canon = getCanonicalType(equivalentType); | 
|  | 3343 | type = new (*this, TypeAlignment) | 
|  | 3344 | AttributedType(canon, attrKind, modifiedType, equivalentType); | 
|  | 3345 |  | 
|  | 3346 | Types.push_back(type); | 
|  | 3347 | AttributedTypes.InsertNode(type, insertPos); | 
|  | 3348 |  | 
|  | 3349 | return QualType(type, 0); | 
|  | 3350 | } | 
|  | 3351 |  | 
| John McCall | cebee16 | 2009-10-18 09:09:24 +0000 | [diff] [blame] | 3352 | /// \brief Retrieve a substitution-result type. | 
|  | 3353 | QualType | 
|  | 3354 | ASTContext::getSubstTemplateTypeParmType(const TemplateTypeParmType *Parm, | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 3355 | QualType Replacement) const { | 
| John McCall | b692a09 | 2009-10-22 20:10:53 +0000 | [diff] [blame] | 3356 | assert(Replacement.isCanonical() | 
| John McCall | cebee16 | 2009-10-18 09:09:24 +0000 | [diff] [blame] | 3357 | && "replacement types must always be canonical"); | 
|  | 3358 |  | 
|  | 3359 | llvm::FoldingSetNodeID ID; | 
|  | 3360 | SubstTemplateTypeParmType::Profile(ID, Parm, Replacement); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 3361 | void *InsertPos = nullptr; | 
| John McCall | cebee16 | 2009-10-18 09:09:24 +0000 | [diff] [blame] | 3362 | SubstTemplateTypeParmType *SubstParm | 
|  | 3363 | = SubstTemplateTypeParmTypes.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 3364 |  | 
|  | 3365 | if (!SubstParm) { | 
|  | 3366 | SubstParm = new (*this, TypeAlignment) | 
|  | 3367 | SubstTemplateTypeParmType(Parm, Replacement); | 
|  | 3368 | Types.push_back(SubstParm); | 
|  | 3369 | SubstTemplateTypeParmTypes.InsertNode(SubstParm, InsertPos); | 
|  | 3370 | } | 
|  | 3371 |  | 
|  | 3372 | return QualType(SubstParm, 0); | 
|  | 3373 | } | 
|  | 3374 |  | 
| Douglas Gregor | ada4b79 | 2011-01-14 02:55:32 +0000 | [diff] [blame] | 3375 | /// \brief Retrieve a | 
|  | 3376 | QualType ASTContext::getSubstTemplateTypeParmPackType( | 
|  | 3377 | const TemplateTypeParmType *Parm, | 
|  | 3378 | const TemplateArgument &ArgPack) { | 
|  | 3379 | #ifndef NDEBUG | 
| Aaron Ballman | 2a89e85 | 2014-07-15 21:32:31 +0000 | [diff] [blame] | 3380 | for (const auto &P : ArgPack.pack_elements()) { | 
|  | 3381 | assert(P.getKind() == TemplateArgument::Type &&"Pack contains a non-type"); | 
|  | 3382 | assert(P.getAsType().isCanonical() && "Pack contains non-canonical type"); | 
| Douglas Gregor | ada4b79 | 2011-01-14 02:55:32 +0000 | [diff] [blame] | 3383 | } | 
|  | 3384 | #endif | 
|  | 3385 |  | 
|  | 3386 | llvm::FoldingSetNodeID ID; | 
|  | 3387 | SubstTemplateTypeParmPackType::Profile(ID, Parm, ArgPack); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 3388 | void *InsertPos = nullptr; | 
| Douglas Gregor | ada4b79 | 2011-01-14 02:55:32 +0000 | [diff] [blame] | 3389 | if (SubstTemplateTypeParmPackType *SubstParm | 
|  | 3390 | = SubstTemplateTypeParmPackTypes.FindNodeOrInsertPos(ID, InsertPos)) | 
|  | 3391 | return QualType(SubstParm, 0); | 
|  | 3392 |  | 
|  | 3393 | QualType Canon; | 
|  | 3394 | if (!Parm->isCanonicalUnqualified()) { | 
|  | 3395 | Canon = getCanonicalType(QualType(Parm, 0)); | 
|  | 3396 | Canon = getSubstTemplateTypeParmPackType(cast<TemplateTypeParmType>(Canon), | 
|  | 3397 | ArgPack); | 
|  | 3398 | SubstTemplateTypeParmPackTypes.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 3399 | } | 
|  | 3400 |  | 
|  | 3401 | SubstTemplateTypeParmPackType *SubstParm | 
|  | 3402 | = new (*this, TypeAlignment) SubstTemplateTypeParmPackType(Parm, Canon, | 
|  | 3403 | ArgPack); | 
|  | 3404 | Types.push_back(SubstParm); | 
|  | 3405 | SubstTemplateTypeParmTypes.InsertNode(SubstParm, InsertPos); | 
|  | 3406 | return QualType(SubstParm, 0); | 
|  | 3407 | } | 
|  | 3408 |  | 
| Douglas Gregor | eff93e0 | 2009-02-05 23:33:38 +0000 | [diff] [blame] | 3409 | /// \brief Retrieve the template type parameter type for a template | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3410 | /// parameter or parameter pack with the given depth, index, and (optionally) | 
| Anders Carlsson | 90036dc | 2009-06-16 00:30:48 +0000 | [diff] [blame] | 3411 | /// name. | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3412 | QualType ASTContext::getTemplateTypeParmType(unsigned Depth, unsigned Index, | 
| Anders Carlsson | 90036dc | 2009-06-16 00:30:48 +0000 | [diff] [blame] | 3413 | bool ParameterPack, | 
| Chandler Carruth | 0883632 | 2011-05-01 00:51:33 +0000 | [diff] [blame] | 3414 | TemplateTypeParmDecl *TTPDecl) const { | 
| Douglas Gregor | eff93e0 | 2009-02-05 23:33:38 +0000 | [diff] [blame] | 3415 | llvm::FoldingSetNodeID ID; | 
| Chandler Carruth | 0883632 | 2011-05-01 00:51:33 +0000 | [diff] [blame] | 3416 | TemplateTypeParmType::Profile(ID, Depth, Index, ParameterPack, TTPDecl); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 3417 | void *InsertPos = nullptr; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3418 | TemplateTypeParmType *TypeParm | 
| Douglas Gregor | eff93e0 | 2009-02-05 23:33:38 +0000 | [diff] [blame] | 3419 | = TemplateTypeParmTypes.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 3420 |  | 
|  | 3421 | if (TypeParm) | 
|  | 3422 | return QualType(TypeParm, 0); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3423 |  | 
| Chandler Carruth | 0883632 | 2011-05-01 00:51:33 +0000 | [diff] [blame] | 3424 | if (TTPDecl) { | 
| Anders Carlsson | 90036dc | 2009-06-16 00:30:48 +0000 | [diff] [blame] | 3425 | QualType Canon = getTemplateTypeParmType(Depth, Index, ParameterPack); | 
| Chandler Carruth | 0883632 | 2011-05-01 00:51:33 +0000 | [diff] [blame] | 3426 | TypeParm = new (*this, TypeAlignment) TemplateTypeParmType(TTPDecl, Canon); | 
| Douglas Gregor | c42075a | 2010-02-04 18:10:26 +0000 | [diff] [blame] | 3427 |  | 
|  | 3428 | TemplateTypeParmType *TypeCheck | 
|  | 3429 | = TemplateTypeParmTypes.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 3430 | assert(!TypeCheck && "Template type parameter canonical type broken"); | 
|  | 3431 | (void)TypeCheck; | 
| Anders Carlsson | 90036dc | 2009-06-16 00:30:48 +0000 | [diff] [blame] | 3432 | } else | 
| John McCall | 90d1c2d | 2009-09-24 23:30:46 +0000 | [diff] [blame] | 3433 | TypeParm = new (*this, TypeAlignment) | 
|  | 3434 | TemplateTypeParmType(Depth, Index, ParameterPack); | 
| Douglas Gregor | eff93e0 | 2009-02-05 23:33:38 +0000 | [diff] [blame] | 3435 |  | 
|  | 3436 | Types.push_back(TypeParm); | 
|  | 3437 | TemplateTypeParmTypes.InsertNode(TypeParm, InsertPos); | 
|  | 3438 |  | 
|  | 3439 | return QualType(TypeParm, 0); | 
|  | 3440 | } | 
|  | 3441 |  | 
| John McCall | e78aac4 | 2010-03-10 03:28:59 +0000 | [diff] [blame] | 3442 | TypeSourceInfo * | 
|  | 3443 | ASTContext::getTemplateSpecializationTypeInfo(TemplateName Name, | 
|  | 3444 | SourceLocation NameLoc, | 
|  | 3445 | const TemplateArgumentListInfo &Args, | 
| Richard Smith | 3f1b5d0 | 2011-05-05 21:57:07 +0000 | [diff] [blame] | 3446 | QualType Underlying) const { | 
| Douglas Gregor | e16af53 | 2011-02-28 18:50:33 +0000 | [diff] [blame] | 3447 | assert(!Name.getAsDependentTemplateName() && | 
|  | 3448 | "No dependent template names here!"); | 
| Richard Smith | 3f1b5d0 | 2011-05-05 21:57:07 +0000 | [diff] [blame] | 3449 | QualType TST = getTemplateSpecializationType(Name, Args, Underlying); | 
| John McCall | e78aac4 | 2010-03-10 03:28:59 +0000 | [diff] [blame] | 3450 |  | 
|  | 3451 | TypeSourceInfo *DI = CreateTypeSourceInfo(TST); | 
| David Blaikie | 6adc78e | 2013-02-18 22:06:02 +0000 | [diff] [blame] | 3452 | TemplateSpecializationTypeLoc TL = | 
|  | 3453 | DI->getTypeLoc().castAs<TemplateSpecializationTypeLoc>(); | 
| Abramo Bagnara | 48c05be | 2012-02-06 14:41:24 +0000 | [diff] [blame] | 3454 | TL.setTemplateKeywordLoc(SourceLocation()); | 
| John McCall | e78aac4 | 2010-03-10 03:28:59 +0000 | [diff] [blame] | 3455 | TL.setTemplateNameLoc(NameLoc); | 
|  | 3456 | TL.setLAngleLoc(Args.getLAngleLoc()); | 
|  | 3457 | TL.setRAngleLoc(Args.getRAngleLoc()); | 
|  | 3458 | for (unsigned i = 0, e = TL.getNumArgs(); i != e; ++i) | 
|  | 3459 | TL.setArgLocInfo(i, Args[i].getLocInfo()); | 
|  | 3460 | return DI; | 
|  | 3461 | } | 
|  | 3462 |  | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3463 | QualType | 
| Douglas Gregor | dc572a3 | 2009-03-30 22:58:21 +0000 | [diff] [blame] | 3464 | ASTContext::getTemplateSpecializationType(TemplateName Template, | 
| John McCall | 6b51f28 | 2009-11-23 01:53:49 +0000 | [diff] [blame] | 3465 | const TemplateArgumentListInfo &Args, | 
| Richard Smith | 3f1b5d0 | 2011-05-05 21:57:07 +0000 | [diff] [blame] | 3466 | QualType Underlying) const { | 
| Douglas Gregor | e16af53 | 2011-02-28 18:50:33 +0000 | [diff] [blame] | 3467 | assert(!Template.getAsDependentTemplateName() && | 
|  | 3468 | "No dependent template names here!"); | 
| John McCall | 6b51f28 | 2009-11-23 01:53:49 +0000 | [diff] [blame] | 3469 |  | 
| Chris Lattner | 0e62c1c | 2011-07-23 10:55:15 +0000 | [diff] [blame] | 3470 | SmallVector<TemplateArgument, 4> ArgVec; | 
| David Majnemer | 6fbeee3 | 2016-07-07 04:43:07 +0000 | [diff] [blame] | 3471 | ArgVec.reserve(Args.size()); | 
|  | 3472 | for (const TemplateArgumentLoc &Arg : Args.arguments()) | 
|  | 3473 | ArgVec.push_back(Arg.getArgument()); | 
| John McCall | 0ad1666 | 2009-10-29 08:12:44 +0000 | [diff] [blame] | 3474 |  | 
| David Majnemer | 6fbeee3 | 2016-07-07 04:43:07 +0000 | [diff] [blame] | 3475 | return getTemplateSpecializationType(Template, ArgVec, Underlying); | 
| John McCall | 0ad1666 | 2009-10-29 08:12:44 +0000 | [diff] [blame] | 3476 | } | 
|  | 3477 |  | 
| Douglas Gregor | 1f79ca8 | 2012-02-03 17:16:23 +0000 | [diff] [blame] | 3478 | #ifndef NDEBUG | 
| David Majnemer | 6fbeee3 | 2016-07-07 04:43:07 +0000 | [diff] [blame] | 3479 | static bool hasAnyPackExpansions(ArrayRef<TemplateArgument> Args) { | 
|  | 3480 | for (const TemplateArgument &Arg : Args) | 
|  | 3481 | if (Arg.isPackExpansion()) | 
| Douglas Gregor | 1f79ca8 | 2012-02-03 17:16:23 +0000 | [diff] [blame] | 3482 | return true; | 
|  | 3483 |  | 
|  | 3484 | return true; | 
|  | 3485 | } | 
|  | 3486 | #endif | 
|  | 3487 |  | 
| John McCall | 0ad1666 | 2009-10-29 08:12:44 +0000 | [diff] [blame] | 3488 | QualType | 
|  | 3489 | ASTContext::getTemplateSpecializationType(TemplateName Template, | 
| David Majnemer | 6fbeee3 | 2016-07-07 04:43:07 +0000 | [diff] [blame] | 3490 | ArrayRef<TemplateArgument> Args, | 
| Richard Smith | 3f1b5d0 | 2011-05-05 21:57:07 +0000 | [diff] [blame] | 3491 | QualType Underlying) const { | 
| Douglas Gregor | e16af53 | 2011-02-28 18:50:33 +0000 | [diff] [blame] | 3492 | assert(!Template.getAsDependentTemplateName() && | 
|  | 3493 | "No dependent template names here!"); | 
| Douglas Gregor | e29139c | 2011-03-03 17:04:51 +0000 | [diff] [blame] | 3494 | // Look through qualified template names. | 
|  | 3495 | if (QualifiedTemplateName *QTN = Template.getAsQualifiedTemplateName()) | 
|  | 3496 | Template = TemplateName(QTN->getTemplateDecl()); | 
| Douglas Gregor | e16af53 | 2011-02-28 18:50:33 +0000 | [diff] [blame] | 3497 |  | 
| Douglas Gregor | 1f79ca8 | 2012-02-03 17:16:23 +0000 | [diff] [blame] | 3498 | bool IsTypeAlias = | 
| Richard Smith | 3f1b5d0 | 2011-05-05 21:57:07 +0000 | [diff] [blame] | 3499 | Template.getAsTemplateDecl() && | 
|  | 3500 | isa<TypeAliasTemplateDecl>(Template.getAsTemplateDecl()); | 
| Richard Smith | 3f1b5d0 | 2011-05-05 21:57:07 +0000 | [diff] [blame] | 3501 | QualType CanonType; | 
|  | 3502 | if (!Underlying.isNull()) | 
|  | 3503 | CanonType = getCanonicalType(Underlying); | 
|  | 3504 | else { | 
| Douglas Gregor | 1f79ca8 | 2012-02-03 17:16:23 +0000 | [diff] [blame] | 3505 | // We can get here with an alias template when the specialization contains | 
|  | 3506 | // a pack expansion that does not match up with a parameter pack. | 
| David Majnemer | 6fbeee3 | 2016-07-07 04:43:07 +0000 | [diff] [blame] | 3507 | assert((!IsTypeAlias || hasAnyPackExpansions(Args)) && | 
| Douglas Gregor | 1f79ca8 | 2012-02-03 17:16:23 +0000 | [diff] [blame] | 3508 | "Caller must compute aliased type"); | 
|  | 3509 | IsTypeAlias = false; | 
| David Majnemer | 6fbeee3 | 2016-07-07 04:43:07 +0000 | [diff] [blame] | 3510 | CanonType = getCanonicalTemplateSpecializationType(Template, Args); | 
| Richard Smith | 3f1b5d0 | 2011-05-05 21:57:07 +0000 | [diff] [blame] | 3511 | } | 
| Douglas Gregor | d56a91e | 2009-02-26 22:19:44 +0000 | [diff] [blame] | 3512 |  | 
| Douglas Gregor | a8e02e7 | 2009-07-28 23:00:59 +0000 | [diff] [blame] | 3513 | // Allocate the (non-canonical) template specialization type, but don't | 
|  | 3514 | // try to unique it: these types typically have location information that | 
|  | 3515 | // we don't unique and don't want to lose. | 
| Richard Smith | 3f1b5d0 | 2011-05-05 21:57:07 +0000 | [diff] [blame] | 3516 | void *Mem = Allocate(sizeof(TemplateSpecializationType) + | 
| David Majnemer | 6fbeee3 | 2016-07-07 04:43:07 +0000 | [diff] [blame] | 3517 | sizeof(TemplateArgument) * Args.size() + | 
| Douglas Gregor | 1f79ca8 | 2012-02-03 17:16:23 +0000 | [diff] [blame] | 3518 | (IsTypeAlias? sizeof(QualType) : 0), | 
| John McCall | 90d1c2d | 2009-09-24 23:30:46 +0000 | [diff] [blame] | 3519 | TypeAlignment); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3520 | TemplateSpecializationType *Spec | 
| David Majnemer | 6fbeee3 | 2016-07-07 04:43:07 +0000 | [diff] [blame] | 3521 | = new (Mem) TemplateSpecializationType(Template, Args, CanonType, | 
| Douglas Gregor | 1f79ca8 | 2012-02-03 17:16:23 +0000 | [diff] [blame] | 3522 | IsTypeAlias ? Underlying : QualType()); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3523 |  | 
| Douglas Gregor | 8bf4205 | 2009-02-09 18:46:07 +0000 | [diff] [blame] | 3524 | Types.push_back(Spec); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3525 | return QualType(Spec, 0); | 
| Douglas Gregor | 8bf4205 | 2009-02-09 18:46:07 +0000 | [diff] [blame] | 3526 | } | 
|  | 3527 |  | 
| David Majnemer | 6fbeee3 | 2016-07-07 04:43:07 +0000 | [diff] [blame] | 3528 | QualType ASTContext::getCanonicalTemplateSpecializationType( | 
|  | 3529 | TemplateName Template, ArrayRef<TemplateArgument> Args) const { | 
| Douglas Gregor | e16af53 | 2011-02-28 18:50:33 +0000 | [diff] [blame] | 3530 | assert(!Template.getAsDependentTemplateName() && | 
|  | 3531 | "No dependent template names here!"); | 
| Richard Smith | 3f1b5d0 | 2011-05-05 21:57:07 +0000 | [diff] [blame] | 3532 |  | 
| Douglas Gregor | e29139c | 2011-03-03 17:04:51 +0000 | [diff] [blame] | 3533 | // Look through qualified template names. | 
|  | 3534 | if (QualifiedTemplateName *QTN = Template.getAsQualifiedTemplateName()) | 
|  | 3535 | Template = TemplateName(QTN->getTemplateDecl()); | 
| Douglas Gregor | e16af53 | 2011-02-28 18:50:33 +0000 | [diff] [blame] | 3536 |  | 
| Argyrios Kyrtzidis | 45a83f9 | 2010-07-02 11:55:11 +0000 | [diff] [blame] | 3537 | // Build the canonical template specialization type. | 
|  | 3538 | TemplateName CanonTemplate = getCanonicalTemplateName(Template); | 
| Chris Lattner | 0e62c1c | 2011-07-23 10:55:15 +0000 | [diff] [blame] | 3539 | SmallVector<TemplateArgument, 4> CanonArgs; | 
| David Majnemer | 6fbeee3 | 2016-07-07 04:43:07 +0000 | [diff] [blame] | 3540 | unsigned NumArgs = Args.size(); | 
| Argyrios Kyrtzidis | 45a83f9 | 2010-07-02 11:55:11 +0000 | [diff] [blame] | 3541 | CanonArgs.reserve(NumArgs); | 
| David Majnemer | 6fbeee3 | 2016-07-07 04:43:07 +0000 | [diff] [blame] | 3542 | for (const TemplateArgument &Arg : Args) | 
|  | 3543 | CanonArgs.push_back(getCanonicalTemplateArgument(Arg)); | 
| Argyrios Kyrtzidis | 45a83f9 | 2010-07-02 11:55:11 +0000 | [diff] [blame] | 3544 |  | 
|  | 3545 | // Determine whether this canonical template specialization type already | 
|  | 3546 | // exists. | 
|  | 3547 | llvm::FoldingSetNodeID ID; | 
|  | 3548 | TemplateSpecializationType::Profile(ID, CanonTemplate, | 
| David Majnemer | 6fbeee3 | 2016-07-07 04:43:07 +0000 | [diff] [blame] | 3549 | CanonArgs, *this); | 
| Argyrios Kyrtzidis | 45a83f9 | 2010-07-02 11:55:11 +0000 | [diff] [blame] | 3550 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 3551 | void *InsertPos = nullptr; | 
| Argyrios Kyrtzidis | 45a83f9 | 2010-07-02 11:55:11 +0000 | [diff] [blame] | 3552 | TemplateSpecializationType *Spec | 
|  | 3553 | = TemplateSpecializationTypes.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 3554 |  | 
|  | 3555 | if (!Spec) { | 
|  | 3556 | // Allocate a new canonical template specialization type. | 
|  | 3557 | void *Mem = Allocate((sizeof(TemplateSpecializationType) + | 
|  | 3558 | sizeof(TemplateArgument) * NumArgs), | 
|  | 3559 | TypeAlignment); | 
|  | 3560 | Spec = new (Mem) TemplateSpecializationType(CanonTemplate, | 
| David Majnemer | 6fbeee3 | 2016-07-07 04:43:07 +0000 | [diff] [blame] | 3561 | CanonArgs, | 
| Richard Smith | 3f1b5d0 | 2011-05-05 21:57:07 +0000 | [diff] [blame] | 3562 | QualType(), QualType()); | 
| Argyrios Kyrtzidis | 45a83f9 | 2010-07-02 11:55:11 +0000 | [diff] [blame] | 3563 | Types.push_back(Spec); | 
|  | 3564 | TemplateSpecializationTypes.InsertNode(Spec, InsertPos); | 
|  | 3565 | } | 
|  | 3566 |  | 
|  | 3567 | assert(Spec->isDependentType() && | 
|  | 3568 | "Non-dependent template-id type must have a canonical type"); | 
|  | 3569 | return QualType(Spec, 0); | 
|  | 3570 | } | 
|  | 3571 |  | 
|  | 3572 | QualType | 
| Abramo Bagnara | 6150c88 | 2010-05-11 21:36:43 +0000 | [diff] [blame] | 3573 | ASTContext::getElaboratedType(ElaboratedTypeKeyword Keyword, | 
|  | 3574 | NestedNameSpecifier *NNS, | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 3575 | QualType NamedType) const { | 
| Douglas Gregor | 5253768 | 2009-03-19 00:18:19 +0000 | [diff] [blame] | 3576 | llvm::FoldingSetNodeID ID; | 
| Abramo Bagnara | 6150c88 | 2010-05-11 21:36:43 +0000 | [diff] [blame] | 3577 | ElaboratedType::Profile(ID, Keyword, NNS, NamedType); | 
| Douglas Gregor | 5253768 | 2009-03-19 00:18:19 +0000 | [diff] [blame] | 3578 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 3579 | void *InsertPos = nullptr; | 
| Abramo Bagnara | 6150c88 | 2010-05-11 21:36:43 +0000 | [diff] [blame] | 3580 | ElaboratedType *T = ElaboratedTypes.FindNodeOrInsertPos(ID, InsertPos); | 
| Douglas Gregor | 5253768 | 2009-03-19 00:18:19 +0000 | [diff] [blame] | 3581 | if (T) | 
|  | 3582 | return QualType(T, 0); | 
|  | 3583 |  | 
| Douglas Gregor | c42075a | 2010-02-04 18:10:26 +0000 | [diff] [blame] | 3584 | QualType Canon = NamedType; | 
|  | 3585 | if (!Canon.isCanonical()) { | 
|  | 3586 | Canon = getCanonicalType(NamedType); | 
| Abramo Bagnara | 6150c88 | 2010-05-11 21:36:43 +0000 | [diff] [blame] | 3587 | ElaboratedType *CheckT = ElaboratedTypes.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 3588 | assert(!CheckT && "Elaborated canonical type broken"); | 
| Douglas Gregor | c42075a | 2010-02-04 18:10:26 +0000 | [diff] [blame] | 3589 | (void)CheckT; | 
|  | 3590 | } | 
|  | 3591 |  | 
| Benjamin Kramer | 8adeef9 | 2015-04-02 16:19:54 +0000 | [diff] [blame] | 3592 | T = new (*this, TypeAlignment) ElaboratedType(Keyword, NNS, NamedType, Canon); | 
| Douglas Gregor | 5253768 | 2009-03-19 00:18:19 +0000 | [diff] [blame] | 3593 | Types.push_back(T); | 
| Abramo Bagnara | 6150c88 | 2010-05-11 21:36:43 +0000 | [diff] [blame] | 3594 | ElaboratedTypes.InsertNode(T, InsertPos); | 
| Douglas Gregor | 5253768 | 2009-03-19 00:18:19 +0000 | [diff] [blame] | 3595 | return QualType(T, 0); | 
|  | 3596 | } | 
|  | 3597 |  | 
| Abramo Bagnara | 924a8f3 | 2010-12-10 16:29:40 +0000 | [diff] [blame] | 3598 | QualType | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 3599 | ASTContext::getParenType(QualType InnerType) const { | 
| Abramo Bagnara | 924a8f3 | 2010-12-10 16:29:40 +0000 | [diff] [blame] | 3600 | llvm::FoldingSetNodeID ID; | 
|  | 3601 | ParenType::Profile(ID, InnerType); | 
|  | 3602 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 3603 | void *InsertPos = nullptr; | 
| Abramo Bagnara | 924a8f3 | 2010-12-10 16:29:40 +0000 | [diff] [blame] | 3604 | ParenType *T = ParenTypes.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 3605 | if (T) | 
|  | 3606 | return QualType(T, 0); | 
|  | 3607 |  | 
|  | 3608 | QualType Canon = InnerType; | 
|  | 3609 | if (!Canon.isCanonical()) { | 
|  | 3610 | Canon = getCanonicalType(InnerType); | 
|  | 3611 | ParenType *CheckT = ParenTypes.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 3612 | assert(!CheckT && "Paren canonical type broken"); | 
|  | 3613 | (void)CheckT; | 
|  | 3614 | } | 
|  | 3615 |  | 
| Benjamin Kramer | 8adeef9 | 2015-04-02 16:19:54 +0000 | [diff] [blame] | 3616 | T = new (*this, TypeAlignment) ParenType(InnerType, Canon); | 
| Abramo Bagnara | 924a8f3 | 2010-12-10 16:29:40 +0000 | [diff] [blame] | 3617 | Types.push_back(T); | 
|  | 3618 | ParenTypes.InsertNode(T, InsertPos); | 
|  | 3619 | return QualType(T, 0); | 
|  | 3620 | } | 
|  | 3621 |  | 
| Douglas Gregor | 0208535 | 2010-03-31 20:19:30 +0000 | [diff] [blame] | 3622 | QualType ASTContext::getDependentNameType(ElaboratedTypeKeyword Keyword, | 
|  | 3623 | NestedNameSpecifier *NNS, | 
|  | 3624 | const IdentifierInfo *Name, | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 3625 | QualType Canon) const { | 
| Douglas Gregor | 333489b | 2009-03-27 23:10:48 +0000 | [diff] [blame] | 3626 | if (Canon.isNull()) { | 
|  | 3627 | NestedNameSpecifier *CanonNNS = getCanonicalNestedNameSpecifier(NNS); | 
| Douglas Gregor | 0208535 | 2010-03-31 20:19:30 +0000 | [diff] [blame] | 3628 | ElaboratedTypeKeyword CanonKeyword = Keyword; | 
|  | 3629 | if (Keyword == ETK_None) | 
|  | 3630 | CanonKeyword = ETK_Typename; | 
|  | 3631 |  | 
|  | 3632 | if (CanonNNS != NNS || CanonKeyword != Keyword) | 
|  | 3633 | Canon = getDependentNameType(CanonKeyword, CanonNNS, Name); | 
| Douglas Gregor | 333489b | 2009-03-27 23:10:48 +0000 | [diff] [blame] | 3634 | } | 
|  | 3635 |  | 
|  | 3636 | llvm::FoldingSetNodeID ID; | 
| Douglas Gregor | 0208535 | 2010-03-31 20:19:30 +0000 | [diff] [blame] | 3637 | DependentNameType::Profile(ID, Keyword, NNS, Name); | 
| Douglas Gregor | 333489b | 2009-03-27 23:10:48 +0000 | [diff] [blame] | 3638 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 3639 | void *InsertPos = nullptr; | 
| Douglas Gregor | c1d2d8a | 2010-03-31 17:34:00 +0000 | [diff] [blame] | 3640 | DependentNameType *T | 
|  | 3641 | = DependentNameTypes.FindNodeOrInsertPos(ID, InsertPos); | 
| Douglas Gregor | 333489b | 2009-03-27 23:10:48 +0000 | [diff] [blame] | 3642 | if (T) | 
|  | 3643 | return QualType(T, 0); | 
|  | 3644 |  | 
| Benjamin Kramer | 8adeef9 | 2015-04-02 16:19:54 +0000 | [diff] [blame] | 3645 | T = new (*this, TypeAlignment) DependentNameType(Keyword, NNS, Name, Canon); | 
| Douglas Gregor | 333489b | 2009-03-27 23:10:48 +0000 | [diff] [blame] | 3646 | Types.push_back(T); | 
| Douglas Gregor | c1d2d8a | 2010-03-31 17:34:00 +0000 | [diff] [blame] | 3647 | DependentNameTypes.InsertNode(T, InsertPos); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3648 | return QualType(T, 0); | 
| Douglas Gregor | 333489b | 2009-03-27 23:10:48 +0000 | [diff] [blame] | 3649 | } | 
|  | 3650 |  | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3651 | QualType | 
| John McCall | c392f37 | 2010-06-11 00:33:02 +0000 | [diff] [blame] | 3652 | ASTContext::getDependentTemplateSpecializationType( | 
|  | 3653 | ElaboratedTypeKeyword Keyword, | 
| Douglas Gregor | 0208535 | 2010-03-31 20:19:30 +0000 | [diff] [blame] | 3654 | NestedNameSpecifier *NNS, | 
| John McCall | c392f37 | 2010-06-11 00:33:02 +0000 | [diff] [blame] | 3655 | const IdentifierInfo *Name, | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 3656 | const TemplateArgumentListInfo &Args) const { | 
| John McCall | c392f37 | 2010-06-11 00:33:02 +0000 | [diff] [blame] | 3657 | // TODO: avoid this copy | 
| Chris Lattner | 0e62c1c | 2011-07-23 10:55:15 +0000 | [diff] [blame] | 3658 | SmallVector<TemplateArgument, 16> ArgCopy; | 
| John McCall | c392f37 | 2010-06-11 00:33:02 +0000 | [diff] [blame] | 3659 | for (unsigned I = 0, E = Args.size(); I != E; ++I) | 
|  | 3660 | ArgCopy.push_back(Args[I].getArgument()); | 
| David Majnemer | 6fbeee3 | 2016-07-07 04:43:07 +0000 | [diff] [blame] | 3661 | return getDependentTemplateSpecializationType(Keyword, NNS, Name, ArgCopy); | 
| John McCall | c392f37 | 2010-06-11 00:33:02 +0000 | [diff] [blame] | 3662 | } | 
|  | 3663 |  | 
|  | 3664 | QualType | 
|  | 3665 | ASTContext::getDependentTemplateSpecializationType( | 
|  | 3666 | ElaboratedTypeKeyword Keyword, | 
|  | 3667 | NestedNameSpecifier *NNS, | 
|  | 3668 | const IdentifierInfo *Name, | 
| David Majnemer | 6fbeee3 | 2016-07-07 04:43:07 +0000 | [diff] [blame] | 3669 | ArrayRef<TemplateArgument> Args) const { | 
| Douglas Gregor | 6e06801 | 2011-02-28 00:04:36 +0000 | [diff] [blame] | 3670 | assert((!NNS || NNS->isDependent()) && | 
|  | 3671 | "nested-name-specifier must be dependent"); | 
| Douglas Gregor | dce2b62 | 2009-04-01 00:28:59 +0000 | [diff] [blame] | 3672 |  | 
| Douglas Gregor | c42075a | 2010-02-04 18:10:26 +0000 | [diff] [blame] | 3673 | llvm::FoldingSetNodeID ID; | 
| John McCall | c392f37 | 2010-06-11 00:33:02 +0000 | [diff] [blame] | 3674 | DependentTemplateSpecializationType::Profile(ID, *this, Keyword, NNS, | 
| David Majnemer | 6fbeee3 | 2016-07-07 04:43:07 +0000 | [diff] [blame] | 3675 | Name, Args); | 
| Douglas Gregor | c42075a | 2010-02-04 18:10:26 +0000 | [diff] [blame] | 3676 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 3677 | void *InsertPos = nullptr; | 
| John McCall | c392f37 | 2010-06-11 00:33:02 +0000 | [diff] [blame] | 3678 | DependentTemplateSpecializationType *T | 
|  | 3679 | = DependentTemplateSpecializationTypes.FindNodeOrInsertPos(ID, InsertPos); | 
| Douglas Gregor | c42075a | 2010-02-04 18:10:26 +0000 | [diff] [blame] | 3680 | if (T) | 
|  | 3681 | return QualType(T, 0); | 
|  | 3682 |  | 
| John McCall | c392f37 | 2010-06-11 00:33:02 +0000 | [diff] [blame] | 3683 | NestedNameSpecifier *CanonNNS = getCanonicalNestedNameSpecifier(NNS); | 
| Douglas Gregor | c42075a | 2010-02-04 18:10:26 +0000 | [diff] [blame] | 3684 |  | 
| John McCall | c392f37 | 2010-06-11 00:33:02 +0000 | [diff] [blame] | 3685 | ElaboratedTypeKeyword CanonKeyword = Keyword; | 
|  | 3686 | if (Keyword == ETK_None) CanonKeyword = ETK_Typename; | 
|  | 3687 |  | 
|  | 3688 | bool AnyNonCanonArgs = false; | 
| David Majnemer | 6fbeee3 | 2016-07-07 04:43:07 +0000 | [diff] [blame] | 3689 | unsigned NumArgs = Args.size(); | 
| Chris Lattner | 0e62c1c | 2011-07-23 10:55:15 +0000 | [diff] [blame] | 3690 | SmallVector<TemplateArgument, 16> CanonArgs(NumArgs); | 
| John McCall | c392f37 | 2010-06-11 00:33:02 +0000 | [diff] [blame] | 3691 | for (unsigned I = 0; I != NumArgs; ++I) { | 
|  | 3692 | CanonArgs[I] = getCanonicalTemplateArgument(Args[I]); | 
|  | 3693 | if (!CanonArgs[I].structurallyEquals(Args[I])) | 
|  | 3694 | AnyNonCanonArgs = true; | 
| Douglas Gregor | dce2b62 | 2009-04-01 00:28:59 +0000 | [diff] [blame] | 3695 | } | 
|  | 3696 |  | 
| John McCall | c392f37 | 2010-06-11 00:33:02 +0000 | [diff] [blame] | 3697 | QualType Canon; | 
|  | 3698 | if (AnyNonCanonArgs || CanonNNS != NNS || CanonKeyword != Keyword) { | 
|  | 3699 | Canon = getDependentTemplateSpecializationType(CanonKeyword, CanonNNS, | 
| David Majnemer | 6fbeee3 | 2016-07-07 04:43:07 +0000 | [diff] [blame] | 3700 | Name, | 
|  | 3701 | CanonArgs); | 
| John McCall | c392f37 | 2010-06-11 00:33:02 +0000 | [diff] [blame] | 3702 |  | 
|  | 3703 | // Find the insert position again. | 
|  | 3704 | DependentTemplateSpecializationTypes.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 3705 | } | 
|  | 3706 |  | 
|  | 3707 | void *Mem = Allocate((sizeof(DependentTemplateSpecializationType) + | 
|  | 3708 | sizeof(TemplateArgument) * NumArgs), | 
|  | 3709 | TypeAlignment); | 
| John McCall | 773cc98 | 2010-06-11 11:07:21 +0000 | [diff] [blame] | 3710 | T = new (Mem) DependentTemplateSpecializationType(Keyword, NNS, | 
| David Majnemer | 6fbeee3 | 2016-07-07 04:43:07 +0000 | [diff] [blame] | 3711 | Name, Args, Canon); | 
| Douglas Gregor | dce2b62 | 2009-04-01 00:28:59 +0000 | [diff] [blame] | 3712 | Types.push_back(T); | 
| John McCall | c392f37 | 2010-06-11 00:33:02 +0000 | [diff] [blame] | 3713 | DependentTemplateSpecializationTypes.InsertNode(T, InsertPos); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3714 | return QualType(T, 0); | 
| Douglas Gregor | dce2b62 | 2009-04-01 00:28:59 +0000 | [diff] [blame] | 3715 | } | 
|  | 3716 |  | 
| Douglas Gregor | 0dca5fd | 2011-01-14 17:04:44 +0000 | [diff] [blame] | 3717 | QualType ASTContext::getPackExpansionType(QualType Pattern, | 
| David Blaikie | 05785d1 | 2013-02-20 22:23:23 +0000 | [diff] [blame] | 3718 | Optional<unsigned> NumExpansions) { | 
| Douglas Gregor | d2fa766 | 2010-12-20 02:24:11 +0000 | [diff] [blame] | 3719 | llvm::FoldingSetNodeID ID; | 
| Douglas Gregor | 0dca5fd | 2011-01-14 17:04:44 +0000 | [diff] [blame] | 3720 | PackExpansionType::Profile(ID, Pattern, NumExpansions); | 
| Douglas Gregor | d2fa766 | 2010-12-20 02:24:11 +0000 | [diff] [blame] | 3721 |  | 
|  | 3722 | assert(Pattern->containsUnexpandedParameterPack() && | 
|  | 3723 | "Pack expansions must expand one or more parameter packs"); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 3724 | void *InsertPos = nullptr; | 
| Douglas Gregor | d2fa766 | 2010-12-20 02:24:11 +0000 | [diff] [blame] | 3725 | PackExpansionType *T | 
|  | 3726 | = PackExpansionTypes.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 3727 | if (T) | 
|  | 3728 | return QualType(T, 0); | 
|  | 3729 |  | 
|  | 3730 | QualType Canon; | 
|  | 3731 | if (!Pattern.isCanonical()) { | 
| Richard Smith | 68eea50 | 2012-07-16 00:20:35 +0000 | [diff] [blame] | 3732 | Canon = getCanonicalType(Pattern); | 
|  | 3733 | // The canonical type might not contain an unexpanded parameter pack, if it | 
|  | 3734 | // contains an alias template specialization which ignores one of its | 
|  | 3735 | // parameters. | 
|  | 3736 | if (Canon->containsUnexpandedParameterPack()) { | 
| Richard Smith | 8b4e1e2 | 2014-07-10 01:20:17 +0000 | [diff] [blame] | 3737 | Canon = getPackExpansionType(Canon, NumExpansions); | 
| Douglas Gregor | d2fa766 | 2010-12-20 02:24:11 +0000 | [diff] [blame] | 3738 |  | 
| Richard Smith | 68eea50 | 2012-07-16 00:20:35 +0000 | [diff] [blame] | 3739 | // Find the insert position again, in case we inserted an element into | 
|  | 3740 | // PackExpansionTypes and invalidated our insert position. | 
|  | 3741 | PackExpansionTypes.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 3742 | } | 
| Douglas Gregor | d2fa766 | 2010-12-20 02:24:11 +0000 | [diff] [blame] | 3743 | } | 
|  | 3744 |  | 
| Benjamin Kramer | 8adeef9 | 2015-04-02 16:19:54 +0000 | [diff] [blame] | 3745 | T = new (*this, TypeAlignment) | 
|  | 3746 | PackExpansionType(Pattern, Canon, NumExpansions); | 
| Douglas Gregor | d2fa766 | 2010-12-20 02:24:11 +0000 | [diff] [blame] | 3747 | Types.push_back(T); | 
|  | 3748 | PackExpansionTypes.InsertNode(T, InsertPos); | 
| Richard Smith | 8b4e1e2 | 2014-07-10 01:20:17 +0000 | [diff] [blame] | 3749 | return QualType(T, 0); | 
| Douglas Gregor | d2fa766 | 2010-12-20 02:24:11 +0000 | [diff] [blame] | 3750 | } | 
|  | 3751 |  | 
| Chris Lattner | e0ea37a | 2008-04-07 04:56:42 +0000 | [diff] [blame] | 3752 | /// CmpProtocolNames - Comparison predicate for sorting protocols | 
|  | 3753 | /// alphabetically. | 
| Benjamin Kramer | 0eb262f | 2015-03-14 13:32:49 +0000 | [diff] [blame] | 3754 | static int CmpProtocolNames(ObjCProtocolDecl *const *LHS, | 
|  | 3755 | ObjCProtocolDecl *const *RHS) { | 
|  | 3756 | return DeclarationName::compare((*LHS)->getDeclName(), (*RHS)->getDeclName()); | 
| Chris Lattner | e0ea37a | 2008-04-07 04:56:42 +0000 | [diff] [blame] | 3757 | } | 
|  | 3758 |  | 
| Craig Topper | 1f26d5b | 2016-01-03 19:43:23 +0000 | [diff] [blame] | 3759 | static bool areSortedAndUniqued(ArrayRef<ObjCProtocolDecl *> Protocols) { | 
|  | 3760 | if (Protocols.empty()) return true; | 
| John McCall | fc93cf9 | 2009-10-22 22:37:11 +0000 | [diff] [blame] | 3761 |  | 
| Douglas Gregor | cf9f3ea | 2012-01-02 02:00:30 +0000 | [diff] [blame] | 3762 | if (Protocols[0]->getCanonicalDecl() != Protocols[0]) | 
|  | 3763 | return false; | 
|  | 3764 |  | 
| Craig Topper | 1f26d5b | 2016-01-03 19:43:23 +0000 | [diff] [blame] | 3765 | for (unsigned i = 1; i != Protocols.size(); ++i) | 
| Benjamin Kramer | 0eb262f | 2015-03-14 13:32:49 +0000 | [diff] [blame] | 3766 | if (CmpProtocolNames(&Protocols[i - 1], &Protocols[i]) >= 0 || | 
| Douglas Gregor | cf9f3ea | 2012-01-02 02:00:30 +0000 | [diff] [blame] | 3767 | Protocols[i]->getCanonicalDecl() != Protocols[i]) | 
| John McCall | fc93cf9 | 2009-10-22 22:37:11 +0000 | [diff] [blame] | 3768 | return false; | 
|  | 3769 | return true; | 
|  | 3770 | } | 
|  | 3771 |  | 
| Craig Topper | 6a8c151 | 2015-10-22 01:56:18 +0000 | [diff] [blame] | 3772 | static void | 
|  | 3773 | SortAndUniqueProtocols(SmallVectorImpl<ObjCProtocolDecl *> &Protocols) { | 
| Chris Lattner | e0ea37a | 2008-04-07 04:56:42 +0000 | [diff] [blame] | 3774 | // Sort protocols, keyed by name. | 
| Craig Topper | 6a8c151 | 2015-10-22 01:56:18 +0000 | [diff] [blame] | 3775 | llvm::array_pod_sort(Protocols.begin(), Protocols.end(), CmpProtocolNames); | 
| Chris Lattner | e0ea37a | 2008-04-07 04:56:42 +0000 | [diff] [blame] | 3776 |  | 
| Douglas Gregor | cf9f3ea | 2012-01-02 02:00:30 +0000 | [diff] [blame] | 3777 | // Canonicalize. | 
| Craig Topper | 6a8c151 | 2015-10-22 01:56:18 +0000 | [diff] [blame] | 3778 | for (ObjCProtocolDecl *&P : Protocols) | 
|  | 3779 | P = P->getCanonicalDecl(); | 
|  | 3780 |  | 
| Chris Lattner | e0ea37a | 2008-04-07 04:56:42 +0000 | [diff] [blame] | 3781 | // Remove duplicates. | 
| Craig Topper | 6a8c151 | 2015-10-22 01:56:18 +0000 | [diff] [blame] | 3782 | auto ProtocolsEnd = std::unique(Protocols.begin(), Protocols.end()); | 
|  | 3783 | Protocols.erase(ProtocolsEnd, Protocols.end()); | 
| Chris Lattner | e0ea37a | 2008-04-07 04:56:42 +0000 | [diff] [blame] | 3784 | } | 
|  | 3785 |  | 
| John McCall | 8b07ec2 | 2010-05-15 11:32:37 +0000 | [diff] [blame] | 3786 | QualType ASTContext::getObjCObjectType(QualType BaseType, | 
|  | 3787 | ObjCProtocolDecl * const *Protocols, | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 3788 | unsigned NumProtocols) const { | 
| Douglas Gregor | e9d95f1 | 2015-07-07 03:57:35 +0000 | [diff] [blame] | 3789 | return getObjCObjectType(BaseType, { }, | 
| Douglas Gregor | ab209d8 | 2015-07-07 03:58:42 +0000 | [diff] [blame] | 3790 | llvm::makeArrayRef(Protocols, NumProtocols), | 
|  | 3791 | /*isKindOf=*/false); | 
| Douglas Gregor | e9d95f1 | 2015-07-07 03:57:35 +0000 | [diff] [blame] | 3792 | } | 
|  | 3793 |  | 
|  | 3794 | QualType ASTContext::getObjCObjectType( | 
|  | 3795 | QualType baseType, | 
|  | 3796 | ArrayRef<QualType> typeArgs, | 
| Douglas Gregor | ab209d8 | 2015-07-07 03:58:42 +0000 | [diff] [blame] | 3797 | ArrayRef<ObjCProtocolDecl *> protocols, | 
|  | 3798 | bool isKindOf) const { | 
| Douglas Gregor | e9d95f1 | 2015-07-07 03:57:35 +0000 | [diff] [blame] | 3799 | // If the base type is an interface and there aren't any protocols or | 
|  | 3800 | // type arguments to add, then the interface type will do just fine. | 
| Douglas Gregor | ab209d8 | 2015-07-07 03:58:42 +0000 | [diff] [blame] | 3801 | if (typeArgs.empty() && protocols.empty() && !isKindOf && | 
|  | 3802 | isa<ObjCInterfaceType>(baseType)) | 
| Douglas Gregor | e9d95f1 | 2015-07-07 03:57:35 +0000 | [diff] [blame] | 3803 | return baseType; | 
| John McCall | 8b07ec2 | 2010-05-15 11:32:37 +0000 | [diff] [blame] | 3804 |  | 
|  | 3805 | // Look in the folding set for an existing type. | 
| Steve Naroff | fb4330f | 2009-06-17 22:40:22 +0000 | [diff] [blame] | 3806 | llvm::FoldingSetNodeID ID; | 
| Douglas Gregor | ab209d8 | 2015-07-07 03:58:42 +0000 | [diff] [blame] | 3807 | ObjCObjectTypeImpl::Profile(ID, baseType, typeArgs, protocols, isKindOf); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 3808 | void *InsertPos = nullptr; | 
| John McCall | 8b07ec2 | 2010-05-15 11:32:37 +0000 | [diff] [blame] | 3809 | if (ObjCObjectType *QT = ObjCObjectTypes.FindNodeOrInsertPos(ID, InsertPos)) | 
|  | 3810 | return QualType(QT, 0); | 
| Steve Naroff | fb4330f | 2009-06-17 22:40:22 +0000 | [diff] [blame] | 3811 |  | 
| Douglas Gregor | e9d95f1 | 2015-07-07 03:57:35 +0000 | [diff] [blame] | 3812 | // Determine the type arguments to be used for canonicalization, | 
|  | 3813 | // which may be explicitly specified here or written on the base | 
|  | 3814 | // type. | 
|  | 3815 | ArrayRef<QualType> effectiveTypeArgs = typeArgs; | 
|  | 3816 | if (effectiveTypeArgs.empty()) { | 
|  | 3817 | if (auto baseObject = baseType->getAs<ObjCObjectType>()) | 
|  | 3818 | effectiveTypeArgs = baseObject->getTypeArgs(); | 
|  | 3819 | } | 
| John McCall | fc93cf9 | 2009-10-22 22:37:11 +0000 | [diff] [blame] | 3820 |  | 
| Douglas Gregor | e9d95f1 | 2015-07-07 03:57:35 +0000 | [diff] [blame] | 3821 | // Build the canonical type, which has the canonical base type and a | 
|  | 3822 | // sorted-and-uniqued list of protocols and the type arguments | 
|  | 3823 | // canonicalized. | 
|  | 3824 | QualType canonical; | 
|  | 3825 | bool typeArgsAreCanonical = std::all_of(effectiveTypeArgs.begin(), | 
|  | 3826 | effectiveTypeArgs.end(), | 
|  | 3827 | [&](QualType type) { | 
|  | 3828 | return type.isCanonical(); | 
|  | 3829 | }); | 
| Craig Topper | 1f26d5b | 2016-01-03 19:43:23 +0000 | [diff] [blame] | 3830 | bool protocolsSorted = areSortedAndUniqued(protocols); | 
| Douglas Gregor | e9d95f1 | 2015-07-07 03:57:35 +0000 | [diff] [blame] | 3831 | if (!typeArgsAreCanonical || !protocolsSorted || !baseType.isCanonical()) { | 
|  | 3832 | // Determine the canonical type arguments. | 
|  | 3833 | ArrayRef<QualType> canonTypeArgs; | 
|  | 3834 | SmallVector<QualType, 4> canonTypeArgsVec; | 
|  | 3835 | if (!typeArgsAreCanonical) { | 
|  | 3836 | canonTypeArgsVec.reserve(effectiveTypeArgs.size()); | 
|  | 3837 | for (auto typeArg : effectiveTypeArgs) | 
|  | 3838 | canonTypeArgsVec.push_back(getCanonicalType(typeArg)); | 
|  | 3839 | canonTypeArgs = canonTypeArgsVec; | 
| John McCall | fc93cf9 | 2009-10-22 22:37:11 +0000 | [diff] [blame] | 3840 | } else { | 
| Douglas Gregor | e9d95f1 | 2015-07-07 03:57:35 +0000 | [diff] [blame] | 3841 | canonTypeArgs = effectiveTypeArgs; | 
| John McCall | fc93cf9 | 2009-10-22 22:37:11 +0000 | [diff] [blame] | 3842 | } | 
|  | 3843 |  | 
| Douglas Gregor | e9d95f1 | 2015-07-07 03:57:35 +0000 | [diff] [blame] | 3844 | ArrayRef<ObjCProtocolDecl *> canonProtocols; | 
|  | 3845 | SmallVector<ObjCProtocolDecl*, 8> canonProtocolsVec; | 
|  | 3846 | if (!protocolsSorted) { | 
| Craig Topper | 6a8c151 | 2015-10-22 01:56:18 +0000 | [diff] [blame] | 3847 | canonProtocolsVec.append(protocols.begin(), protocols.end()); | 
|  | 3848 | SortAndUniqueProtocols(canonProtocolsVec); | 
|  | 3849 | canonProtocols = canonProtocolsVec; | 
| Douglas Gregor | e9d95f1 | 2015-07-07 03:57:35 +0000 | [diff] [blame] | 3850 | } else { | 
|  | 3851 | canonProtocols = protocols; | 
|  | 3852 | } | 
|  | 3853 |  | 
|  | 3854 | canonical = getObjCObjectType(getCanonicalType(baseType), canonTypeArgs, | 
| Douglas Gregor | ab209d8 | 2015-07-07 03:58:42 +0000 | [diff] [blame] | 3855 | canonProtocols, isKindOf); | 
| Douglas Gregor | e9d95f1 | 2015-07-07 03:57:35 +0000 | [diff] [blame] | 3856 |  | 
| John McCall | fc93cf9 | 2009-10-22 22:37:11 +0000 | [diff] [blame] | 3857 | // Regenerate InsertPos. | 
| John McCall | 8b07ec2 | 2010-05-15 11:32:37 +0000 | [diff] [blame] | 3858 | ObjCObjectTypes.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 3859 | } | 
|  | 3860 |  | 
| Douglas Gregor | e9d95f1 | 2015-07-07 03:57:35 +0000 | [diff] [blame] | 3861 | unsigned size = sizeof(ObjCObjectTypeImpl); | 
|  | 3862 | size += typeArgs.size() * sizeof(QualType); | 
|  | 3863 | size += protocols.size() * sizeof(ObjCProtocolDecl *); | 
|  | 3864 | void *mem = Allocate(size, TypeAlignment); | 
| John McCall | 8b07ec2 | 2010-05-15 11:32:37 +0000 | [diff] [blame] | 3865 | ObjCObjectTypeImpl *T = | 
| Douglas Gregor | ab209d8 | 2015-07-07 03:58:42 +0000 | [diff] [blame] | 3866 | new (mem) ObjCObjectTypeImpl(canonical, baseType, typeArgs, protocols, | 
|  | 3867 | isKindOf); | 
| John McCall | 8b07ec2 | 2010-05-15 11:32:37 +0000 | [diff] [blame] | 3868 |  | 
|  | 3869 | Types.push_back(T); | 
|  | 3870 | ObjCObjectTypes.InsertNode(T, InsertPos); | 
|  | 3871 | return QualType(T, 0); | 
|  | 3872 | } | 
|  | 3873 |  | 
| Fariborz Jahanian | 92ab298 | 2013-11-20 00:32:12 +0000 | [diff] [blame] | 3874 | /// ObjCObjectAdoptsQTypeProtocols - Checks that protocols in IC's | 
|  | 3875 | /// protocol list adopt all protocols in QT's qualified-id protocol | 
|  | 3876 | /// list. | 
|  | 3877 | bool ASTContext::ObjCObjectAdoptsQTypeProtocols(QualType QT, | 
|  | 3878 | ObjCInterfaceDecl *IC) { | 
|  | 3879 | if (!QT->isObjCQualifiedIdType()) | 
|  | 3880 | return false; | 
|  | 3881 |  | 
|  | 3882 | if (const ObjCObjectPointerType *OPT = QT->getAs<ObjCObjectPointerType>()) { | 
|  | 3883 | // If both the right and left sides have qualifiers. | 
| Aaron Ballman | 8373146 | 2014-03-17 16:14:00 +0000 | [diff] [blame] | 3884 | for (auto *Proto : OPT->quals()) { | 
| Fariborz Jahanian | 92ab298 | 2013-11-20 00:32:12 +0000 | [diff] [blame] | 3885 | if (!IC->ClassImplementsProtocol(Proto, false)) | 
|  | 3886 | return false; | 
|  | 3887 | } | 
|  | 3888 | return true; | 
|  | 3889 | } | 
|  | 3890 | return false; | 
|  | 3891 | } | 
|  | 3892 |  | 
|  | 3893 | /// QIdProtocolsAdoptObjCObjectProtocols - Checks that protocols in | 
|  | 3894 | /// QT's qualified-id protocol list adopt all protocols in IDecl's list | 
|  | 3895 | /// of protocols. | 
|  | 3896 | bool ASTContext::QIdProtocolsAdoptObjCObjectProtocols(QualType QT, | 
|  | 3897 | ObjCInterfaceDecl *IDecl) { | 
|  | 3898 | if (!QT->isObjCQualifiedIdType()) | 
|  | 3899 | return false; | 
|  | 3900 | const ObjCObjectPointerType *OPT = QT->getAs<ObjCObjectPointerType>(); | 
|  | 3901 | if (!OPT) | 
|  | 3902 | return false; | 
|  | 3903 | if (!IDecl->hasDefinition()) | 
|  | 3904 | return false; | 
| Fariborz Jahanian | 91fb0be | 2013-11-20 19:01:50 +0000 | [diff] [blame] | 3905 | llvm::SmallPtrSet<ObjCProtocolDecl *, 8> InheritedProtocols; | 
|  | 3906 | CollectInheritedProtocols(IDecl, InheritedProtocols); | 
|  | 3907 | if (InheritedProtocols.empty()) | 
|  | 3908 | return false; | 
| Fariborz Jahanian | 48a01cb | 2014-04-04 23:53:45 +0000 | [diff] [blame] | 3909 | // Check that if every protocol in list of id<plist> conforms to a protcol | 
|  | 3910 | // of IDecl's, then bridge casting is ok. | 
|  | 3911 | bool Conforms = false; | 
|  | 3912 | for (auto *Proto : OPT->quals()) { | 
|  | 3913 | Conforms = false; | 
|  | 3914 | for (auto *PI : InheritedProtocols) { | 
|  | 3915 | if (ProtocolCompatibleWithProtocol(Proto, PI)) { | 
|  | 3916 | Conforms = true; | 
|  | 3917 | break; | 
|  | 3918 | } | 
|  | 3919 | } | 
|  | 3920 | if (!Conforms) | 
|  | 3921 | break; | 
|  | 3922 | } | 
|  | 3923 | if (Conforms) | 
|  | 3924 | return true; | 
|  | 3925 |  | 
| Aaron Ballman | 8373146 | 2014-03-17 16:14:00 +0000 | [diff] [blame] | 3926 | for (auto *PI : InheritedProtocols) { | 
| Fariborz Jahanian | 92ab298 | 2013-11-20 00:32:12 +0000 | [diff] [blame] | 3927 | // If both the right and left sides have qualifiers. | 
|  | 3928 | bool Adopts = false; | 
| Aaron Ballman | 8373146 | 2014-03-17 16:14:00 +0000 | [diff] [blame] | 3929 | for (auto *Proto : OPT->quals()) { | 
| Fariborz Jahanian | 48a01cb | 2014-04-04 23:53:45 +0000 | [diff] [blame] | 3930 | // return 'true' if 'PI' is in the inheritance hierarchy of Proto | 
| Aaron Ballman | 8373146 | 2014-03-17 16:14:00 +0000 | [diff] [blame] | 3931 | if ((Adopts = ProtocolCompatibleWithProtocol(PI, Proto))) | 
| Fariborz Jahanian | 92ab298 | 2013-11-20 00:32:12 +0000 | [diff] [blame] | 3932 | break; | 
|  | 3933 | } | 
|  | 3934 | if (!Adopts) | 
| Fariborz Jahanian | 91fb0be | 2013-11-20 19:01:50 +0000 | [diff] [blame] | 3935 | return false; | 
| Fariborz Jahanian | 92ab298 | 2013-11-20 00:32:12 +0000 | [diff] [blame] | 3936 | } | 
|  | 3937 | return true; | 
|  | 3938 | } | 
|  | 3939 |  | 
| John McCall | 8b07ec2 | 2010-05-15 11:32:37 +0000 | [diff] [blame] | 3940 | /// getObjCObjectPointerType - Return a ObjCObjectPointerType type for | 
|  | 3941 | /// the given object type. | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 3942 | QualType ASTContext::getObjCObjectPointerType(QualType ObjectT) const { | 
| John McCall | 8b07ec2 | 2010-05-15 11:32:37 +0000 | [diff] [blame] | 3943 | llvm::FoldingSetNodeID ID; | 
|  | 3944 | ObjCObjectPointerType::Profile(ID, ObjectT); | 
|  | 3945 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 3946 | void *InsertPos = nullptr; | 
| John McCall | 8b07ec2 | 2010-05-15 11:32:37 +0000 | [diff] [blame] | 3947 | if (ObjCObjectPointerType *QT = | 
|  | 3948 | ObjCObjectPointerTypes.FindNodeOrInsertPos(ID, InsertPos)) | 
|  | 3949 | return QualType(QT, 0); | 
|  | 3950 |  | 
|  | 3951 | // Find the canonical object type. | 
|  | 3952 | QualType Canonical; | 
|  | 3953 | if (!ObjectT.isCanonical()) { | 
|  | 3954 | Canonical = getObjCObjectPointerType(getCanonicalType(ObjectT)); | 
|  | 3955 |  | 
|  | 3956 | // Regenerate InsertPos. | 
| John McCall | fc93cf9 | 2009-10-22 22:37:11 +0000 | [diff] [blame] | 3957 | ObjCObjectPointerTypes.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 3958 | } | 
|  | 3959 |  | 
| Douglas Gregor | f85bee6 | 2010-02-08 22:59:26 +0000 | [diff] [blame] | 3960 | // No match. | 
| John McCall | 8b07ec2 | 2010-05-15 11:32:37 +0000 | [diff] [blame] | 3961 | void *Mem = Allocate(sizeof(ObjCObjectPointerType), TypeAlignment); | 
|  | 3962 | ObjCObjectPointerType *QType = | 
|  | 3963 | new (Mem) ObjCObjectPointerType(Canonical, ObjectT); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3964 |  | 
| Steve Naroff | fb4330f | 2009-06-17 22:40:22 +0000 | [diff] [blame] | 3965 | Types.push_back(QType); | 
|  | 3966 | ObjCObjectPointerTypes.InsertNode(QType, InsertPos); | 
| John McCall | 8b07ec2 | 2010-05-15 11:32:37 +0000 | [diff] [blame] | 3967 | return QualType(QType, 0); | 
| Steve Naroff | fb4330f | 2009-06-17 22:40:22 +0000 | [diff] [blame] | 3968 | } | 
| Chris Lattner | e0ea37a | 2008-04-07 04:56:42 +0000 | [diff] [blame] | 3969 |  | 
| Douglas Gregor | 1c28331 | 2010-08-11 12:19:30 +0000 | [diff] [blame] | 3970 | /// getObjCInterfaceType - Return the unique reference to the type for the | 
|  | 3971 | /// specified ObjC interface decl. The list of protocols is optional. | 
| Douglas Gregor | ab1ec82e | 2011-12-16 03:12:41 +0000 | [diff] [blame] | 3972 | QualType ASTContext::getObjCInterfaceType(const ObjCInterfaceDecl *Decl, | 
|  | 3973 | ObjCInterfaceDecl *PrevDecl) const { | 
| Douglas Gregor | 1c28331 | 2010-08-11 12:19:30 +0000 | [diff] [blame] | 3974 | if (Decl->TypeForDecl) | 
|  | 3975 | return QualType(Decl->TypeForDecl, 0); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3976 |  | 
| Douglas Gregor | ab1ec82e | 2011-12-16 03:12:41 +0000 | [diff] [blame] | 3977 | if (PrevDecl) { | 
|  | 3978 | assert(PrevDecl->TypeForDecl && "previous decl has no TypeForDecl"); | 
|  | 3979 | Decl->TypeForDecl = PrevDecl->TypeForDecl; | 
|  | 3980 | return QualType(PrevDecl->TypeForDecl, 0); | 
|  | 3981 | } | 
|  | 3982 |  | 
| Douglas Gregor | 7671e53 | 2011-12-16 16:34:57 +0000 | [diff] [blame] | 3983 | // Prefer the definition, if there is one. | 
|  | 3984 | if (const ObjCInterfaceDecl *Def = Decl->getDefinition()) | 
|  | 3985 | Decl = Def; | 
|  | 3986 |  | 
| Douglas Gregor | 1c28331 | 2010-08-11 12:19:30 +0000 | [diff] [blame] | 3987 | void *Mem = Allocate(sizeof(ObjCInterfaceType), TypeAlignment); | 
|  | 3988 | ObjCInterfaceType *T = new (Mem) ObjCInterfaceType(Decl); | 
|  | 3989 | Decl->TypeForDecl = T; | 
|  | 3990 | Types.push_back(T); | 
|  | 3991 | return QualType(T, 0); | 
| Fariborz Jahanian | 70e8f10 | 2007-10-11 00:55:41 +0000 | [diff] [blame] | 3992 | } | 
|  | 3993 |  | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 3994 | /// getTypeOfExprType - Unlike many "get<Type>" functions, we can't unique | 
|  | 3995 | /// TypeOfExprType AST's (since expression's are never shared). For example, | 
| Steve Naroff | a773cd5 | 2007-08-01 18:02:17 +0000 | [diff] [blame] | 3996 | /// multiple declarations that refer to "typeof(x)" all contain different | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 3997 | /// DeclRefExpr's. This doesn't effect the type checker, since it operates | 
| Steve Naroff | a773cd5 | 2007-08-01 18:02:17 +0000 | [diff] [blame] | 3998 | /// on canonical type's (which are always unique). | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 3999 | QualType ASTContext::getTypeOfExprType(Expr *tofExpr) const { | 
| Douglas Gregor | abd6813 | 2009-07-08 00:03:05 +0000 | [diff] [blame] | 4000 | TypeOfExprType *toe; | 
| Douglas Gregor | a5dd9f8 | 2009-07-30 23:18:24 +0000 | [diff] [blame] | 4001 | if (tofExpr->isTypeDependent()) { | 
|  | 4002 | llvm::FoldingSetNodeID ID; | 
|  | 4003 | DependentTypeOfExprType::Profile(ID, *this, tofExpr); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4004 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 4005 | void *InsertPos = nullptr; | 
| Douglas Gregor | a5dd9f8 | 2009-07-30 23:18:24 +0000 | [diff] [blame] | 4006 | DependentTypeOfExprType *Canon | 
|  | 4007 | = DependentTypeOfExprTypes.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 4008 | if (Canon) { | 
|  | 4009 | // We already have a "canonical" version of an identical, dependent | 
|  | 4010 | // typeof(expr) type. Use that as our canonical type. | 
| John McCall | 90d1c2d | 2009-09-24 23:30:46 +0000 | [diff] [blame] | 4011 | toe = new (*this, TypeAlignment) TypeOfExprType(tofExpr, | 
| Douglas Gregor | a5dd9f8 | 2009-07-30 23:18:24 +0000 | [diff] [blame] | 4012 | QualType((TypeOfExprType*)Canon, 0)); | 
| Chad Rosier | 6fdf38b | 2011-08-17 23:08:45 +0000 | [diff] [blame] | 4013 | } else { | 
| Douglas Gregor | a5dd9f8 | 2009-07-30 23:18:24 +0000 | [diff] [blame] | 4014 | // Build a new, canonical typeof(expr) type. | 
| John McCall | 90d1c2d | 2009-09-24 23:30:46 +0000 | [diff] [blame] | 4015 | Canon | 
|  | 4016 | = new (*this, TypeAlignment) DependentTypeOfExprType(*this, tofExpr); | 
| Douglas Gregor | a5dd9f8 | 2009-07-30 23:18:24 +0000 | [diff] [blame] | 4017 | DependentTypeOfExprTypes.InsertNode(Canon, InsertPos); | 
|  | 4018 | toe = Canon; | 
|  | 4019 | } | 
|  | 4020 | } else { | 
| Douglas Gregor | abd6813 | 2009-07-08 00:03:05 +0000 | [diff] [blame] | 4021 | QualType Canonical = getCanonicalType(tofExpr->getType()); | 
| John McCall | 90d1c2d | 2009-09-24 23:30:46 +0000 | [diff] [blame] | 4022 | toe = new (*this, TypeAlignment) TypeOfExprType(tofExpr, Canonical); | 
| Douglas Gregor | abd6813 | 2009-07-08 00:03:05 +0000 | [diff] [blame] | 4023 | } | 
| Steve Naroff | a773cd5 | 2007-08-01 18:02:17 +0000 | [diff] [blame] | 4024 | Types.push_back(toe); | 
|  | 4025 | return QualType(toe, 0); | 
| Steve Naroff | ad373bd | 2007-07-31 12:34:36 +0000 | [diff] [blame] | 4026 | } | 
|  | 4027 |  | 
| Steve Naroff | a773cd5 | 2007-08-01 18:02:17 +0000 | [diff] [blame] | 4028 | /// getTypeOfType -  Unlike many "get<Type>" functions, we don't unique | 
| Richard Smith | 8eb1d32 | 2014-06-05 20:13:13 +0000 | [diff] [blame] | 4029 | /// TypeOfType nodes. The only motivation to unique these nodes would be | 
| Steve Naroff | a773cd5 | 2007-08-01 18:02:17 +0000 | [diff] [blame] | 4030 | /// memory savings. Since typeof(t) is fairly uncommon, space shouldn't be | 
| Richard Smith | 8eb1d32 | 2014-06-05 20:13:13 +0000 | [diff] [blame] | 4031 | /// an issue. This doesn't affect the type checker, since it operates | 
|  | 4032 | /// on canonical types (which are always unique). | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 4033 | QualType ASTContext::getTypeOfType(QualType tofType) const { | 
| Chris Lattner | 76a00cf | 2008-04-06 22:59:24 +0000 | [diff] [blame] | 4034 | QualType Canonical = getCanonicalType(tofType); | 
| John McCall | 90d1c2d | 2009-09-24 23:30:46 +0000 | [diff] [blame] | 4035 | TypeOfType *tot = new (*this, TypeAlignment) TypeOfType(tofType, Canonical); | 
| Steve Naroff | a773cd5 | 2007-08-01 18:02:17 +0000 | [diff] [blame] | 4036 | Types.push_back(tot); | 
|  | 4037 | return QualType(tot, 0); | 
| Steve Naroff | ad373bd | 2007-07-31 12:34:36 +0000 | [diff] [blame] | 4038 | } | 
|  | 4039 |  | 
| Richard Smith | 8eb1d32 | 2014-06-05 20:13:13 +0000 | [diff] [blame] | 4040 | /// \brief Unlike many "get<Type>" functions, we don't unique DecltypeType | 
|  | 4041 | /// nodes. This would never be helpful, since each such type has its own | 
|  | 4042 | /// expression, and would not give a significant memory saving, since there | 
|  | 4043 | /// is an Expr tree under each such type. | 
| Douglas Gregor | 81495f3 | 2012-02-12 18:42:33 +0000 | [diff] [blame] | 4044 | QualType ASTContext::getDecltypeType(Expr *e, QualType UnderlyingType) const { | 
| Douglas Gregor | abd6813 | 2009-07-08 00:03:05 +0000 | [diff] [blame] | 4045 | DecltypeType *dt; | 
| Richard Smith | 8eb1d32 | 2014-06-05 20:13:13 +0000 | [diff] [blame] | 4046 |  | 
|  | 4047 | // C++11 [temp.type]p2: | 
| Douglas Gregor | 678d76c | 2011-07-01 01:22:09 +0000 | [diff] [blame] | 4048 | //   If an expression e involves a template parameter, decltype(e) denotes a | 
| Richard Smith | 8eb1d32 | 2014-06-05 20:13:13 +0000 | [diff] [blame] | 4049 | //   unique dependent type. Two such decltype-specifiers refer to the same | 
|  | 4050 | //   type only if their expressions are equivalent (14.5.6.1). | 
| Douglas Gregor | 678d76c | 2011-07-01 01:22:09 +0000 | [diff] [blame] | 4051 | if (e->isInstantiationDependent()) { | 
| Douglas Gregor | a21f6c3 | 2009-07-30 23:36:40 +0000 | [diff] [blame] | 4052 | llvm::FoldingSetNodeID ID; | 
|  | 4053 | DependentDecltypeType::Profile(ID, *this, e); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4054 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 4055 | void *InsertPos = nullptr; | 
| Douglas Gregor | a21f6c3 | 2009-07-30 23:36:40 +0000 | [diff] [blame] | 4056 | DependentDecltypeType *Canon | 
|  | 4057 | = DependentDecltypeTypes.FindNodeOrInsertPos(ID, InsertPos); | 
| Richard Smith | 8eb1d32 | 2014-06-05 20:13:13 +0000 | [diff] [blame] | 4058 | if (!Canon) { | 
| Douglas Gregor | a21f6c3 | 2009-07-30 23:36:40 +0000 | [diff] [blame] | 4059 | // Build a new, canonical typeof(expr) type. | 
| John McCall | 90d1c2d | 2009-09-24 23:30:46 +0000 | [diff] [blame] | 4060 | Canon = new (*this, TypeAlignment) DependentDecltypeType(*this, e); | 
| Douglas Gregor | a21f6c3 | 2009-07-30 23:36:40 +0000 | [diff] [blame] | 4061 | DependentDecltypeTypes.InsertNode(Canon, InsertPos); | 
| Douglas Gregor | a21f6c3 | 2009-07-30 23:36:40 +0000 | [diff] [blame] | 4062 | } | 
| Richard Smith | 8eb1d32 | 2014-06-05 20:13:13 +0000 | [diff] [blame] | 4063 | dt = new (*this, TypeAlignment) | 
|  | 4064 | DecltypeType(e, UnderlyingType, QualType((DecltypeType *)Canon, 0)); | 
| Douglas Gregor | a21f6c3 | 2009-07-30 23:36:40 +0000 | [diff] [blame] | 4065 | } else { | 
| Richard Smith | 8eb1d32 | 2014-06-05 20:13:13 +0000 | [diff] [blame] | 4066 | dt = new (*this, TypeAlignment) | 
|  | 4067 | DecltypeType(e, UnderlyingType, getCanonicalType(UnderlyingType)); | 
| Douglas Gregor | abd6813 | 2009-07-08 00:03:05 +0000 | [diff] [blame] | 4068 | } | 
| Anders Carlsson | 81df7b8 | 2009-06-24 19:06:50 +0000 | [diff] [blame] | 4069 | Types.push_back(dt); | 
|  | 4070 | return QualType(dt, 0); | 
|  | 4071 | } | 
|  | 4072 |  | 
| Alexis Hunt | e852b10 | 2011-05-24 22:41:36 +0000 | [diff] [blame] | 4073 | /// getUnaryTransformationType - We don't unique these, since the memory | 
|  | 4074 | /// savings are minimal and these are rare. | 
|  | 4075 | QualType ASTContext::getUnaryTransformType(QualType BaseType, | 
|  | 4076 | QualType UnderlyingType, | 
|  | 4077 | UnaryTransformType::UTTKind Kind) | 
|  | 4078 | const { | 
| Vassil Vassilev | bab6f96 | 2016-03-30 22:18:29 +0000 | [diff] [blame] | 4079 | UnaryTransformType *ut = nullptr; | 
|  | 4080 |  | 
|  | 4081 | if (BaseType->isDependentType()) { | 
|  | 4082 | // Look in the folding set for an existing type. | 
|  | 4083 | llvm::FoldingSetNodeID ID; | 
|  | 4084 | DependentUnaryTransformType::Profile(ID, getCanonicalType(BaseType), Kind); | 
|  | 4085 |  | 
|  | 4086 | void *InsertPos = nullptr; | 
|  | 4087 | DependentUnaryTransformType *Canon | 
|  | 4088 | = DependentUnaryTransformTypes.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 4089 |  | 
|  | 4090 | if (!Canon) { | 
|  | 4091 | // Build a new, canonical __underlying_type(type) type. | 
|  | 4092 | Canon = new (*this, TypeAlignment) | 
|  | 4093 | DependentUnaryTransformType(*this, getCanonicalType(BaseType), | 
|  | 4094 | Kind); | 
|  | 4095 | DependentUnaryTransformTypes.InsertNode(Canon, InsertPos); | 
|  | 4096 | } | 
|  | 4097 | ut = new (*this, TypeAlignment) UnaryTransformType (BaseType, | 
|  | 4098 | QualType(), Kind, | 
|  | 4099 | QualType(Canon, 0)); | 
|  | 4100 | } else { | 
|  | 4101 | QualType CanonType = getCanonicalType(UnderlyingType); | 
|  | 4102 | ut = new (*this, TypeAlignment) UnaryTransformType (BaseType, | 
|  | 4103 | UnderlyingType, Kind, | 
|  | 4104 | CanonType); | 
|  | 4105 | } | 
|  | 4106 | Types.push_back(ut); | 
|  | 4107 | return QualType(ut, 0); | 
| Alexis Hunt | e852b10 | 2011-05-24 22:41:36 +0000 | [diff] [blame] | 4108 | } | 
|  | 4109 |  | 
| Richard Smith | 2a7d481 | 2013-05-04 07:00:32 +0000 | [diff] [blame] | 4110 | /// getAutoType - Return the uniqued reference to the 'auto' type which has been | 
|  | 4111 | /// deduced to the given type, or to the canonical undeduced 'auto' type, or the | 
|  | 4112 | /// canonical deduced-but-dependent 'auto' type. | 
| Richard Smith | e301ba2 | 2015-11-11 02:02:15 +0000 | [diff] [blame] | 4113 | QualType ASTContext::getAutoType(QualType DeducedType, AutoTypeKeyword Keyword, | 
| Manuel Klimek | 2fdbea2 | 2013-08-22 12:12:24 +0000 | [diff] [blame] | 4114 | bool IsDependent) const { | 
| Richard Smith | e301ba2 | 2015-11-11 02:02:15 +0000 | [diff] [blame] | 4115 | if (DeducedType.isNull() && Keyword == AutoTypeKeyword::Auto && !IsDependent) | 
| Richard Smith | 2a7d481 | 2013-05-04 07:00:32 +0000 | [diff] [blame] | 4116 | return getAutoDeductType(); | 
| Manuel Klimek | 2fdbea2 | 2013-08-22 12:12:24 +0000 | [diff] [blame] | 4117 |  | 
| Richard Smith | 2a7d481 | 2013-05-04 07:00:32 +0000 | [diff] [blame] | 4118 | // Look in the folding set for an existing type. | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 4119 | void *InsertPos = nullptr; | 
| Richard Smith | 2a7d481 | 2013-05-04 07:00:32 +0000 | [diff] [blame] | 4120 | llvm::FoldingSetNodeID ID; | 
| Richard Smith | e301ba2 | 2015-11-11 02:02:15 +0000 | [diff] [blame] | 4121 | AutoType::Profile(ID, DeducedType, Keyword, IsDependent); | 
| Richard Smith | 2a7d481 | 2013-05-04 07:00:32 +0000 | [diff] [blame] | 4122 | if (AutoType *AT = AutoTypes.FindNodeOrInsertPos(ID, InsertPos)) | 
|  | 4123 | return QualType(AT, 0); | 
| Richard Smith | b2bc2e6 | 2011-02-21 20:05:19 +0000 | [diff] [blame] | 4124 |  | 
| Richard Smith | 74aeef5 | 2013-04-26 16:15:35 +0000 | [diff] [blame] | 4125 | AutoType *AT = new (*this, TypeAlignment) AutoType(DeducedType, | 
| Richard Smith | e301ba2 | 2015-11-11 02:02:15 +0000 | [diff] [blame] | 4126 | Keyword, | 
| Manuel Klimek | 2fdbea2 | 2013-08-22 12:12:24 +0000 | [diff] [blame] | 4127 | IsDependent); | 
| Richard Smith | b2bc2e6 | 2011-02-21 20:05:19 +0000 | [diff] [blame] | 4128 | Types.push_back(AT); | 
|  | 4129 | if (InsertPos) | 
|  | 4130 | AutoTypes.InsertNode(AT, InsertPos); | 
|  | 4131 | return QualType(AT, 0); | 
| Richard Smith | 30482bc | 2011-02-20 03:19:35 +0000 | [diff] [blame] | 4132 | } | 
|  | 4133 |  | 
| Eli Friedman | 0dfb889 | 2011-10-06 23:00:33 +0000 | [diff] [blame] | 4134 | /// getAtomicType - Return the uniqued reference to the atomic type for | 
|  | 4135 | /// the given value type. | 
|  | 4136 | QualType ASTContext::getAtomicType(QualType T) const { | 
|  | 4137 | // Unique pointers, to guarantee there is only one pointer of a particular | 
|  | 4138 | // structure. | 
|  | 4139 | llvm::FoldingSetNodeID ID; | 
|  | 4140 | AtomicType::Profile(ID, T); | 
|  | 4141 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 4142 | void *InsertPos = nullptr; | 
| Eli Friedman | 0dfb889 | 2011-10-06 23:00:33 +0000 | [diff] [blame] | 4143 | if (AtomicType *AT = AtomicTypes.FindNodeOrInsertPos(ID, InsertPos)) | 
|  | 4144 | return QualType(AT, 0); | 
|  | 4145 |  | 
|  | 4146 | // If the atomic value type isn't canonical, this won't be a canonical type | 
|  | 4147 | // either, so fill in the canonical type field. | 
|  | 4148 | QualType Canonical; | 
|  | 4149 | if (!T.isCanonical()) { | 
|  | 4150 | Canonical = getAtomicType(getCanonicalType(T)); | 
|  | 4151 |  | 
|  | 4152 | // Get the new insert position for the node we care about. | 
|  | 4153 | AtomicType *NewIP = AtomicTypes.FindNodeOrInsertPos(ID, InsertPos); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 4154 | assert(!NewIP && "Shouldn't be in the map!"); (void)NewIP; | 
| Eli Friedman | 0dfb889 | 2011-10-06 23:00:33 +0000 | [diff] [blame] | 4155 | } | 
|  | 4156 | AtomicType *New = new (*this, TypeAlignment) AtomicType(T, Canonical); | 
|  | 4157 | Types.push_back(New); | 
|  | 4158 | AtomicTypes.InsertNode(New, InsertPos); | 
|  | 4159 | return QualType(New, 0); | 
|  | 4160 | } | 
|  | 4161 |  | 
| Richard Smith | 02e85f3 | 2011-04-14 22:09:26 +0000 | [diff] [blame] | 4162 | /// getAutoDeductType - Get type pattern for deducing against 'auto'. | 
|  | 4163 | QualType ASTContext::getAutoDeductType() const { | 
|  | 4164 | if (AutoDeductTy.isNull()) | 
| Richard Smith | 2a7d481 | 2013-05-04 07:00:32 +0000 | [diff] [blame] | 4165 | AutoDeductTy = QualType( | 
| Richard Smith | e301ba2 | 2015-11-11 02:02:15 +0000 | [diff] [blame] | 4166 | new (*this, TypeAlignment) AutoType(QualType(), AutoTypeKeyword::Auto, | 
| Manuel Klimek | 2fdbea2 | 2013-08-22 12:12:24 +0000 | [diff] [blame] | 4167 | /*dependent*/false), | 
| Richard Smith | 2a7d481 | 2013-05-04 07:00:32 +0000 | [diff] [blame] | 4168 | 0); | 
| Richard Smith | 02e85f3 | 2011-04-14 22:09:26 +0000 | [diff] [blame] | 4169 | return AutoDeductTy; | 
|  | 4170 | } | 
|  | 4171 |  | 
|  | 4172 | /// getAutoRRefDeductType - Get type pattern for deducing against 'auto &&'. | 
|  | 4173 | QualType ASTContext::getAutoRRefDeductType() const { | 
|  | 4174 | if (AutoRRefDeductTy.isNull()) | 
|  | 4175 | AutoRRefDeductTy = getRValueReferenceType(getAutoDeductType()); | 
|  | 4176 | assert(!AutoRRefDeductTy.isNull() && "can't build 'auto &&' pattern"); | 
|  | 4177 | return AutoRRefDeductTy; | 
|  | 4178 | } | 
|  | 4179 |  | 
| Chris Lattner | fb07246 | 2007-01-23 05:45:31 +0000 | [diff] [blame] | 4180 | /// getTagDeclType - Return the unique reference to the type for the | 
|  | 4181 | /// specified TagDecl (struct/union/class/enum) decl. | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 4182 | QualType ASTContext::getTagDeclType(const TagDecl *Decl) const { | 
| Ted Kremenek | 2b0ce11 | 2007-11-26 21:16:01 +0000 | [diff] [blame] | 4183 | assert (Decl); | 
| Mike Stump | b93185d | 2009-08-07 18:05:12 +0000 | [diff] [blame] | 4184 | // FIXME: What is the design on getTagDeclType when it requires casting | 
|  | 4185 | // away const?  mutable? | 
|  | 4186 | return getTypeDeclType(const_cast<TagDecl*>(Decl)); | 
| Chris Lattner | fb07246 | 2007-01-23 05:45:31 +0000 | [diff] [blame] | 4187 | } | 
|  | 4188 |  | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4189 | /// getSizeType - Return the unique type for "size_t" (C99 7.17), the result | 
|  | 4190 | /// of the sizeof operator (C99 6.5.3.4p4). The value is target dependent and | 
|  | 4191 | /// needs to agree with the definition in <stddef.h>. | 
| Anders Carlsson | 22f443f | 2009-12-12 00:26:23 +0000 | [diff] [blame] | 4192 | CanQualType ASTContext::getSizeType() const { | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 4193 | return getFromTargetType(Target->getSizeType()); | 
| Steve Naroff | 92e30f8 | 2007-04-02 22:35:25 +0000 | [diff] [blame] | 4194 | } | 
| Chris Lattner | fb07246 | 2007-01-23 05:45:31 +0000 | [diff] [blame] | 4195 |  | 
| Hans Wennborg | 27541db | 2011-10-27 08:29:09 +0000 | [diff] [blame] | 4196 | /// getIntMaxType - Return the unique type for "intmax_t" (C99 7.18.1.5). | 
|  | 4197 | CanQualType ASTContext::getIntMaxType() const { | 
|  | 4198 | return getFromTargetType(Target->getIntMaxType()); | 
|  | 4199 | } | 
|  | 4200 |  | 
|  | 4201 | /// getUIntMaxType - Return the unique type for "uintmax_t" (C99 7.18.1.5). | 
|  | 4202 | CanQualType ASTContext::getUIntMaxType() const { | 
|  | 4203 | return getFromTargetType(Target->getUIntMaxType()); | 
|  | 4204 | } | 
|  | 4205 |  | 
| Argyrios Kyrtzidis | 40e9e48 | 2008-08-09 16:51:54 +0000 | [diff] [blame] | 4206 | /// getSignedWCharType - Return the type of "signed wchar_t". | 
|  | 4207 | /// Used when in C++, as a GCC extension. | 
|  | 4208 | QualType ASTContext::getSignedWCharType() const { | 
|  | 4209 | // FIXME: derive from "Target" ? | 
|  | 4210 | return WCharTy; | 
|  | 4211 | } | 
|  | 4212 |  | 
|  | 4213 | /// getUnsignedWCharType - Return the type of "unsigned wchar_t". | 
|  | 4214 | /// Used when in C++, as a GCC extension. | 
|  | 4215 | QualType ASTContext::getUnsignedWCharType() const { | 
|  | 4216 | // FIXME: derive from "Target" ? | 
|  | 4217 | return UnsignedIntTy; | 
|  | 4218 | } | 
|  | 4219 |  | 
| Enea Zaffanella | f11ceb6 | 2013-01-26 17:08:37 +0000 | [diff] [blame] | 4220 | QualType ASTContext::getIntPtrType() const { | 
|  | 4221 | return getFromTargetType(Target->getIntPtrType()); | 
|  | 4222 | } | 
|  | 4223 |  | 
|  | 4224 | QualType ASTContext::getUIntPtrType() const { | 
|  | 4225 | return getCorrespondingUnsignedType(getIntPtrType()); | 
|  | 4226 | } | 
|  | 4227 |  | 
| Hans Wennborg | 27541db | 2011-10-27 08:29:09 +0000 | [diff] [blame] | 4228 | /// getPointerDiffType - Return the unique type for "ptrdiff_t" (C99 7.17) | 
| Chris Lattner | d2b88ab | 2007-07-13 03:05:23 +0000 | [diff] [blame] | 4229 | /// defined in <stddef.h>. Pointer - pointer requires this (C99 6.5.6p9). | 
|  | 4230 | QualType ASTContext::getPointerDiffType() const { | 
| Douglas Gregor | e8bbc12 | 2011-09-02 00:18:52 +0000 | [diff] [blame] | 4231 | return getFromTargetType(Target->getPtrDiffType(0)); | 
| Chris Lattner | d2b88ab | 2007-07-13 03:05:23 +0000 | [diff] [blame] | 4232 | } | 
|  | 4233 |  | 
| Eli Friedman | 4e91899e | 2012-11-27 02:58:24 +0000 | [diff] [blame] | 4234 | /// \brief Return the unique type for "pid_t" defined in | 
|  | 4235 | /// <sys/types.h>. We need this to compute the correct type for vfork(). | 
|  | 4236 | QualType ASTContext::getProcessIDType() const { | 
|  | 4237 | return getFromTargetType(Target->getProcessIDType()); | 
|  | 4238 | } | 
|  | 4239 |  | 
| Chris Lattner | a21ad80 | 2008-04-02 05:18:44 +0000 | [diff] [blame] | 4240 | //===----------------------------------------------------------------------===// | 
|  | 4241 | //                              Type Operators | 
|  | 4242 | //===----------------------------------------------------------------------===// | 
|  | 4243 |  | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 4244 | CanQualType ASTContext::getCanonicalParamType(QualType T) const { | 
| John McCall | fc93cf9 | 2009-10-22 22:37:11 +0000 | [diff] [blame] | 4245 | // Push qualifiers into arrays, and then discard any remaining | 
|  | 4246 | // qualifiers. | 
|  | 4247 | T = getCanonicalType(T); | 
| Fariborz Jahanian | 8fb87ae | 2010-09-24 17:30:16 +0000 | [diff] [blame] | 4248 | T = getVariableArrayDecayedType(T); | 
| John McCall | fc93cf9 | 2009-10-22 22:37:11 +0000 | [diff] [blame] | 4249 | const Type *Ty = T.getTypePtr(); | 
| John McCall | fc93cf9 | 2009-10-22 22:37:11 +0000 | [diff] [blame] | 4250 | QualType Result; | 
|  | 4251 | if (isa<ArrayType>(Ty)) { | 
|  | 4252 | Result = getArrayDecayedType(QualType(Ty,0)); | 
|  | 4253 | } else if (isa<FunctionType>(Ty)) { | 
|  | 4254 | Result = getPointerType(QualType(Ty, 0)); | 
|  | 4255 | } else { | 
|  | 4256 | Result = QualType(Ty, 0); | 
|  | 4257 | } | 
|  | 4258 |  | 
|  | 4259 | return CanQualType::CreateUnsafe(Result); | 
|  | 4260 | } | 
|  | 4261 |  | 
| John McCall | 6c9dd52 | 2011-01-18 07:41:22 +0000 | [diff] [blame] | 4262 | QualType ASTContext::getUnqualifiedArrayType(QualType type, | 
|  | 4263 | Qualifiers &quals) { | 
|  | 4264 | SplitQualType splitType = type.getSplitUnqualifiedType(); | 
|  | 4265 |  | 
|  | 4266 | // FIXME: getSplitUnqualifiedType() actually walks all the way to | 
|  | 4267 | // the unqualified desugared type and then drops it on the floor. | 
|  | 4268 | // We then have to strip that sugar back off with | 
|  | 4269 | // getUnqualifiedDesugaredType(), which is silly. | 
|  | 4270 | const ArrayType *AT = | 
| John McCall | 18ce25e | 2012-02-08 00:46:36 +0000 | [diff] [blame] | 4271 | dyn_cast<ArrayType>(splitType.Ty->getUnqualifiedDesugaredType()); | 
| John McCall | 6c9dd52 | 2011-01-18 07:41:22 +0000 | [diff] [blame] | 4272 |  | 
|  | 4273 | // If we don't have an array, just use the results in splitType. | 
| Douglas Gregor | 3b05bdb | 2010-05-17 18:45:21 +0000 | [diff] [blame] | 4274 | if (!AT) { | 
| John McCall | 18ce25e | 2012-02-08 00:46:36 +0000 | [diff] [blame] | 4275 | quals = splitType.Quals; | 
|  | 4276 | return QualType(splitType.Ty, 0); | 
| Chandler Carruth | 607f38e | 2009-12-29 07:16:59 +0000 | [diff] [blame] | 4277 | } | 
|  | 4278 |  | 
| John McCall | 6c9dd52 | 2011-01-18 07:41:22 +0000 | [diff] [blame] | 4279 | // Otherwise, recurse on the array's element type. | 
|  | 4280 | QualType elementType = AT->getElementType(); | 
|  | 4281 | QualType unqualElementType = getUnqualifiedArrayType(elementType, quals); | 
|  | 4282 |  | 
|  | 4283 | // If that didn't change the element type, AT has no qualifiers, so we | 
|  | 4284 | // can just use the results in splitType. | 
|  | 4285 | if (elementType == unqualElementType) { | 
|  | 4286 | assert(quals.empty()); // from the recursive call | 
| John McCall | 18ce25e | 2012-02-08 00:46:36 +0000 | [diff] [blame] | 4287 | quals = splitType.Quals; | 
|  | 4288 | return QualType(splitType.Ty, 0); | 
| John McCall | 6c9dd52 | 2011-01-18 07:41:22 +0000 | [diff] [blame] | 4289 | } | 
|  | 4290 |  | 
|  | 4291 | // Otherwise, add in the qualifiers from the outermost type, then | 
|  | 4292 | // build the type back up. | 
| John McCall | 18ce25e | 2012-02-08 00:46:36 +0000 | [diff] [blame] | 4293 | quals.addConsistentQualifiers(splitType.Quals); | 
| Chandler Carruth | 607f38e | 2009-12-29 07:16:59 +0000 | [diff] [blame] | 4294 |  | 
| Douglas Gregor | 3b05bdb | 2010-05-17 18:45:21 +0000 | [diff] [blame] | 4295 | if (const ConstantArrayType *CAT = dyn_cast<ConstantArrayType>(AT)) { | 
| John McCall | 6c9dd52 | 2011-01-18 07:41:22 +0000 | [diff] [blame] | 4296 | return getConstantArrayType(unqualElementType, CAT->getSize(), | 
| Chandler Carruth | 607f38e | 2009-12-29 07:16:59 +0000 | [diff] [blame] | 4297 | CAT->getSizeModifier(), 0); | 
|  | 4298 | } | 
|  | 4299 |  | 
| Douglas Gregor | 3b05bdb | 2010-05-17 18:45:21 +0000 | [diff] [blame] | 4300 | if (const IncompleteArrayType *IAT = dyn_cast<IncompleteArrayType>(AT)) { | 
| John McCall | 6c9dd52 | 2011-01-18 07:41:22 +0000 | [diff] [blame] | 4301 | return getIncompleteArrayType(unqualElementType, IAT->getSizeModifier(), 0); | 
| Chandler Carruth | 607f38e | 2009-12-29 07:16:59 +0000 | [diff] [blame] | 4302 | } | 
|  | 4303 |  | 
| Douglas Gregor | 3b05bdb | 2010-05-17 18:45:21 +0000 | [diff] [blame] | 4304 | if (const VariableArrayType *VAT = dyn_cast<VariableArrayType>(AT)) { | 
| John McCall | 6c9dd52 | 2011-01-18 07:41:22 +0000 | [diff] [blame] | 4305 | return getVariableArrayType(unqualElementType, | 
| John McCall | c3007a2 | 2010-10-26 07:05:15 +0000 | [diff] [blame] | 4306 | VAT->getSizeExpr(), | 
| Douglas Gregor | 3b05bdb | 2010-05-17 18:45:21 +0000 | [diff] [blame] | 4307 | VAT->getSizeModifier(), | 
|  | 4308 | VAT->getIndexTypeCVRQualifiers(), | 
|  | 4309 | VAT->getBracketsRange()); | 
|  | 4310 | } | 
|  | 4311 |  | 
|  | 4312 | const DependentSizedArrayType *DSAT = cast<DependentSizedArrayType>(AT); | 
| John McCall | 6c9dd52 | 2011-01-18 07:41:22 +0000 | [diff] [blame] | 4313 | return getDependentSizedArrayType(unqualElementType, DSAT->getSizeExpr(), | 
| Chandler Carruth | 607f38e | 2009-12-29 07:16:59 +0000 | [diff] [blame] | 4314 | DSAT->getSizeModifier(), 0, | 
|  | 4315 | SourceRange()); | 
|  | 4316 | } | 
|  | 4317 |  | 
| Douglas Gregor | 1fc3d66 | 2010-06-09 03:53:18 +0000 | [diff] [blame] | 4318 | /// UnwrapSimilarPointerTypes - If T1 and T2 are pointer types  that | 
|  | 4319 | /// may be similar (C++ 4.4), replaces T1 and T2 with the type that | 
|  | 4320 | /// they point to and return true. If T1 and T2 aren't pointer types | 
|  | 4321 | /// or pointer-to-member types, or if they are not similar at this | 
|  | 4322 | /// level, returns false and leaves T1 and T2 unchanged. Top-level | 
|  | 4323 | /// qualifiers on T1 and T2 are ignored. This function will typically | 
|  | 4324 | /// be called in a loop that successively "unwraps" pointer and | 
|  | 4325 | /// pointer-to-member types to compare them at each level. | 
|  | 4326 | bool ASTContext::UnwrapSimilarPointerTypes(QualType &T1, QualType &T2) { | 
|  | 4327 | const PointerType *T1PtrType = T1->getAs<PointerType>(), | 
|  | 4328 | *T2PtrType = T2->getAs<PointerType>(); | 
|  | 4329 | if (T1PtrType && T2PtrType) { | 
|  | 4330 | T1 = T1PtrType->getPointeeType(); | 
|  | 4331 | T2 = T2PtrType->getPointeeType(); | 
|  | 4332 | return true; | 
|  | 4333 | } | 
|  | 4334 |  | 
|  | 4335 | const MemberPointerType *T1MPType = T1->getAs<MemberPointerType>(), | 
|  | 4336 | *T2MPType = T2->getAs<MemberPointerType>(); | 
|  | 4337 | if (T1MPType && T2MPType && | 
|  | 4338 | hasSameUnqualifiedType(QualType(T1MPType->getClass(), 0), | 
|  | 4339 | QualType(T2MPType->getClass(), 0))) { | 
|  | 4340 | T1 = T1MPType->getPointeeType(); | 
|  | 4341 | T2 = T2MPType->getPointeeType(); | 
|  | 4342 | return true; | 
|  | 4343 | } | 
|  | 4344 |  | 
| David Blaikie | bbafb8a | 2012-03-11 07:00:24 +0000 | [diff] [blame] | 4345 | if (getLangOpts().ObjC1) { | 
| Douglas Gregor | 1fc3d66 | 2010-06-09 03:53:18 +0000 | [diff] [blame] | 4346 | const ObjCObjectPointerType *T1OPType = T1->getAs<ObjCObjectPointerType>(), | 
|  | 4347 | *T2OPType = T2->getAs<ObjCObjectPointerType>(); | 
|  | 4348 | if (T1OPType && T2OPType) { | 
|  | 4349 | T1 = T1OPType->getPointeeType(); | 
|  | 4350 | T2 = T2OPType->getPointeeType(); | 
|  | 4351 | return true; | 
|  | 4352 | } | 
|  | 4353 | } | 
|  | 4354 |  | 
|  | 4355 | // FIXME: Block pointers, too? | 
|  | 4356 |  | 
|  | 4357 | return false; | 
|  | 4358 | } | 
|  | 4359 |  | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 4360 | DeclarationNameInfo | 
|  | 4361 | ASTContext::getNameForTemplate(TemplateName Name, | 
|  | 4362 | SourceLocation NameLoc) const { | 
| John McCall | d9dfe3a | 2011-06-30 08:33:18 +0000 | [diff] [blame] | 4363 | switch (Name.getKind()) { | 
|  | 4364 | case TemplateName::QualifiedTemplate: | 
|  | 4365 | case TemplateName::Template: | 
| Abramo Bagnara | d6d2f18 | 2010-08-11 22:01:17 +0000 | [diff] [blame] | 4366 | // DNInfo work in progress: CHECKME: what about DNLoc? | 
| John McCall | d9dfe3a | 2011-06-30 08:33:18 +0000 | [diff] [blame] | 4367 | return DeclarationNameInfo(Name.getAsTemplateDecl()->getDeclName(), | 
|  | 4368 | NameLoc); | 
| Abramo Bagnara | d6d2f18 | 2010-08-11 22:01:17 +0000 | [diff] [blame] | 4369 |  | 
| John McCall | d9dfe3a | 2011-06-30 08:33:18 +0000 | [diff] [blame] | 4370 | case TemplateName::OverloadedTemplate: { | 
|  | 4371 | OverloadedTemplateStorage *Storage = Name.getAsOverloadedTemplate(); | 
|  | 4372 | // DNInfo work in progress: CHECKME: what about DNLoc? | 
|  | 4373 | return DeclarationNameInfo((*Storage->begin())->getDeclName(), NameLoc); | 
|  | 4374 | } | 
|  | 4375 |  | 
|  | 4376 | case TemplateName::DependentTemplate: { | 
|  | 4377 | DependentTemplateName *DTN = Name.getAsDependentTemplateName(); | 
| Abramo Bagnara | d6d2f18 | 2010-08-11 22:01:17 +0000 | [diff] [blame] | 4378 | DeclarationName DName; | 
| John McCall | 847e2a1 | 2009-11-24 18:42:40 +0000 | [diff] [blame] | 4379 | if (DTN->isIdentifier()) { | 
| Abramo Bagnara | d6d2f18 | 2010-08-11 22:01:17 +0000 | [diff] [blame] | 4380 | DName = DeclarationNames.getIdentifier(DTN->getIdentifier()); | 
|  | 4381 | return DeclarationNameInfo(DName, NameLoc); | 
| John McCall | 847e2a1 | 2009-11-24 18:42:40 +0000 | [diff] [blame] | 4382 | } else { | 
| Abramo Bagnara | d6d2f18 | 2010-08-11 22:01:17 +0000 | [diff] [blame] | 4383 | DName = DeclarationNames.getCXXOperatorName(DTN->getOperator()); | 
|  | 4384 | // DNInfo work in progress: FIXME: source locations? | 
|  | 4385 | DeclarationNameLoc DNLoc; | 
|  | 4386 | DNLoc.CXXOperatorName.BeginOpNameLoc = SourceLocation().getRawEncoding(); | 
|  | 4387 | DNLoc.CXXOperatorName.EndOpNameLoc = SourceLocation().getRawEncoding(); | 
|  | 4388 | return DeclarationNameInfo(DName, NameLoc, DNLoc); | 
| John McCall | 847e2a1 | 2009-11-24 18:42:40 +0000 | [diff] [blame] | 4389 | } | 
|  | 4390 | } | 
|  | 4391 |  | 
| John McCall | d9dfe3a | 2011-06-30 08:33:18 +0000 | [diff] [blame] | 4392 | case TemplateName::SubstTemplateTemplateParm: { | 
|  | 4393 | SubstTemplateTemplateParmStorage *subst | 
|  | 4394 | = Name.getAsSubstTemplateTemplateParm(); | 
|  | 4395 | return DeclarationNameInfo(subst->getParameter()->getDeclName(), | 
|  | 4396 | NameLoc); | 
|  | 4397 | } | 
|  | 4398 |  | 
|  | 4399 | case TemplateName::SubstTemplateTemplateParmPack: { | 
|  | 4400 | SubstTemplateTemplateParmPackStorage *subst | 
|  | 4401 | = Name.getAsSubstTemplateTemplateParmPack(); | 
|  | 4402 | return DeclarationNameInfo(subst->getParameterPack()->getDeclName(), | 
|  | 4403 | NameLoc); | 
|  | 4404 | } | 
|  | 4405 | } | 
|  | 4406 |  | 
|  | 4407 | llvm_unreachable("bad template name kind!"); | 
| John McCall | 847e2a1 | 2009-11-24 18:42:40 +0000 | [diff] [blame] | 4408 | } | 
|  | 4409 |  | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 4410 | TemplateName ASTContext::getCanonicalTemplateName(TemplateName Name) const { | 
| John McCall | d9dfe3a | 2011-06-30 08:33:18 +0000 | [diff] [blame] | 4411 | switch (Name.getKind()) { | 
|  | 4412 | case TemplateName::QualifiedTemplate: | 
|  | 4413 | case TemplateName::Template: { | 
|  | 4414 | TemplateDecl *Template = Name.getAsTemplateDecl(); | 
| Douglas Gregor | 7dbfb46 | 2010-06-16 21:09:37 +0000 | [diff] [blame] | 4415 | if (TemplateTemplateParmDecl *TTP | 
| John McCall | d9dfe3a | 2011-06-30 08:33:18 +0000 | [diff] [blame] | 4416 | = dyn_cast<TemplateTemplateParmDecl>(Template)) | 
| Douglas Gregor | 7dbfb46 | 2010-06-16 21:09:37 +0000 | [diff] [blame] | 4417 | Template = getCanonicalTemplateTemplateParmDecl(TTP); | 
|  | 4418 |  | 
|  | 4419 | // The canonical template name is the canonical template declaration. | 
| Argyrios Kyrtzidis | 6b7e376 | 2009-07-18 00:34:25 +0000 | [diff] [blame] | 4420 | return TemplateName(cast<TemplateDecl>(Template->getCanonicalDecl())); | 
| Douglas Gregor | 7dbfb46 | 2010-06-16 21:09:37 +0000 | [diff] [blame] | 4421 | } | 
| Douglas Gregor | 6bc5058 | 2009-05-07 06:41:52 +0000 | [diff] [blame] | 4422 |  | 
| John McCall | d9dfe3a | 2011-06-30 08:33:18 +0000 | [diff] [blame] | 4423 | case TemplateName::OverloadedTemplate: | 
|  | 4424 | llvm_unreachable("cannot canonicalize overloaded template"); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4425 |  | 
| John McCall | d9dfe3a | 2011-06-30 08:33:18 +0000 | [diff] [blame] | 4426 | case TemplateName::DependentTemplate: { | 
|  | 4427 | DependentTemplateName *DTN = Name.getAsDependentTemplateName(); | 
|  | 4428 | assert(DTN && "Non-dependent template names must refer to template decls."); | 
|  | 4429 | return DTN->CanonicalTemplateName; | 
|  | 4430 | } | 
|  | 4431 |  | 
|  | 4432 | case TemplateName::SubstTemplateTemplateParm: { | 
|  | 4433 | SubstTemplateTemplateParmStorage *subst | 
|  | 4434 | = Name.getAsSubstTemplateTemplateParm(); | 
|  | 4435 | return getCanonicalTemplateName(subst->getReplacement()); | 
|  | 4436 | } | 
|  | 4437 |  | 
|  | 4438 | case TemplateName::SubstTemplateTemplateParmPack: { | 
|  | 4439 | SubstTemplateTemplateParmPackStorage *subst | 
|  | 4440 | = Name.getAsSubstTemplateTemplateParmPack(); | 
|  | 4441 | TemplateTemplateParmDecl *canonParameter | 
|  | 4442 | = getCanonicalTemplateTemplateParmDecl(subst->getParameterPack()); | 
|  | 4443 | TemplateArgument canonArgPack | 
|  | 4444 | = getCanonicalTemplateArgument(subst->getArgumentPack()); | 
|  | 4445 | return getSubstTemplateTemplateParmPack(canonParameter, canonArgPack); | 
|  | 4446 | } | 
|  | 4447 | } | 
|  | 4448 |  | 
|  | 4449 | llvm_unreachable("bad template name!"); | 
| Douglas Gregor | 6bc5058 | 2009-05-07 06:41:52 +0000 | [diff] [blame] | 4450 | } | 
|  | 4451 |  | 
| Douglas Gregor | adee3e3 | 2009-11-11 23:06:43 +0000 | [diff] [blame] | 4452 | bool ASTContext::hasSameTemplateName(TemplateName X, TemplateName Y) { | 
|  | 4453 | X = getCanonicalTemplateName(X); | 
|  | 4454 | Y = getCanonicalTemplateName(Y); | 
|  | 4455 | return X.getAsVoidPointer() == Y.getAsVoidPointer(); | 
|  | 4456 | } | 
|  | 4457 |  | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4458 | TemplateArgument | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 4459 | ASTContext::getCanonicalTemplateArgument(const TemplateArgument &Arg) const { | 
| Douglas Gregor | a8e02e7 | 2009-07-28 23:00:59 +0000 | [diff] [blame] | 4460 | switch (Arg.getKind()) { | 
|  | 4461 | case TemplateArgument::Null: | 
|  | 4462 | return Arg; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4463 |  | 
| Douglas Gregor | a8e02e7 | 2009-07-28 23:00:59 +0000 | [diff] [blame] | 4464 | case TemplateArgument::Expression: | 
| Douglas Gregor | a8e02e7 | 2009-07-28 23:00:59 +0000 | [diff] [blame] | 4465 | return Arg; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4466 |  | 
| Douglas Gregor | 31f55dc | 2012-04-06 22:40:38 +0000 | [diff] [blame] | 4467 | case TemplateArgument::Declaration: { | 
| Eli Friedman | b826a00 | 2012-09-26 02:36:12 +0000 | [diff] [blame] | 4468 | ValueDecl *D = cast<ValueDecl>(Arg.getAsDecl()->getCanonicalDecl()); | 
| David Blaikie | 952a9b1 | 2014-10-17 18:00:12 +0000 | [diff] [blame] | 4469 | return TemplateArgument(D, Arg.getParamTypeForDecl()); | 
| Douglas Gregor | 31f55dc | 2012-04-06 22:40:38 +0000 | [diff] [blame] | 4470 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4471 |  | 
| Eli Friedman | b826a00 | 2012-09-26 02:36:12 +0000 | [diff] [blame] | 4472 | case TemplateArgument::NullPtr: | 
|  | 4473 | return TemplateArgument(getCanonicalType(Arg.getNullPtrType()), | 
|  | 4474 | /*isNullPtr*/true); | 
|  | 4475 |  | 
| Douglas Gregor | 9167f8b | 2009-11-11 01:00:40 +0000 | [diff] [blame] | 4476 | case TemplateArgument::Template: | 
|  | 4477 | return TemplateArgument(getCanonicalTemplateName(Arg.getAsTemplate())); | 
| Douglas Gregor | e4ff4b5 | 2011-01-05 18:58:31 +0000 | [diff] [blame] | 4478 |  | 
|  | 4479 | case TemplateArgument::TemplateExpansion: | 
|  | 4480 | return TemplateArgument(getCanonicalTemplateName( | 
|  | 4481 | Arg.getAsTemplateOrTemplatePattern()), | 
| Douglas Gregor | e1d60df | 2011-01-14 23:41:42 +0000 | [diff] [blame] | 4482 | Arg.getNumTemplateExpansions()); | 
| Douglas Gregor | e4ff4b5 | 2011-01-05 18:58:31 +0000 | [diff] [blame] | 4483 |  | 
| Douglas Gregor | a8e02e7 | 2009-07-28 23:00:59 +0000 | [diff] [blame] | 4484 | case TemplateArgument::Integral: | 
| Benjamin Kramer | 6003ad5 | 2012-06-07 15:09:51 +0000 | [diff] [blame] | 4485 | return TemplateArgument(Arg, getCanonicalType(Arg.getIntegralType())); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4486 |  | 
| Douglas Gregor | a8e02e7 | 2009-07-28 23:00:59 +0000 | [diff] [blame] | 4487 | case TemplateArgument::Type: | 
| John McCall | 0ad1666 | 2009-10-29 08:12:44 +0000 | [diff] [blame] | 4488 | return TemplateArgument(getCanonicalType(Arg.getAsType())); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4489 |  | 
| Douglas Gregor | a8e02e7 | 2009-07-28 23:00:59 +0000 | [diff] [blame] | 4490 | case TemplateArgument::Pack: { | 
| Douglas Gregor | 0192c23 | 2010-12-20 16:52:59 +0000 | [diff] [blame] | 4491 | if (Arg.pack_size() == 0) | 
|  | 4492 | return Arg; | 
|  | 4493 |  | 
| Douglas Gregor | 1ccc841 | 2010-11-07 23:05:16 +0000 | [diff] [blame] | 4494 | TemplateArgument *CanonArgs | 
|  | 4495 | = new (*this) TemplateArgument[Arg.pack_size()]; | 
| Douglas Gregor | a8e02e7 | 2009-07-28 23:00:59 +0000 | [diff] [blame] | 4496 | unsigned Idx = 0; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4497 | for (TemplateArgument::pack_iterator A = Arg.pack_begin(), | 
| Douglas Gregor | a8e02e7 | 2009-07-28 23:00:59 +0000 | [diff] [blame] | 4498 | AEnd = Arg.pack_end(); | 
|  | 4499 | A != AEnd; (void)++A, ++Idx) | 
|  | 4500 | CanonArgs[Idx] = getCanonicalTemplateArgument(*A); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4501 |  | 
| Benjamin Kramer | cce6347 | 2015-08-05 09:40:22 +0000 | [diff] [blame] | 4502 | return TemplateArgument(llvm::makeArrayRef(CanonArgs, Arg.pack_size())); | 
| Douglas Gregor | a8e02e7 | 2009-07-28 23:00:59 +0000 | [diff] [blame] | 4503 | } | 
|  | 4504 | } | 
|  | 4505 |  | 
|  | 4506 | // Silence GCC warning | 
| David Blaikie | 83d382b | 2011-09-23 05:06:16 +0000 | [diff] [blame] | 4507 | llvm_unreachable("Unhandled template argument kind"); | 
| Douglas Gregor | a8e02e7 | 2009-07-28 23:00:59 +0000 | [diff] [blame] | 4508 | } | 
|  | 4509 |  | 
| Douglas Gregor | 333489b | 2009-03-27 23:10:48 +0000 | [diff] [blame] | 4510 | NestedNameSpecifier * | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 4511 | ASTContext::getCanonicalNestedNameSpecifier(NestedNameSpecifier *NNS) const { | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4512 | if (!NNS) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 4513 | return nullptr; | 
| Douglas Gregor | 333489b | 2009-03-27 23:10:48 +0000 | [diff] [blame] | 4514 |  | 
|  | 4515 | switch (NNS->getKind()) { | 
|  | 4516 | case NestedNameSpecifier::Identifier: | 
|  | 4517 | // Canonicalize the prefix but keep the identifier the same. | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4518 | return NestedNameSpecifier::Create(*this, | 
| Douglas Gregor | 333489b | 2009-03-27 23:10:48 +0000 | [diff] [blame] | 4519 | getCanonicalNestedNameSpecifier(NNS->getPrefix()), | 
|  | 4520 | NNS->getAsIdentifier()); | 
|  | 4521 |  | 
|  | 4522 | case NestedNameSpecifier::Namespace: | 
|  | 4523 | // A namespace is canonical; build a nested-name-specifier with | 
|  | 4524 | // this namespace and no prefix. | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 4525 | return NestedNameSpecifier::Create(*this, nullptr, | 
| Douglas Gregor | 7b26ff9 | 2011-02-24 02:36:08 +0000 | [diff] [blame] | 4526 | NNS->getAsNamespace()->getOriginalNamespace()); | 
|  | 4527 |  | 
|  | 4528 | case NestedNameSpecifier::NamespaceAlias: | 
|  | 4529 | // A namespace is canonical; build a nested-name-specifier with | 
|  | 4530 | // this namespace and no prefix. | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 4531 | return NestedNameSpecifier::Create(*this, nullptr, | 
| Douglas Gregor | 7b26ff9 | 2011-02-24 02:36:08 +0000 | [diff] [blame] | 4532 | NNS->getAsNamespaceAlias()->getNamespace() | 
|  | 4533 | ->getOriginalNamespace()); | 
| Douglas Gregor | 333489b | 2009-03-27 23:10:48 +0000 | [diff] [blame] | 4534 |  | 
|  | 4535 | case NestedNameSpecifier::TypeSpec: | 
|  | 4536 | case NestedNameSpecifier::TypeSpecWithTemplate: { | 
|  | 4537 | QualType T = getCanonicalType(QualType(NNS->getAsType(), 0)); | 
| Douglas Gregor | 3ade570 | 2010-11-04 00:09:33 +0000 | [diff] [blame] | 4538 |  | 
|  | 4539 | // If we have some kind of dependent-named type (e.g., "typename T::type"), | 
|  | 4540 | // break it apart into its prefix and identifier, then reconsititute those | 
|  | 4541 | // as the canonical nested-name-specifier. This is required to canonicalize | 
|  | 4542 | // a dependent nested-name-specifier involving typedefs of dependent-name | 
|  | 4543 | // types, e.g., | 
|  | 4544 | //   typedef typename T::type T1; | 
|  | 4545 | //   typedef typename T1::type T2; | 
| Eli Friedman | 5358a0a | 2012-03-03 04:09:56 +0000 | [diff] [blame] | 4546 | if (const DependentNameType *DNT = T->getAs<DependentNameType>()) | 
|  | 4547 | return NestedNameSpecifier::Create(*this, DNT->getQualifier(), | 
| Douglas Gregor | 3ade570 | 2010-11-04 00:09:33 +0000 | [diff] [blame] | 4548 | const_cast<IdentifierInfo *>(DNT->getIdentifier())); | 
| Douglas Gregor | 3ade570 | 2010-11-04 00:09:33 +0000 | [diff] [blame] | 4549 |  | 
| Eli Friedman | 5358a0a | 2012-03-03 04:09:56 +0000 | [diff] [blame] | 4550 | // Otherwise, just canonicalize the type, and force it to be a TypeSpec. | 
|  | 4551 | // FIXME: Why are TypeSpec and TypeSpecWithTemplate distinct in the | 
|  | 4552 | // first place? | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 4553 | return NestedNameSpecifier::Create(*this, nullptr, false, | 
|  | 4554 | const_cast<Type *>(T.getTypePtr())); | 
| Douglas Gregor | 333489b | 2009-03-27 23:10:48 +0000 | [diff] [blame] | 4555 | } | 
|  | 4556 |  | 
|  | 4557 | case NestedNameSpecifier::Global: | 
| Nikola Smiljanic | 6786024 | 2014-09-26 00:28:20 +0000 | [diff] [blame] | 4558 | case NestedNameSpecifier::Super: | 
|  | 4559 | // The global specifier and __super specifer are canonical and unique. | 
| Douglas Gregor | 333489b | 2009-03-27 23:10:48 +0000 | [diff] [blame] | 4560 | return NNS; | 
|  | 4561 | } | 
|  | 4562 |  | 
| David Blaikie | 8a40f70 | 2012-01-17 06:56:22 +0000 | [diff] [blame] | 4563 | llvm_unreachable("Invalid NestedNameSpecifier::Kind!"); | 
| Douglas Gregor | 333489b | 2009-03-27 23:10:48 +0000 | [diff] [blame] | 4564 | } | 
|  | 4565 |  | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 4566 | const ArrayType *ASTContext::getAsArrayType(QualType T) const { | 
| Chris Lattner | 7adf076 | 2008-08-04 07:31:14 +0000 | [diff] [blame] | 4567 | // Handle the non-qualified case efficiently. | 
| Douglas Gregor | 1b8fe5b7 | 2009-11-16 21:35:15 +0000 | [diff] [blame] | 4568 | if (!T.hasLocalQualifiers()) { | 
| Chris Lattner | 7adf076 | 2008-08-04 07:31:14 +0000 | [diff] [blame] | 4569 | // Handle the common positive case fast. | 
|  | 4570 | if (const ArrayType *AT = dyn_cast<ArrayType>(T)) | 
|  | 4571 | return AT; | 
|  | 4572 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4573 |  | 
| John McCall | 8ccfcb5 | 2009-09-24 19:53:00 +0000 | [diff] [blame] | 4574 | // Handle the common negative case fast. | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 4575 | if (!isa<ArrayType>(T.getCanonicalType())) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 4576 | return nullptr; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4577 |  | 
| John McCall | 8ccfcb5 | 2009-09-24 19:53:00 +0000 | [diff] [blame] | 4578 | // Apply any qualifiers from the array type to the element type.  This | 
| Chris Lattner | 7adf076 | 2008-08-04 07:31:14 +0000 | [diff] [blame] | 4579 | // implements C99 6.7.3p8: "If the specification of an array type includes | 
|  | 4580 | // any type qualifiers, the element type is so qualified, not the array type." | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4581 |  | 
| Chris Lattner | 7adf076 | 2008-08-04 07:31:14 +0000 | [diff] [blame] | 4582 | // If we get here, we either have type qualifiers on the type, or we have | 
|  | 4583 | // sugar such as a typedef in the way.  If we have type qualifiers on the type | 
| Douglas Gregor | 2211d34 | 2009-08-05 05:36:45 +0000 | [diff] [blame] | 4584 | // we must propagate them down into the element type. | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4585 |  | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 4586 | SplitQualType split = T.getSplitDesugaredType(); | 
| John McCall | 18ce25e | 2012-02-08 00:46:36 +0000 | [diff] [blame] | 4587 | Qualifiers qs = split.Quals; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4588 |  | 
| Chris Lattner | 7adf076 | 2008-08-04 07:31:14 +0000 | [diff] [blame] | 4589 | // If we have a simple case, just return now. | 
| John McCall | 18ce25e | 2012-02-08 00:46:36 +0000 | [diff] [blame] | 4590 | const ArrayType *ATy = dyn_cast<ArrayType>(split.Ty); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 4591 | if (!ATy || qs.empty()) | 
| Chris Lattner | 7adf076 | 2008-08-04 07:31:14 +0000 | [diff] [blame] | 4592 | return ATy; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4593 |  | 
| Chris Lattner | 7adf076 | 2008-08-04 07:31:14 +0000 | [diff] [blame] | 4594 | // Otherwise, we have an array and we have qualifiers on it.  Push the | 
|  | 4595 | // qualifiers into the array element type and return a new array type. | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 4596 | QualType NewEltTy = getQualifiedType(ATy->getElementType(), qs); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4597 |  | 
| Chris Lattner | 7adf076 | 2008-08-04 07:31:14 +0000 | [diff] [blame] | 4598 | if (const ConstantArrayType *CAT = dyn_cast<ConstantArrayType>(ATy)) | 
|  | 4599 | return cast<ArrayType>(getConstantArrayType(NewEltTy, CAT->getSize(), | 
|  | 4600 | CAT->getSizeModifier(), | 
| John McCall | 8ccfcb5 | 2009-09-24 19:53:00 +0000 | [diff] [blame] | 4601 | CAT->getIndexTypeCVRQualifiers())); | 
| Chris Lattner | 7adf076 | 2008-08-04 07:31:14 +0000 | [diff] [blame] | 4602 | if (const IncompleteArrayType *IAT = dyn_cast<IncompleteArrayType>(ATy)) | 
|  | 4603 | return cast<ArrayType>(getIncompleteArrayType(NewEltTy, | 
|  | 4604 | IAT->getSizeModifier(), | 
| John McCall | 8ccfcb5 | 2009-09-24 19:53:00 +0000 | [diff] [blame] | 4605 | IAT->getIndexTypeCVRQualifiers())); | 
| Douglas Gregor | 4619e43 | 2008-12-05 23:32:09 +0000 | [diff] [blame] | 4606 |  | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4607 | if (const DependentSizedArrayType *DSAT | 
| Douglas Gregor | 4619e43 | 2008-12-05 23:32:09 +0000 | [diff] [blame] | 4608 | = dyn_cast<DependentSizedArrayType>(ATy)) | 
|  | 4609 | return cast<ArrayType>( | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4610 | getDependentSizedArrayType(NewEltTy, | 
| John McCall | c3007a2 | 2010-10-26 07:05:15 +0000 | [diff] [blame] | 4611 | DSAT->getSizeExpr(), | 
| Douglas Gregor | 4619e43 | 2008-12-05 23:32:09 +0000 | [diff] [blame] | 4612 | DSAT->getSizeModifier(), | 
| John McCall | 8ccfcb5 | 2009-09-24 19:53:00 +0000 | [diff] [blame] | 4613 | DSAT->getIndexTypeCVRQualifiers(), | 
| Douglas Gregor | 0431825 | 2009-07-06 15:59:29 +0000 | [diff] [blame] | 4614 | DSAT->getBracketsRange())); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4615 |  | 
| Chris Lattner | 7adf076 | 2008-08-04 07:31:14 +0000 | [diff] [blame] | 4616 | const VariableArrayType *VAT = cast<VariableArrayType>(ATy); | 
| Douglas Gregor | 0431825 | 2009-07-06 15:59:29 +0000 | [diff] [blame] | 4617 | return cast<ArrayType>(getVariableArrayType(NewEltTy, | 
| John McCall | c3007a2 | 2010-10-26 07:05:15 +0000 | [diff] [blame] | 4618 | VAT->getSizeExpr(), | 
| Chris Lattner | 7adf076 | 2008-08-04 07:31:14 +0000 | [diff] [blame] | 4619 | VAT->getSizeModifier(), | 
| John McCall | 8ccfcb5 | 2009-09-24 19:53:00 +0000 | [diff] [blame] | 4620 | VAT->getIndexTypeCVRQualifiers(), | 
| Douglas Gregor | 0431825 | 2009-07-06 15:59:29 +0000 | [diff] [blame] | 4621 | VAT->getBracketsRange())); | 
| Chris Lattner | ed0d079 | 2008-04-06 22:41:35 +0000 | [diff] [blame] | 4622 | } | 
|  | 4623 |  | 
| Abramo Bagnara | e1decdf | 2012-05-17 12:44:05 +0000 | [diff] [blame] | 4624 | QualType ASTContext::getAdjustedParameterType(QualType T) const { | 
| Reid Kleckner | 8a36502 | 2013-06-24 17:51:48 +0000 | [diff] [blame] | 4625 | if (T->isArrayType() || T->isFunctionType()) | 
|  | 4626 | return getDecayedType(T); | 
|  | 4627 | return T; | 
| Douglas Gregor | 8428064 | 2011-07-12 04:42:08 +0000 | [diff] [blame] | 4628 | } | 
|  | 4629 |  | 
| Abramo Bagnara | e1decdf | 2012-05-17 12:44:05 +0000 | [diff] [blame] | 4630 | QualType ASTContext::getSignatureParameterType(QualType T) const { | 
| Douglas Gregor | 8428064 | 2011-07-12 04:42:08 +0000 | [diff] [blame] | 4631 | T = getVariableArrayDecayedType(T); | 
|  | 4632 | T = getAdjustedParameterType(T); | 
|  | 4633 | return T.getUnqualifiedType(); | 
|  | 4634 | } | 
|  | 4635 |  | 
| David Majnemer | d09a51c | 2015-03-03 01:50:05 +0000 | [diff] [blame] | 4636 | QualType ASTContext::getExceptionObjectType(QualType T) const { | 
|  | 4637 | // C++ [except.throw]p3: | 
|  | 4638 | //   A throw-expression initializes a temporary object, called the exception | 
|  | 4639 | //   object, the type of which is determined by removing any top-level | 
|  | 4640 | //   cv-qualifiers from the static type of the operand of throw and adjusting | 
|  | 4641 | //   the type from "array of T" or "function returning T" to "pointer to T" | 
|  | 4642 | //   or "pointer to function returning T", [...] | 
|  | 4643 | T = getVariableArrayDecayedType(T); | 
|  | 4644 | if (T->isArrayType() || T->isFunctionType()) | 
|  | 4645 | T = getDecayedType(T); | 
|  | 4646 | return T.getUnqualifiedType(); | 
|  | 4647 | } | 
|  | 4648 |  | 
| Chris Lattner | a21ad80 | 2008-04-02 05:18:44 +0000 | [diff] [blame] | 4649 | /// getArrayDecayedType - Return the properly qualified result of decaying the | 
|  | 4650 | /// specified array type to a pointer.  This operation is non-trivial when | 
|  | 4651 | /// handling typedefs etc.  The canonical type of "T" must be an array type, | 
|  | 4652 | /// this returns a pointer to a properly qualified element of the array. | 
|  | 4653 | /// | 
|  | 4654 | /// See C99 6.7.5.3p7 and C99 6.3.2.1p3. | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 4655 | QualType ASTContext::getArrayDecayedType(QualType Ty) const { | 
| Chris Lattner | 7adf076 | 2008-08-04 07:31:14 +0000 | [diff] [blame] | 4656 | // Get the element type with 'getAsArrayType' so that we don't lose any | 
|  | 4657 | // typedefs in the element type of the array.  This also handles propagation | 
|  | 4658 | // of type qualifiers from the array type into the element type if present | 
|  | 4659 | // (C99 6.7.3p8). | 
|  | 4660 | const ArrayType *PrettyArrayType = getAsArrayType(Ty); | 
|  | 4661 | assert(PrettyArrayType && "Not an array type!"); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4662 |  | 
| Chris Lattner | 7adf076 | 2008-08-04 07:31:14 +0000 | [diff] [blame] | 4663 | QualType PtrTy = getPointerType(PrettyArrayType->getElementType()); | 
| Chris Lattner | a21ad80 | 2008-04-02 05:18:44 +0000 | [diff] [blame] | 4664 |  | 
|  | 4665 | // int x[restrict 4] ->  int *restrict | 
| John McCall | 8ccfcb5 | 2009-09-24 19:53:00 +0000 | [diff] [blame] | 4666 | return getQualifiedType(PtrTy, PrettyArrayType->getIndexTypeQualifiers()); | 
| Chris Lattner | a21ad80 | 2008-04-02 05:18:44 +0000 | [diff] [blame] | 4667 | } | 
|  | 4668 |  | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 4669 | QualType ASTContext::getBaseElementType(const ArrayType *array) const { | 
|  | 4670 | return getBaseElementType(array->getElementType()); | 
| Douglas Gregor | 79f83ed | 2009-07-23 23:49:00 +0000 | [diff] [blame] | 4671 | } | 
|  | 4672 |  | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 4673 | QualType ASTContext::getBaseElementType(QualType type) const { | 
|  | 4674 | Qualifiers qs; | 
|  | 4675 | while (true) { | 
|  | 4676 | SplitQualType split = type.getSplitDesugaredType(); | 
| John McCall | 18ce25e | 2012-02-08 00:46:36 +0000 | [diff] [blame] | 4677 | const ArrayType *array = split.Ty->getAsArrayTypeUnsafe(); | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 4678 | if (!array) break; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4679 |  | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 4680 | type = array->getElementType(); | 
| John McCall | 18ce25e | 2012-02-08 00:46:36 +0000 | [diff] [blame] | 4681 | qs.addConsistentQualifiers(split.Quals); | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 4682 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4683 |  | 
| John McCall | 33ddac0 | 2011-01-19 10:06:00 +0000 | [diff] [blame] | 4684 | return getQualifiedType(type, qs); | 
| Anders Carlsson | e0808df | 2008-12-21 03:44:36 +0000 | [diff] [blame] | 4685 | } | 
|  | 4686 |  | 
| Fariborz Jahanian | 6c9e5a2 | 2009-08-21 16:31:06 +0000 | [diff] [blame] | 4687 | /// getConstantArrayElementCount - Returns number of constant array elements. | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4688 | uint64_t | 
| Fariborz Jahanian | 6c9e5a2 | 2009-08-21 16:31:06 +0000 | [diff] [blame] | 4689 | ASTContext::getConstantArrayElementCount(const ConstantArrayType *CA)  const { | 
|  | 4690 | uint64_t ElementCount = 1; | 
|  | 4691 | do { | 
|  | 4692 | ElementCount *= CA->getSize().getZExtValue(); | 
| Richard Smith | 7808c6a | 2012-12-06 03:04:50 +0000 | [diff] [blame] | 4693 | CA = dyn_cast_or_null<ConstantArrayType>( | 
|  | 4694 | CA->getElementType()->getAsArrayTypeUnsafe()); | 
| Fariborz Jahanian | 6c9e5a2 | 2009-08-21 16:31:06 +0000 | [diff] [blame] | 4695 | } while (CA); | 
|  | 4696 | return ElementCount; | 
|  | 4697 | } | 
|  | 4698 |  | 
| Steve Naroff | 0af9120 | 2007-04-27 21:51:21 +0000 | [diff] [blame] | 4699 | /// getFloatingRank - Return a relative rank for floating point types. | 
|  | 4700 | /// This routine will assert if passed a built-in type that isn't a float. | 
| Chris Lattner | b90739d | 2008-04-06 23:38:49 +0000 | [diff] [blame] | 4701 | static FloatingRank getFloatingRank(QualType T) { | 
| John McCall | 9dd450b | 2009-09-21 23:43:11 +0000 | [diff] [blame] | 4702 | if (const ComplexType *CT = T->getAs<ComplexType>()) | 
| Chris Lattner | c639593 | 2007-06-22 20:56:16 +0000 | [diff] [blame] | 4703 | return getFloatingRank(CT->getElementType()); | 
| Chris Lattner | b90739d | 2008-04-06 23:38:49 +0000 | [diff] [blame] | 4704 |  | 
| John McCall | 9dd450b | 2009-09-21 23:43:11 +0000 | [diff] [blame] | 4705 | assert(T->getAs<BuiltinType>() && "getFloatingRank(): not a floating type"); | 
|  | 4706 | switch (T->getAs<BuiltinType>()->getKind()) { | 
| David Blaikie | 83d382b | 2011-09-23 05:06:16 +0000 | [diff] [blame] | 4707 | default: llvm_unreachable("getFloatingRank(): not a floating type"); | 
| Anton Korobeynikov | f0c267e | 2011-10-14 23:23:15 +0000 | [diff] [blame] | 4708 | case BuiltinType::Half:       return HalfRank; | 
| Chris Lattner | c639593 | 2007-06-22 20:56:16 +0000 | [diff] [blame] | 4709 | case BuiltinType::Float:      return FloatRank; | 
|  | 4710 | case BuiltinType::Double:     return DoubleRank; | 
|  | 4711 | case BuiltinType::LongDouble: return LongDoubleRank; | 
| Nemanja Ivanovic | bb1ea2d | 2016-05-09 08:52:33 +0000 | [diff] [blame] | 4712 | case BuiltinType::Float128:   return Float128Rank; | 
| Steve Naroff | e471889 | 2007-04-27 18:30:00 +0000 | [diff] [blame] | 4713 | } | 
|  | 4714 | } | 
|  | 4715 |  | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4716 | /// getFloatingTypeOfSizeWithinDomain - Returns a real floating | 
|  | 4717 | /// point or a complex type (based on typeDomain/typeSize). | 
| Steve Naroff | fc6ffa2 | 2007-08-27 01:41:48 +0000 | [diff] [blame] | 4718 | /// 'typeDomain' is a real floating point or complex type. | 
|  | 4719 | /// 'typeSize' is a real floating point or complex type. | 
| Chris Lattner | b9dfb03 | 2008-04-06 23:58:54 +0000 | [diff] [blame] | 4720 | QualType ASTContext::getFloatingTypeOfSizeWithinDomain(QualType Size, | 
|  | 4721 | QualType Domain) const { | 
|  | 4722 | FloatingRank EltRank = getFloatingRank(Size); | 
|  | 4723 | if (Domain->isComplexType()) { | 
|  | 4724 | switch (EltRank) { | 
| David Blaikie | f47fa30 | 2012-01-17 02:30:50 +0000 | [diff] [blame] | 4725 | case HalfRank: llvm_unreachable("Complex half is not supported"); | 
| Steve Naroff | 9091ef7 | 2007-08-27 01:27:54 +0000 | [diff] [blame] | 4726 | case FloatRank:      return FloatComplexTy; | 
|  | 4727 | case DoubleRank:     return DoubleComplexTy; | 
|  | 4728 | case LongDoubleRank: return LongDoubleComplexTy; | 
| Nemanja Ivanovic | bb1ea2d | 2016-05-09 08:52:33 +0000 | [diff] [blame] | 4729 | case Float128Rank:   return Float128ComplexTy; | 
| Steve Naroff | 9091ef7 | 2007-08-27 01:27:54 +0000 | [diff] [blame] | 4730 | } | 
| Steve Naroff | 0af9120 | 2007-04-27 21:51:21 +0000 | [diff] [blame] | 4731 | } | 
| Chris Lattner | b9dfb03 | 2008-04-06 23:58:54 +0000 | [diff] [blame] | 4732 |  | 
|  | 4733 | assert(Domain->isRealFloatingType() && "Unknown domain!"); | 
|  | 4734 | switch (EltRank) { | 
| Joey Gouly | dd7f456 | 2013-01-23 11:56:20 +0000 | [diff] [blame] | 4735 | case HalfRank:       return HalfTy; | 
| Chris Lattner | b9dfb03 | 2008-04-06 23:58:54 +0000 | [diff] [blame] | 4736 | case FloatRank:      return FloatTy; | 
|  | 4737 | case DoubleRank:     return DoubleTy; | 
|  | 4738 | case LongDoubleRank: return LongDoubleTy; | 
| Nemanja Ivanovic | bb1ea2d | 2016-05-09 08:52:33 +0000 | [diff] [blame] | 4739 | case Float128Rank:   return Float128Ty; | 
| Steve Naroff | 9091ef7 | 2007-08-27 01:27:54 +0000 | [diff] [blame] | 4740 | } | 
| David Blaikie | f47fa30 | 2012-01-17 02:30:50 +0000 | [diff] [blame] | 4741 | llvm_unreachable("getFloatingRank(): illegal value for rank"); | 
| Steve Naroff | e471889 | 2007-04-27 18:30:00 +0000 | [diff] [blame] | 4742 | } | 
|  | 4743 |  | 
| Chris Lattner | d4bacd6 | 2008-04-06 23:55:33 +0000 | [diff] [blame] | 4744 | /// getFloatingTypeOrder - Compare the rank of the two specified floating | 
|  | 4745 | /// point types, ignoring the domain of the type (i.e. 'double' == | 
|  | 4746 | /// '_Complex double').  If LHS > RHS, return 1.  If LHS == RHS, return 0. If | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4747 | /// LHS < RHS, return -1. | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 4748 | int ASTContext::getFloatingTypeOrder(QualType LHS, QualType RHS) const { | 
| Chris Lattner | b90739d | 2008-04-06 23:38:49 +0000 | [diff] [blame] | 4749 | FloatingRank LHSR = getFloatingRank(LHS); | 
|  | 4750 | FloatingRank RHSR = getFloatingRank(RHS); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4751 |  | 
| Chris Lattner | b90739d | 2008-04-06 23:38:49 +0000 | [diff] [blame] | 4752 | if (LHSR == RHSR) | 
| Steve Naroff | 7af82d4 | 2007-08-27 15:30:22 +0000 | [diff] [blame] | 4753 | return 0; | 
| Chris Lattner | b90739d | 2008-04-06 23:38:49 +0000 | [diff] [blame] | 4754 | if (LHSR > RHSR) | 
| Steve Naroff | 7af82d4 | 2007-08-27 15:30:22 +0000 | [diff] [blame] | 4755 | return 1; | 
|  | 4756 | return -1; | 
| Steve Naroff | e471889 | 2007-04-27 18:30:00 +0000 | [diff] [blame] | 4757 | } | 
|  | 4758 |  | 
| Chris Lattner | 76a00cf | 2008-04-06 22:59:24 +0000 | [diff] [blame] | 4759 | /// getIntegerRank - Return an integer conversion rank (C99 6.3.1.1p1). This | 
|  | 4760 | /// routine will assert if passed a built-in type that isn't an integer or enum, | 
|  | 4761 | /// or if it is not canonicalized. | 
| John McCall | 424cec9 | 2011-01-19 06:33:43 +0000 | [diff] [blame] | 4762 | unsigned ASTContext::getIntegerRank(const Type *T) const { | 
| John McCall | b692a09 | 2009-10-22 20:10:53 +0000 | [diff] [blame] | 4763 | assert(T->isCanonicalUnqualified() && "T should be canonicalized"); | 
| Alisdair Meredith | a9ad47d | 2009-07-14 06:30:34 +0000 | [diff] [blame] | 4764 |  | 
| Chris Lattner | 76a00cf | 2008-04-06 22:59:24 +0000 | [diff] [blame] | 4765 | switch (cast<BuiltinType>(T)->getKind()) { | 
| David Blaikie | 83d382b | 2011-09-23 05:06:16 +0000 | [diff] [blame] | 4766 | default: llvm_unreachable("getIntegerRank(): not a built-in integer"); | 
| Chris Lattner | d4bacd6 | 2008-04-06 23:55:33 +0000 | [diff] [blame] | 4767 | case BuiltinType::Bool: | 
| Eli Friedman | 1efaaea | 2009-02-13 02:31:07 +0000 | [diff] [blame] | 4768 | return 1 + (getIntWidth(BoolTy) << 3); | 
| Chris Lattner | d4bacd6 | 2008-04-06 23:55:33 +0000 | [diff] [blame] | 4769 | case BuiltinType::Char_S: | 
|  | 4770 | case BuiltinType::Char_U: | 
|  | 4771 | case BuiltinType::SChar: | 
|  | 4772 | case BuiltinType::UChar: | 
| Eli Friedman | 1efaaea | 2009-02-13 02:31:07 +0000 | [diff] [blame] | 4773 | return 2 + (getIntWidth(CharTy) << 3); | 
| Chris Lattner | d4bacd6 | 2008-04-06 23:55:33 +0000 | [diff] [blame] | 4774 | case BuiltinType::Short: | 
|  | 4775 | case BuiltinType::UShort: | 
| Eli Friedman | 1efaaea | 2009-02-13 02:31:07 +0000 | [diff] [blame] | 4776 | return 3 + (getIntWidth(ShortTy) << 3); | 
| Chris Lattner | d4bacd6 | 2008-04-06 23:55:33 +0000 | [diff] [blame] | 4777 | case BuiltinType::Int: | 
|  | 4778 | case BuiltinType::UInt: | 
| Eli Friedman | 1efaaea | 2009-02-13 02:31:07 +0000 | [diff] [blame] | 4779 | return 4 + (getIntWidth(IntTy) << 3); | 
| Chris Lattner | d4bacd6 | 2008-04-06 23:55:33 +0000 | [diff] [blame] | 4780 | case BuiltinType::Long: | 
|  | 4781 | case BuiltinType::ULong: | 
| Eli Friedman | 1efaaea | 2009-02-13 02:31:07 +0000 | [diff] [blame] | 4782 | return 5 + (getIntWidth(LongTy) << 3); | 
| Chris Lattner | d4bacd6 | 2008-04-06 23:55:33 +0000 | [diff] [blame] | 4783 | case BuiltinType::LongLong: | 
|  | 4784 | case BuiltinType::ULongLong: | 
| Eli Friedman | 1efaaea | 2009-02-13 02:31:07 +0000 | [diff] [blame] | 4785 | return 6 + (getIntWidth(LongLongTy) << 3); | 
| Chris Lattner | f122cef | 2009-04-30 02:43:43 +0000 | [diff] [blame] | 4786 | case BuiltinType::Int128: | 
|  | 4787 | case BuiltinType::UInt128: | 
|  | 4788 | return 7 + (getIntWidth(Int128Ty) << 3); | 
| Chris Lattner | 76a00cf | 2008-04-06 22:59:24 +0000 | [diff] [blame] | 4789 | } | 
|  | 4790 | } | 
|  | 4791 |  | 
| Eli Friedman | 629ffb9 | 2009-08-20 04:21:42 +0000 | [diff] [blame] | 4792 | /// \brief Whether this is a promotable bitfield reference according | 
|  | 4793 | /// to C99 6.3.1.1p2, bullet 2 (and GCC extensions). | 
|  | 4794 | /// | 
|  | 4795 | /// \returns the type this bit-field will promote to, or NULL if no | 
|  | 4796 | /// promotion occurs. | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 4797 | QualType ASTContext::isPromotableBitField(Expr *E) const { | 
| Douglas Gregor | e05d3cb | 2010-05-24 20:13:53 +0000 | [diff] [blame] | 4798 | if (E->isTypeDependent() || E->isValueDependent()) | 
|  | 4799 | return QualType(); | 
| Richard Smith | 5b57167 | 2014-09-24 23:55:00 +0000 | [diff] [blame] | 4800 |  | 
|  | 4801 | // FIXME: We should not do this unless E->refersToBitField() is true. This | 
|  | 4802 | // matters in C where getSourceBitField() will find bit-fields for various | 
|  | 4803 | // cases where the source expression is not a bit-field designator. | 
|  | 4804 |  | 
| John McCall | d25db7e | 2013-05-06 21:39:12 +0000 | [diff] [blame] | 4805 | FieldDecl *Field = E->getSourceBitField(); // FIXME: conditional bit-fields? | 
| Eli Friedman | 629ffb9 | 2009-08-20 04:21:42 +0000 | [diff] [blame] | 4806 | if (!Field) | 
|  | 4807 | return QualType(); | 
|  | 4808 |  | 
|  | 4809 | QualType FT = Field->getType(); | 
|  | 4810 |  | 
| Richard Smith | caf3390 | 2011-10-10 18:28:20 +0000 | [diff] [blame] | 4811 | uint64_t BitWidth = Field->getBitWidthValue(*this); | 
| Eli Friedman | 629ffb9 | 2009-08-20 04:21:42 +0000 | [diff] [blame] | 4812 | uint64_t IntSize = getTypeSize(IntTy); | 
| Richard Smith | 5b57167 | 2014-09-24 23:55:00 +0000 | [diff] [blame] | 4813 | // C++ [conv.prom]p5: | 
|  | 4814 | //   A prvalue for an integral bit-field can be converted to a prvalue of type | 
|  | 4815 | //   int if int can represent all the values of the bit-field; otherwise, it | 
|  | 4816 | //   can be converted to unsigned int if unsigned int can represent all the | 
|  | 4817 | //   values of the bit-field. If the bit-field is larger yet, no integral | 
|  | 4818 | //   promotion applies to it. | 
|  | 4819 | // C11 6.3.1.1/2: | 
|  | 4820 | //   [For a bit-field of type _Bool, int, signed int, or unsigned int:] | 
|  | 4821 | //   If an int can represent all values of the original type (as restricted by | 
|  | 4822 | //   the width, for a bit-field), the value is converted to an int; otherwise, | 
|  | 4823 | //   it is converted to an unsigned int. | 
|  | 4824 | // | 
|  | 4825 | // FIXME: C does not permit promotion of a 'long : 3' bitfield to int. | 
|  | 4826 | //        We perform that promotion here to match GCC and C++. | 
| Eli Friedman | 629ffb9 | 2009-08-20 04:21:42 +0000 | [diff] [blame] | 4827 | if (BitWidth < IntSize) | 
|  | 4828 | return IntTy; | 
|  | 4829 |  | 
|  | 4830 | if (BitWidth == IntSize) | 
|  | 4831 | return FT->isSignedIntegerType() ? IntTy : UnsignedIntTy; | 
|  | 4832 |  | 
|  | 4833 | // Types bigger than int are not subject to promotions, and therefore act | 
| Richard Smith | 5b57167 | 2014-09-24 23:55:00 +0000 | [diff] [blame] | 4834 | // like the base type. GCC has some weird bugs in this area that we | 
|  | 4835 | // deliberately do not follow (GCC follows a pre-standard resolution to | 
|  | 4836 | // C's DR315 which treats bit-width as being part of the type, and this leaks | 
|  | 4837 | // into their semantics in some cases). | 
| Eli Friedman | 629ffb9 | 2009-08-20 04:21:42 +0000 | [diff] [blame] | 4838 | return QualType(); | 
|  | 4839 | } | 
|  | 4840 |  | 
| Eli Friedman | 5ae98ee | 2009-08-19 07:44:53 +0000 | [diff] [blame] | 4841 | /// getPromotedIntegerType - Returns the type that Promotable will | 
|  | 4842 | /// promote to: C99 6.3.1.1p2, assuming that Promotable is a promotable | 
|  | 4843 | /// integer type. | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 4844 | QualType ASTContext::getPromotedIntegerType(QualType Promotable) const { | 
| Eli Friedman | 5ae98ee | 2009-08-19 07:44:53 +0000 | [diff] [blame] | 4845 | assert(!Promotable.isNull()); | 
|  | 4846 | assert(Promotable->isPromotableIntegerType()); | 
| John McCall | 5677499 | 2009-12-09 09:09:27 +0000 | [diff] [blame] | 4847 | if (const EnumType *ET = Promotable->getAs<EnumType>()) | 
|  | 4848 | return ET->getDecl()->getPromotionType(); | 
| Eli Friedman | 3f37c1c | 2011-10-26 07:22:48 +0000 | [diff] [blame] | 4849 |  | 
|  | 4850 | if (const BuiltinType *BT = Promotable->getAs<BuiltinType>()) { | 
|  | 4851 | // C++ [conv.prom]: A prvalue of type char16_t, char32_t, or wchar_t | 
|  | 4852 | // (3.9.1) can be converted to a prvalue of the first of the following | 
|  | 4853 | // types that can represent all the values of its underlying type: | 
|  | 4854 | // int, unsigned int, long int, unsigned long int, long long int, or | 
|  | 4855 | // unsigned long long int [...] | 
|  | 4856 | // FIXME: Is there some better way to compute this? | 
|  | 4857 | if (BT->getKind() == BuiltinType::WChar_S || | 
|  | 4858 | BT->getKind() == BuiltinType::WChar_U || | 
|  | 4859 | BT->getKind() == BuiltinType::Char16 || | 
|  | 4860 | BT->getKind() == BuiltinType::Char32) { | 
|  | 4861 | bool FromIsSigned = BT->getKind() == BuiltinType::WChar_S; | 
|  | 4862 | uint64_t FromSize = getTypeSize(BT); | 
|  | 4863 | QualType PromoteTypes[] = { IntTy, UnsignedIntTy, LongTy, UnsignedLongTy, | 
|  | 4864 | LongLongTy, UnsignedLongLongTy }; | 
|  | 4865 | for (size_t Idx = 0; Idx < llvm::array_lengthof(PromoteTypes); ++Idx) { | 
|  | 4866 | uint64_t ToSize = getTypeSize(PromoteTypes[Idx]); | 
|  | 4867 | if (FromSize < ToSize || | 
|  | 4868 | (FromSize == ToSize && | 
|  | 4869 | FromIsSigned == PromoteTypes[Idx]->isSignedIntegerType())) | 
|  | 4870 | return PromoteTypes[Idx]; | 
|  | 4871 | } | 
|  | 4872 | llvm_unreachable("char type should fit into long long"); | 
|  | 4873 | } | 
|  | 4874 | } | 
|  | 4875 |  | 
|  | 4876 | // At this point, we should have a signed or unsigned integer type. | 
| Eli Friedman | 5ae98ee | 2009-08-19 07:44:53 +0000 | [diff] [blame] | 4877 | if (Promotable->isSignedIntegerType()) | 
|  | 4878 | return IntTy; | 
| Eli Friedman | 6745c3b | 2012-11-15 01:21:59 +0000 | [diff] [blame] | 4879 | uint64_t PromotableSize = getIntWidth(Promotable); | 
|  | 4880 | uint64_t IntSize = getIntWidth(IntTy); | 
| Eli Friedman | 5ae98ee | 2009-08-19 07:44:53 +0000 | [diff] [blame] | 4881 | assert(Promotable->isUnsignedIntegerType() && PromotableSize <= IntSize); | 
|  | 4882 | return (PromotableSize != IntSize) ? IntTy : UnsignedIntTy; | 
|  | 4883 | } | 
|  | 4884 |  | 
| Argyrios Kyrtzidis | 7451d1c | 2011-07-01 22:22:50 +0000 | [diff] [blame] | 4885 | /// \brief Recurses in pointer/array types until it finds an objc retainable | 
|  | 4886 | /// type and returns its ownership. | 
|  | 4887 | Qualifiers::ObjCLifetime ASTContext::getInnerObjCOwnership(QualType T) const { | 
|  | 4888 | while (!T.isNull()) { | 
|  | 4889 | if (T.getObjCLifetime() != Qualifiers::OCL_None) | 
|  | 4890 | return T.getObjCLifetime(); | 
|  | 4891 | if (T->isArrayType()) | 
|  | 4892 | T = getBaseElementType(T); | 
|  | 4893 | else if (const PointerType *PT = T->getAs<PointerType>()) | 
|  | 4894 | T = PT->getPointeeType(); | 
|  | 4895 | else if (const ReferenceType *RT = T->getAs<ReferenceType>()) | 
| Argyrios Kyrtzidis | 8e25253 | 2011-07-01 23:01:46 +0000 | [diff] [blame] | 4896 | T = RT->getPointeeType(); | 
| Argyrios Kyrtzidis | 7451d1c | 2011-07-01 22:22:50 +0000 | [diff] [blame] | 4897 | else | 
|  | 4898 | break; | 
|  | 4899 | } | 
|  | 4900 |  | 
|  | 4901 | return Qualifiers::OCL_None; | 
|  | 4902 | } | 
|  | 4903 |  | 
| Ted Kremenek | e65ab9e | 2013-10-10 00:54:01 +0000 | [diff] [blame] | 4904 | static const Type *getIntegerTypeForEnum(const EnumType *ET) { | 
|  | 4905 | // Incomplete enum types are not treated as integer types. | 
|  | 4906 | // FIXME: In C++, enum types are never integer types. | 
|  | 4907 | if (ET->getDecl()->isComplete() && !ET->getDecl()->isScoped()) | 
|  | 4908 | return ET->getDecl()->getIntegerType().getTypePtr(); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 4909 | return nullptr; | 
| Ted Kremenek | e65ab9e | 2013-10-10 00:54:01 +0000 | [diff] [blame] | 4910 | } | 
|  | 4911 |  | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4912 | /// getIntegerTypeOrder - Returns the highest ranked integer type: | 
| Chris Lattner | d4bacd6 | 2008-04-06 23:55:33 +0000 | [diff] [blame] | 4913 | /// C99 6.3.1.8p1.  If LHS > RHS, return 1.  If LHS == RHS, return 0. If | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4914 | /// LHS < RHS, return -1. | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 4915 | int ASTContext::getIntegerTypeOrder(QualType LHS, QualType RHS) const { | 
| John McCall | 424cec9 | 2011-01-19 06:33:43 +0000 | [diff] [blame] | 4916 | const Type *LHSC = getCanonicalType(LHS).getTypePtr(); | 
|  | 4917 | const Type *RHSC = getCanonicalType(RHS).getTypePtr(); | 
| Ted Kremenek | e65ab9e | 2013-10-10 00:54:01 +0000 | [diff] [blame] | 4918 |  | 
|  | 4919 | // Unwrap enums to their underlying type. | 
|  | 4920 | if (const EnumType *ET = dyn_cast<EnumType>(LHSC)) | 
|  | 4921 | LHSC = getIntegerTypeForEnum(ET); | 
|  | 4922 | if (const EnumType *ET = dyn_cast<EnumType>(RHSC)) | 
|  | 4923 | RHSC = getIntegerTypeForEnum(ET); | 
|  | 4924 |  | 
| Chris Lattner | d4bacd6 | 2008-04-06 23:55:33 +0000 | [diff] [blame] | 4925 | if (LHSC == RHSC) return 0; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4926 |  | 
| Chris Lattner | 76a00cf | 2008-04-06 22:59:24 +0000 | [diff] [blame] | 4927 | bool LHSUnsigned = LHSC->isUnsignedIntegerType(); | 
|  | 4928 | bool RHSUnsigned = RHSC->isUnsignedIntegerType(); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4929 |  | 
| Chris Lattner | d4bacd6 | 2008-04-06 23:55:33 +0000 | [diff] [blame] | 4930 | unsigned LHSRank = getIntegerRank(LHSC); | 
|  | 4931 | unsigned RHSRank = getIntegerRank(RHSC); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4932 |  | 
| Chris Lattner | d4bacd6 | 2008-04-06 23:55:33 +0000 | [diff] [blame] | 4933 | if (LHSUnsigned == RHSUnsigned) {  // Both signed or both unsigned. | 
|  | 4934 | if (LHSRank == RHSRank) return 0; | 
|  | 4935 | return LHSRank > RHSRank ? 1 : -1; | 
|  | 4936 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4937 |  | 
| Chris Lattner | d4bacd6 | 2008-04-06 23:55:33 +0000 | [diff] [blame] | 4938 | // Otherwise, the LHS is signed and the RHS is unsigned or visa versa. | 
|  | 4939 | if (LHSUnsigned) { | 
|  | 4940 | // If the unsigned [LHS] type is larger, return it. | 
|  | 4941 | if (LHSRank >= RHSRank) | 
|  | 4942 | return 1; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4943 |  | 
| Chris Lattner | d4bacd6 | 2008-04-06 23:55:33 +0000 | [diff] [blame] | 4944 | // If the signed type can represent all values of the unsigned type, it | 
|  | 4945 | // wins.  Because we are dealing with 2's complement and types that are | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4946 | // powers of two larger than each other, this is always safe. | 
| Chris Lattner | d4bacd6 | 2008-04-06 23:55:33 +0000 | [diff] [blame] | 4947 | return -1; | 
|  | 4948 | } | 
| Chris Lattner | 76a00cf | 2008-04-06 22:59:24 +0000 | [diff] [blame] | 4949 |  | 
| Chris Lattner | d4bacd6 | 2008-04-06 23:55:33 +0000 | [diff] [blame] | 4950 | // If the unsigned [RHS] type is larger, return it. | 
|  | 4951 | if (RHSRank >= LHSRank) | 
|  | 4952 | return -1; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4953 |  | 
| Chris Lattner | d4bacd6 | 2008-04-06 23:55:33 +0000 | [diff] [blame] | 4954 | // If the signed type can represent all values of the unsigned type, it | 
|  | 4955 | // wins.  Because we are dealing with 2's complement and types that are | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4956 | // powers of two larger than each other, this is always safe. | 
| Chris Lattner | d4bacd6 | 2008-04-06 23:55:33 +0000 | [diff] [blame] | 4957 | return 1; | 
| Steve Naroff | e471889 | 2007-04-27 18:30:00 +0000 | [diff] [blame] | 4958 | } | 
| Anders Carlsson | 98f0790 | 2007-08-17 05:31:46 +0000 | [diff] [blame] | 4959 |  | 
| Ben Langmuir | f541674 | 2016-02-04 00:55:24 +0000 | [diff] [blame] | 4960 | TypedefDecl *ASTContext::getCFConstantStringDecl() const { | 
| Anders Carlsson | 98f0790 | 2007-08-17 05:31:46 +0000 | [diff] [blame] | 4961 | if (!CFConstantStringTypeDecl) { | 
| Ben Langmuir | f541674 | 2016-02-04 00:55:24 +0000 | [diff] [blame] | 4962 | assert(!CFConstantStringTagDecl && | 
|  | 4963 | "tag and typedef should be initialized together"); | 
|  | 4964 | CFConstantStringTagDecl = buildImplicitRecord("__NSConstantString_tag"); | 
|  | 4965 | CFConstantStringTagDecl->startDefinition(); | 
| Anders Carlsson | 6d41727 | 2009-11-14 21:45:58 +0000 | [diff] [blame] | 4966 |  | 
| Anders Carlsson | 9c1011c | 2007-11-19 00:25:30 +0000 | [diff] [blame] | 4967 | QualType FieldTypes[4]; | 
| Ben Langmuir | 4791a80 | 2016-02-25 16:36:26 +0000 | [diff] [blame] | 4968 | const char *FieldNames[4]; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4969 |  | 
| Anders Carlsson | 98f0790 | 2007-08-17 05:31:46 +0000 | [diff] [blame] | 4970 | // const int *isa; | 
| John McCall | 8ccfcb5 | 2009-09-24 19:53:00 +0000 | [diff] [blame] | 4971 | FieldTypes[0] = getPointerType(IntTy.withConst()); | 
| Ben Langmuir | 4791a80 | 2016-02-25 16:36:26 +0000 | [diff] [blame] | 4972 | FieldNames[0] = "isa"; | 
| Anders Carlsson | 9c1011c | 2007-11-19 00:25:30 +0000 | [diff] [blame] | 4973 | // int flags; | 
|  | 4974 | FieldTypes[1] = IntTy; | 
| Ben Langmuir | 4791a80 | 2016-02-25 16:36:26 +0000 | [diff] [blame] | 4975 | FieldNames[1] = "flags"; | 
| Anders Carlsson | 98f0790 | 2007-08-17 05:31:46 +0000 | [diff] [blame] | 4976 | // const char *str; | 
| John McCall | 8ccfcb5 | 2009-09-24 19:53:00 +0000 | [diff] [blame] | 4977 | FieldTypes[2] = getPointerType(CharTy.withConst()); | 
| Ben Langmuir | 4791a80 | 2016-02-25 16:36:26 +0000 | [diff] [blame] | 4978 | FieldNames[2] = "str"; | 
| Anders Carlsson | 98f0790 | 2007-08-17 05:31:46 +0000 | [diff] [blame] | 4979 | // long length; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4980 | FieldTypes[3] = LongTy; | 
| Ben Langmuir | 4791a80 | 2016-02-25 16:36:26 +0000 | [diff] [blame] | 4981 | FieldNames[3] = "length"; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 4982 |  | 
| Anders Carlsson | 98f0790 | 2007-08-17 05:31:46 +0000 | [diff] [blame] | 4983 | // Create fields | 
| Douglas Gregor | 91f8421 | 2008-12-11 16:49:14 +0000 | [diff] [blame] | 4984 | for (unsigned i = 0; i < 4; ++i) { | 
| Ben Langmuir | f541674 | 2016-02-04 00:55:24 +0000 | [diff] [blame] | 4985 | FieldDecl *Field = FieldDecl::Create(*this, CFConstantStringTagDecl, | 
| Abramo Bagnara | dff1930 | 2011-03-08 08:55:46 +0000 | [diff] [blame] | 4986 | SourceLocation(), | 
| Ben Langmuir | 4791a80 | 2016-02-25 16:36:26 +0000 | [diff] [blame] | 4987 | SourceLocation(), | 
|  | 4988 | &Idents.get(FieldNames[i]), | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 4989 | FieldTypes[i], /*TInfo=*/nullptr, | 
|  | 4990 | /*BitWidth=*/nullptr, | 
| Richard Smith | 938f40b | 2011-06-11 17:19:42 +0000 | [diff] [blame] | 4991 | /*Mutable=*/false, | 
| Richard Smith | 2b01318 | 2012-06-10 03:12:00 +0000 | [diff] [blame] | 4992 | ICIS_NoInit); | 
| John McCall | 4d4dcc8 | 2010-04-30 21:35:41 +0000 | [diff] [blame] | 4993 | Field->setAccess(AS_public); | 
| Ben Langmuir | f541674 | 2016-02-04 00:55:24 +0000 | [diff] [blame] | 4994 | CFConstantStringTagDecl->addDecl(Field); | 
| Douglas Gregor | 91f8421 | 2008-12-11 16:49:14 +0000 | [diff] [blame] | 4995 | } | 
|  | 4996 |  | 
| Ben Langmuir | f541674 | 2016-02-04 00:55:24 +0000 | [diff] [blame] | 4997 | CFConstantStringTagDecl->completeDefinition(); | 
|  | 4998 | // This type is designed to be compatible with NSConstantString, but cannot | 
|  | 4999 | // use the same name, since NSConstantString is an interface. | 
|  | 5000 | auto tagType = getTagDeclType(CFConstantStringTagDecl); | 
|  | 5001 | CFConstantStringTypeDecl = | 
|  | 5002 | buildImplicitTypedef(tagType, "__NSConstantString"); | 
| Anders Carlsson | 98f0790 | 2007-08-17 05:31:46 +0000 | [diff] [blame] | 5003 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5004 |  | 
| Quentin Colombet | 043406b | 2016-02-03 22:41:00 +0000 | [diff] [blame] | 5005 | return CFConstantStringTypeDecl; | 
|  | 5006 | } | 
|  | 5007 |  | 
| Ben Langmuir | f541674 | 2016-02-04 00:55:24 +0000 | [diff] [blame] | 5008 | RecordDecl *ASTContext::getCFConstantStringTagDecl() const { | 
|  | 5009 | if (!CFConstantStringTagDecl) | 
|  | 5010 | getCFConstantStringDecl(); // Build the tag and the typedef. | 
|  | 5011 | return CFConstantStringTagDecl; | 
|  | 5012 | } | 
|  | 5013 |  | 
| Quentin Colombet | 043406b | 2016-02-03 22:41:00 +0000 | [diff] [blame] | 5014 | // getCFConstantStringType - Return the type used for constant CFStrings. | 
|  | 5015 | QualType ASTContext::getCFConstantStringType() const { | 
| Ben Langmuir | f541674 | 2016-02-04 00:55:24 +0000 | [diff] [blame] | 5016 | return getTypedefType(getCFConstantStringDecl()); | 
| Gabor Greif | 412af03 | 2007-09-11 15:32:40 +0000 | [diff] [blame] | 5017 | } | 
| Anders Carlsson | 87c149b | 2007-10-11 01:00:40 +0000 | [diff] [blame] | 5018 |  | 
| Fariborz Jahanian | cb6c867 | 2013-01-04 18:45:40 +0000 | [diff] [blame] | 5019 | QualType ASTContext::getObjCSuperType() const { | 
|  | 5020 | if (ObjCSuperType.isNull()) { | 
| Alp Toker | 2dea15b | 2013-12-17 01:22:38 +0000 | [diff] [blame] | 5021 | RecordDecl *ObjCSuperTypeDecl = buildImplicitRecord("objc_super"); | 
| Fariborz Jahanian | cb6c867 | 2013-01-04 18:45:40 +0000 | [diff] [blame] | 5022 | TUDecl->addDecl(ObjCSuperTypeDecl); | 
|  | 5023 | ObjCSuperType = getTagDeclType(ObjCSuperTypeDecl); | 
|  | 5024 | } | 
|  | 5025 | return ObjCSuperType; | 
|  | 5026 | } | 
|  | 5027 |  | 
| Douglas Gregor | 512b077 | 2009-04-23 22:29:11 +0000 | [diff] [blame] | 5028 | void ASTContext::setCFConstantStringType(QualType T) { | 
| Ben Langmuir | f541674 | 2016-02-04 00:55:24 +0000 | [diff] [blame] | 5029 | const TypedefType *TD = T->getAs<TypedefType>(); | 
|  | 5030 | assert(TD && "Invalid CFConstantStringType"); | 
|  | 5031 | CFConstantStringTypeDecl = cast<TypedefDecl>(TD->getDecl()); | 
|  | 5032 | auto TagType = | 
|  | 5033 | CFConstantStringTypeDecl->getUnderlyingType()->getAs<RecordType>(); | 
|  | 5034 | assert(TagType && "Invalid CFConstantStringType"); | 
|  | 5035 | CFConstantStringTagDecl = TagType->getDecl(); | 
| Douglas Gregor | 512b077 | 2009-04-23 22:29:11 +0000 | [diff] [blame] | 5036 | } | 
|  | 5037 |  | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 5038 | QualType ASTContext::getBlockDescriptorType() const { | 
| Mike Stump | d015328 | 2009-10-20 02:12:22 +0000 | [diff] [blame] | 5039 | if (BlockDescriptorType) | 
|  | 5040 | return getTagDeclType(BlockDescriptorType); | 
|  | 5041 |  | 
| Alp Toker | 2dea15b | 2013-12-17 01:22:38 +0000 | [diff] [blame] | 5042 | RecordDecl *RD; | 
| Mike Stump | d015328 | 2009-10-20 02:12:22 +0000 | [diff] [blame] | 5043 | // FIXME: Needs the FlagAppleBlock bit. | 
| Alp Toker | 2dea15b | 2013-12-17 01:22:38 +0000 | [diff] [blame] | 5044 | RD = buildImplicitRecord("__block_descriptor"); | 
|  | 5045 | RD->startDefinition(); | 
|  | 5046 |  | 
| Mike Stump | d015328 | 2009-10-20 02:12:22 +0000 | [diff] [blame] | 5047 | QualType FieldTypes[] = { | 
|  | 5048 | UnsignedLongTy, | 
|  | 5049 | UnsignedLongTy, | 
|  | 5050 | }; | 
|  | 5051 |  | 
| Craig Topper | d6d31ac | 2013-07-15 08:24:27 +0000 | [diff] [blame] | 5052 | static const char *const FieldNames[] = { | 
| Mike Stump | d015328 | 2009-10-20 02:12:22 +0000 | [diff] [blame] | 5053 | "reserved", | 
| Mike Stump | e1b19ba | 2009-10-22 00:49:09 +0000 | [diff] [blame] | 5054 | "Size" | 
| Mike Stump | d015328 | 2009-10-20 02:12:22 +0000 | [diff] [blame] | 5055 | }; | 
|  | 5056 |  | 
|  | 5057 | for (size_t i = 0; i < 2; ++i) { | 
| Alp Toker | 2dea15b | 2013-12-17 01:22:38 +0000 | [diff] [blame] | 5058 | FieldDecl *Field = FieldDecl::Create( | 
|  | 5059 | *this, RD, SourceLocation(), SourceLocation(), | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 5060 | &Idents.get(FieldNames[i]), FieldTypes[i], /*TInfo=*/nullptr, | 
|  | 5061 | /*BitWidth=*/nullptr, /*Mutable=*/false, ICIS_NoInit); | 
| John McCall | 4d4dcc8 | 2010-04-30 21:35:41 +0000 | [diff] [blame] | 5062 | Field->setAccess(AS_public); | 
| Alp Toker | 2dea15b | 2013-12-17 01:22:38 +0000 | [diff] [blame] | 5063 | RD->addDecl(Field); | 
| Mike Stump | d015328 | 2009-10-20 02:12:22 +0000 | [diff] [blame] | 5064 | } | 
|  | 5065 |  | 
| Alp Toker | 2dea15b | 2013-12-17 01:22:38 +0000 | [diff] [blame] | 5066 | RD->completeDefinition(); | 
| Mike Stump | d015328 | 2009-10-20 02:12:22 +0000 | [diff] [blame] | 5067 |  | 
| Alp Toker | 2dea15b | 2013-12-17 01:22:38 +0000 | [diff] [blame] | 5068 | BlockDescriptorType = RD; | 
| Mike Stump | d015328 | 2009-10-20 02:12:22 +0000 | [diff] [blame] | 5069 |  | 
|  | 5070 | return getTagDeclType(BlockDescriptorType); | 
|  | 5071 | } | 
|  | 5072 |  | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 5073 | QualType ASTContext::getBlockDescriptorExtendedType() const { | 
| Mike Stump | e1b19ba | 2009-10-22 00:49:09 +0000 | [diff] [blame] | 5074 | if (BlockDescriptorExtendedType) | 
|  | 5075 | return getTagDeclType(BlockDescriptorExtendedType); | 
|  | 5076 |  | 
| Alp Toker | 2dea15b | 2013-12-17 01:22:38 +0000 | [diff] [blame] | 5077 | RecordDecl *RD; | 
| Mike Stump | e1b19ba | 2009-10-22 00:49:09 +0000 | [diff] [blame] | 5078 | // FIXME: Needs the FlagAppleBlock bit. | 
| Alp Toker | 2dea15b | 2013-12-17 01:22:38 +0000 | [diff] [blame] | 5079 | RD = buildImplicitRecord("__block_descriptor_withcopydispose"); | 
|  | 5080 | RD->startDefinition(); | 
|  | 5081 |  | 
| Mike Stump | e1b19ba | 2009-10-22 00:49:09 +0000 | [diff] [blame] | 5082 | QualType FieldTypes[] = { | 
|  | 5083 | UnsignedLongTy, | 
|  | 5084 | UnsignedLongTy, | 
|  | 5085 | getPointerType(VoidPtrTy), | 
|  | 5086 | getPointerType(VoidPtrTy) | 
|  | 5087 | }; | 
|  | 5088 |  | 
| Craig Topper | d6d31ac | 2013-07-15 08:24:27 +0000 | [diff] [blame] | 5089 | static const char *const FieldNames[] = { | 
| Mike Stump | e1b19ba | 2009-10-22 00:49:09 +0000 | [diff] [blame] | 5090 | "reserved", | 
|  | 5091 | "Size", | 
|  | 5092 | "CopyFuncPtr", | 
|  | 5093 | "DestroyFuncPtr" | 
|  | 5094 | }; | 
|  | 5095 |  | 
|  | 5096 | for (size_t i = 0; i < 4; ++i) { | 
| Alp Toker | 2dea15b | 2013-12-17 01:22:38 +0000 | [diff] [blame] | 5097 | FieldDecl *Field = FieldDecl::Create( | 
|  | 5098 | *this, RD, SourceLocation(), SourceLocation(), | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 5099 | &Idents.get(FieldNames[i]), FieldTypes[i], /*TInfo=*/nullptr, | 
|  | 5100 | /*BitWidth=*/nullptr, | 
| Alp Toker | 2dea15b | 2013-12-17 01:22:38 +0000 | [diff] [blame] | 5101 | /*Mutable=*/false, ICIS_NoInit); | 
| John McCall | 4d4dcc8 | 2010-04-30 21:35:41 +0000 | [diff] [blame] | 5102 | Field->setAccess(AS_public); | 
| Alp Toker | 2dea15b | 2013-12-17 01:22:38 +0000 | [diff] [blame] | 5103 | RD->addDecl(Field); | 
| Mike Stump | e1b19ba | 2009-10-22 00:49:09 +0000 | [diff] [blame] | 5104 | } | 
|  | 5105 |  | 
| Alp Toker | 2dea15b | 2013-12-17 01:22:38 +0000 | [diff] [blame] | 5106 | RD->completeDefinition(); | 
| Mike Stump | e1b19ba | 2009-10-22 00:49:09 +0000 | [diff] [blame] | 5107 |  | 
| Alp Toker | 2dea15b | 2013-12-17 01:22:38 +0000 | [diff] [blame] | 5108 | BlockDescriptorExtendedType = RD; | 
| Mike Stump | e1b19ba | 2009-10-22 00:49:09 +0000 | [diff] [blame] | 5109 | return getTagDeclType(BlockDescriptorExtendedType); | 
|  | 5110 | } | 
|  | 5111 |  | 
| Fariborz Jahanian | 998f0a3 | 2012-11-28 23:12:17 +0000 | [diff] [blame] | 5112 | /// BlockRequiresCopying - Returns true if byref variable "D" of type "Ty" | 
|  | 5113 | /// requires copy/dispose. Note that this must match the logic | 
|  | 5114 | /// in buildByrefHelpers. | 
|  | 5115 | bool ASTContext::BlockRequiresCopying(QualType Ty, | 
|  | 5116 | const VarDecl *D) { | 
|  | 5117 | if (const CXXRecordDecl *record = Ty->getAsCXXRecordDecl()) { | 
|  | 5118 | const Expr *copyExpr = getBlockVarCopyInits(D); | 
|  | 5119 | if (!copyExpr && record->hasTrivialDestructor()) return false; | 
|  | 5120 |  | 
| Mike Stump | 9496790 | 2009-10-21 18:16:27 +0000 | [diff] [blame] | 5121 | return true; | 
| Fariborz Jahanian | a00076c | 2010-11-17 00:21:28 +0000 | [diff] [blame] | 5122 | } | 
| Fariborz Jahanian | 998f0a3 | 2012-11-28 23:12:17 +0000 | [diff] [blame] | 5123 |  | 
|  | 5124 | if (!Ty->isObjCRetainableType()) return false; | 
|  | 5125 |  | 
|  | 5126 | Qualifiers qs = Ty.getQualifiers(); | 
|  | 5127 |  | 
|  | 5128 | // If we have lifetime, that dominates. | 
|  | 5129 | if (Qualifiers::ObjCLifetime lifetime = qs.getObjCLifetime()) { | 
| Fariborz Jahanian | 998f0a3 | 2012-11-28 23:12:17 +0000 | [diff] [blame] | 5130 | switch (lifetime) { | 
|  | 5131 | case Qualifiers::OCL_None: llvm_unreachable("impossible"); | 
|  | 5132 |  | 
|  | 5133 | // These are just bits as far as the runtime is concerned. | 
|  | 5134 | case Qualifiers::OCL_ExplicitNone: | 
|  | 5135 | case Qualifiers::OCL_Autoreleasing: | 
|  | 5136 | return false; | 
|  | 5137 |  | 
|  | 5138 | // Tell the runtime that this is ARC __weak, called by the | 
|  | 5139 | // byref routines. | 
|  | 5140 | case Qualifiers::OCL_Weak: | 
|  | 5141 | // ARC __strong __block variables need to be retained. | 
|  | 5142 | case Qualifiers::OCL_Strong: | 
|  | 5143 | return true; | 
|  | 5144 | } | 
|  | 5145 | llvm_unreachable("fell out of lifetime switch!"); | 
|  | 5146 | } | 
|  | 5147 | return (Ty->isBlockPointerType() || isObjCNSObjectType(Ty) || | 
|  | 5148 | Ty->isObjCObjectPointerType()); | 
| Mike Stump | 9496790 | 2009-10-21 18:16:27 +0000 | [diff] [blame] | 5149 | } | 
|  | 5150 |  | 
| Fariborz Jahanian | a9d4464 | 2012-11-14 17:15:51 +0000 | [diff] [blame] | 5151 | bool ASTContext::getByrefLifetime(QualType Ty, | 
|  | 5152 | Qualifiers::ObjCLifetime &LifeTime, | 
|  | 5153 | bool &HasByrefExtendedLayout) const { | 
|  | 5154 |  | 
|  | 5155 | if (!getLangOpts().ObjC1 || | 
|  | 5156 | getLangOpts().getGC() != LangOptions::NonGC) | 
|  | 5157 | return false; | 
|  | 5158 |  | 
|  | 5159 | HasByrefExtendedLayout = false; | 
| Fariborz Jahanian | f762b72 | 2012-12-11 19:58:01 +0000 | [diff] [blame] | 5160 | if (Ty->isRecordType()) { | 
| Fariborz Jahanian | a9d4464 | 2012-11-14 17:15:51 +0000 | [diff] [blame] | 5161 | HasByrefExtendedLayout = true; | 
|  | 5162 | LifeTime = Qualifiers::OCL_None; | 
| John McCall | 460ce58 | 2015-10-22 18:38:17 +0000 | [diff] [blame] | 5163 | } else if ((LifeTime = Ty.getObjCLifetime())) { | 
|  | 5164 | // Honor the ARC qualifiers. | 
|  | 5165 | } else if (Ty->isObjCObjectPointerType() || Ty->isBlockPointerType()) { | 
|  | 5166 | // The MRR rule. | 
| Fariborz Jahanian | a9d4464 | 2012-11-14 17:15:51 +0000 | [diff] [blame] | 5167 | LifeTime = Qualifiers::OCL_ExplicitNone; | 
| John McCall | 460ce58 | 2015-10-22 18:38:17 +0000 | [diff] [blame] | 5168 | } else { | 
| Fariborz Jahanian | a9d4464 | 2012-11-14 17:15:51 +0000 | [diff] [blame] | 5169 | LifeTime = Qualifiers::OCL_None; | 
| John McCall | 460ce58 | 2015-10-22 18:38:17 +0000 | [diff] [blame] | 5170 | } | 
| Fariborz Jahanian | a9d4464 | 2012-11-14 17:15:51 +0000 | [diff] [blame] | 5171 | return true; | 
|  | 5172 | } | 
|  | 5173 |  | 
| Douglas Gregor | bab8a96 | 2011-09-08 01:46:34 +0000 | [diff] [blame] | 5174 | TypedefDecl *ASTContext::getObjCInstanceTypeDecl() { | 
|  | 5175 | if (!ObjCInstanceTypeDecl) | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 5176 | ObjCInstanceTypeDecl = | 
|  | 5177 | buildImplicitTypedef(getObjCIdType(), "instancetype"); | 
| Douglas Gregor | bab8a96 | 2011-09-08 01:46:34 +0000 | [diff] [blame] | 5178 | return ObjCInstanceTypeDecl; | 
|  | 5179 | } | 
|  | 5180 |  | 
| Anders Carlsson | 18acd44 | 2007-10-29 06:33:42 +0000 | [diff] [blame] | 5181 | // This returns true if a type has been typedefed to BOOL: | 
|  | 5182 | // typedef <type> BOOL; | 
| Chris Lattner | e021899 | 2007-10-30 20:27:44 +0000 | [diff] [blame] | 5183 | static bool isTypeTypedefedAsBOOL(QualType T) { | 
| Anders Carlsson | 18acd44 | 2007-10-29 06:33:42 +0000 | [diff] [blame] | 5184 | if (const TypedefType *TT = dyn_cast<TypedefType>(T)) | 
| Chris Lattner | 9b1f279 | 2008-11-24 03:52:59 +0000 | [diff] [blame] | 5185 | if (IdentifierInfo *II = TT->getDecl()->getIdentifier()) | 
|  | 5186 | return II->isStr("BOOL"); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5187 |  | 
| Anders Carlsson | d849982 | 2007-10-29 05:01:08 +0000 | [diff] [blame] | 5188 | return false; | 
|  | 5189 | } | 
|  | 5190 |  | 
| Ted Kremenek | 1b0ea82 | 2008-01-07 19:49:32 +0000 | [diff] [blame] | 5191 | /// getObjCEncodingTypeSize returns size of type for objective-c encoding | 
| Fariborz Jahanian | 797f24c | 2007-10-29 22:57:28 +0000 | [diff] [blame] | 5192 | /// purpose. | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 5193 | CharUnits ASTContext::getObjCEncodingTypeSize(QualType type) const { | 
| Douglas Gregor | a9d8493 | 2011-05-27 01:19:52 +0000 | [diff] [blame] | 5194 | if (!type->isIncompleteArrayType() && type->isIncompleteType()) | 
|  | 5195 | return CharUnits::Zero(); | 
|  | 5196 |  | 
| Ken Dyck | 4077500 | 2010-01-11 17:06:35 +0000 | [diff] [blame] | 5197 | CharUnits sz = getTypeSizeInChars(type); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5198 |  | 
| Fariborz Jahanian | 797f24c | 2007-10-29 22:57:28 +0000 | [diff] [blame] | 5199 | // Make all integer and enum types at least as large as an int | 
| Douglas Gregor | b90df60 | 2010-06-16 00:17:44 +0000 | [diff] [blame] | 5200 | if (sz.isPositive() && type->isIntegralOrEnumerationType()) | 
| Ken Dyck | 4077500 | 2010-01-11 17:06:35 +0000 | [diff] [blame] | 5201 | sz = std::max(sz, getTypeSizeInChars(IntTy)); | 
| Fariborz Jahanian | 797f24c | 2007-10-29 22:57:28 +0000 | [diff] [blame] | 5202 | // Treat arrays as pointers, since that's how they're passed in. | 
|  | 5203 | else if (type->isArrayType()) | 
| Ken Dyck | 4077500 | 2010-01-11 17:06:35 +0000 | [diff] [blame] | 5204 | sz = getTypeSizeInChars(VoidPtrTy); | 
| Ken Dyck | de37a67 | 2010-01-11 19:19:56 +0000 | [diff] [blame] | 5205 | return sz; | 
| Ken Dyck | 4077500 | 2010-01-11 17:06:35 +0000 | [diff] [blame] | 5206 | } | 
|  | 5207 |  | 
| Hans Wennborg | 56fc62b | 2014-07-17 20:25:23 +0000 | [diff] [blame] | 5208 | bool ASTContext::isMSStaticDataMemberInlineDefinition(const VarDecl *VD) const { | 
| David Majnemer | 3f02150 | 2015-10-08 04:53:31 +0000 | [diff] [blame] | 5209 | return getTargetInfo().getCXXABI().isMicrosoft() && | 
|  | 5210 | VD->isStaticDataMember() && | 
| David Majnemer | fac5243 | 2015-10-19 23:22:49 +0000 | [diff] [blame] | 5211 | VD->getType()->isIntegralOrEnumerationType() && | 
|  | 5212 | !VD->getFirstDecl()->isOutOfLine() && VD->getFirstDecl()->hasInit(); | 
| Hans Wennborg | 56fc62b | 2014-07-17 20:25:23 +0000 | [diff] [blame] | 5213 | } | 
|  | 5214 |  | 
| Richard Smith | d9b9009 | 2016-07-02 01:32:16 +0000 | [diff] [blame] | 5215 | ASTContext::InlineVariableDefinitionKind | 
|  | 5216 | ASTContext::getInlineVariableDefinitionKind(const VarDecl *VD) const { | 
|  | 5217 | if (!VD->isInline()) | 
|  | 5218 | return InlineVariableDefinitionKind::None; | 
|  | 5219 |  | 
|  | 5220 | // In almost all cases, it's a weak definition. | 
|  | 5221 | auto *First = VD->getFirstDecl(); | 
|  | 5222 | if (!First->isConstexpr() || First->isInlineSpecified() || | 
|  | 5223 | !VD->isStaticDataMember()) | 
|  | 5224 | return InlineVariableDefinitionKind::Weak; | 
|  | 5225 |  | 
|  | 5226 | // If there's a file-context declaration in this translation unit, it's a | 
|  | 5227 | // non-discardable definition. | 
|  | 5228 | for (auto *D : VD->redecls()) | 
|  | 5229 | if (D->getLexicalDeclContext()->isFileContext()) | 
|  | 5230 | return InlineVariableDefinitionKind::Strong; | 
|  | 5231 |  | 
|  | 5232 | // If we've not seen one yet, we don't know. | 
|  | 5233 | return InlineVariableDefinitionKind::WeakUnknown; | 
|  | 5234 | } | 
|  | 5235 |  | 
| Ken Dyck | 4077500 | 2010-01-11 17:06:35 +0000 | [diff] [blame] | 5236 | static inline | 
|  | 5237 | std::string charUnitsToString(const CharUnits &CU) { | 
|  | 5238 | return llvm::itostr(CU.getQuantity()); | 
| Fariborz Jahanian | 797f24c | 2007-10-29 22:57:28 +0000 | [diff] [blame] | 5239 | } | 
|  | 5240 |  | 
| John McCall | 351762c | 2011-02-07 10:33:21 +0000 | [diff] [blame] | 5241 | /// getObjCEncodingForBlock - Return the encoded type for this block | 
| David Chisnall | 950a951 | 2009-11-17 19:33:30 +0000 | [diff] [blame] | 5242 | /// declaration. | 
| John McCall | 351762c | 2011-02-07 10:33:21 +0000 | [diff] [blame] | 5243 | std::string ASTContext::getObjCEncodingForBlock(const BlockExpr *Expr) const { | 
|  | 5244 | std::string S; | 
|  | 5245 |  | 
| David Chisnall | 950a951 | 2009-11-17 19:33:30 +0000 | [diff] [blame] | 5246 | const BlockDecl *Decl = Expr->getBlockDecl(); | 
|  | 5247 | QualType BlockTy = | 
|  | 5248 | Expr->getType()->getAs<BlockPointerType>()->getPointeeType(); | 
|  | 5249 | // Encode result type. | 
| Fariborz Jahanian | 0e3043b | 2012-11-15 19:02:45 +0000 | [diff] [blame] | 5250 | if (getLangOpts().EncodeExtendedBlockSig) | 
| Alp Toker | 314cc81 | 2014-01-25 16:55:45 +0000 | [diff] [blame] | 5251 | getObjCEncodingForMethodParameter( | 
|  | 5252 | Decl::OBJC_TQ_None, BlockTy->getAs<FunctionType>()->getReturnType(), S, | 
|  | 5253 | true /*Extended*/); | 
| Fariborz Jahanian | 64223e6 | 2012-11-14 23:11:38 +0000 | [diff] [blame] | 5254 | else | 
| Alp Toker | 314cc81 | 2014-01-25 16:55:45 +0000 | [diff] [blame] | 5255 | getObjCEncodingForType(BlockTy->getAs<FunctionType>()->getReturnType(), S); | 
| David Chisnall | 950a951 | 2009-11-17 19:33:30 +0000 | [diff] [blame] | 5256 | // Compute size of all parameters. | 
|  | 5257 | // Start with computing size of a pointer in number of bytes. | 
|  | 5258 | // FIXME: There might(should) be a better way of doing this computation! | 
|  | 5259 | SourceLocation Loc; | 
| Ken Dyck | 4077500 | 2010-01-11 17:06:35 +0000 | [diff] [blame] | 5260 | CharUnits PtrSize = getTypeSizeInChars(VoidPtrTy); | 
|  | 5261 | CharUnits ParmOffset = PtrSize; | 
| David Majnemer | 59f7792 | 2016-06-24 04:05:48 +0000 | [diff] [blame] | 5262 | for (auto PI : Decl->parameters()) { | 
| Aaron Ballman | b2b8b1d | 2014-03-07 16:09:59 +0000 | [diff] [blame] | 5263 | QualType PType = PI->getType(); | 
| Ken Dyck | de37a67 | 2010-01-11 19:19:56 +0000 | [diff] [blame] | 5264 | CharUnits sz = getObjCEncodingTypeSize(PType); | 
| Fariborz Jahanian | d487941 | 2012-06-30 00:48:59 +0000 | [diff] [blame] | 5265 | if (sz.isZero()) | 
|  | 5266 | continue; | 
| Ken Dyck | 4077500 | 2010-01-11 17:06:35 +0000 | [diff] [blame] | 5267 | assert (sz.isPositive() && "BlockExpr - Incomplete param type"); | 
| David Chisnall | 950a951 | 2009-11-17 19:33:30 +0000 | [diff] [blame] | 5268 | ParmOffset += sz; | 
|  | 5269 | } | 
|  | 5270 | // Size of the argument frame | 
| Ken Dyck | 4077500 | 2010-01-11 17:06:35 +0000 | [diff] [blame] | 5271 | S += charUnitsToString(ParmOffset); | 
| David Chisnall | 950a951 | 2009-11-17 19:33:30 +0000 | [diff] [blame] | 5272 | // Block pointer and offset. | 
|  | 5273 | S += "@?0"; | 
| David Chisnall | 950a951 | 2009-11-17 19:33:30 +0000 | [diff] [blame] | 5274 |  | 
|  | 5275 | // Argument types. | 
|  | 5276 | ParmOffset = PtrSize; | 
| David Majnemer | 59f7792 | 2016-06-24 04:05:48 +0000 | [diff] [blame] | 5277 | for (auto PVDecl : Decl->parameters()) { | 
| David Chisnall | 950a951 | 2009-11-17 19:33:30 +0000 | [diff] [blame] | 5278 | QualType PType = PVDecl->getOriginalType(); | 
|  | 5279 | if (const ArrayType *AT = | 
|  | 5280 | dyn_cast<ArrayType>(PType->getCanonicalTypeInternal())) { | 
|  | 5281 | // Use array's original type only if it has known number of | 
|  | 5282 | // elements. | 
|  | 5283 | if (!isa<ConstantArrayType>(AT)) | 
|  | 5284 | PType = PVDecl->getType(); | 
|  | 5285 | } else if (PType->isFunctionType()) | 
|  | 5286 | PType = PVDecl->getType(); | 
| Fariborz Jahanian | 0e3043b | 2012-11-15 19:02:45 +0000 | [diff] [blame] | 5287 | if (getLangOpts().EncodeExtendedBlockSig) | 
| Fariborz Jahanian | 64223e6 | 2012-11-14 23:11:38 +0000 | [diff] [blame] | 5288 | getObjCEncodingForMethodParameter(Decl::OBJC_TQ_None, PType, | 
|  | 5289 | S, true /*Extended*/); | 
|  | 5290 | else | 
|  | 5291 | getObjCEncodingForType(PType, S); | 
| Ken Dyck | 4077500 | 2010-01-11 17:06:35 +0000 | [diff] [blame] | 5292 | S += charUnitsToString(ParmOffset); | 
| Ken Dyck | de37a67 | 2010-01-11 19:19:56 +0000 | [diff] [blame] | 5293 | ParmOffset += getObjCEncodingTypeSize(PType); | 
| David Chisnall | 950a951 | 2009-11-17 19:33:30 +0000 | [diff] [blame] | 5294 | } | 
| John McCall | 351762c | 2011-02-07 10:33:21 +0000 | [diff] [blame] | 5295 |  | 
|  | 5296 | return S; | 
| David Chisnall | 950a951 | 2009-11-17 19:33:30 +0000 | [diff] [blame] | 5297 | } | 
|  | 5298 |  | 
| Douglas Gregor | a9d8493 | 2011-05-27 01:19:52 +0000 | [diff] [blame] | 5299 | bool ASTContext::getObjCEncodingForFunctionDecl(const FunctionDecl *Decl, | 
| David Chisnall | 50e4eba | 2010-12-30 14:05:53 +0000 | [diff] [blame] | 5300 | std::string& S) { | 
|  | 5301 | // Encode result type. | 
| Alp Toker | 314cc81 | 2014-01-25 16:55:45 +0000 | [diff] [blame] | 5302 | getObjCEncodingForType(Decl->getReturnType(), S); | 
| David Chisnall | 50e4eba | 2010-12-30 14:05:53 +0000 | [diff] [blame] | 5303 | CharUnits ParmOffset; | 
|  | 5304 | // Compute size of all parameters. | 
| David Majnemer | 59f7792 | 2016-06-24 04:05:48 +0000 | [diff] [blame] | 5305 | for (auto PI : Decl->parameters()) { | 
| Aaron Ballman | f6bf62e | 2014-03-07 15:12:56 +0000 | [diff] [blame] | 5306 | QualType PType = PI->getType(); | 
| David Chisnall | 50e4eba | 2010-12-30 14:05:53 +0000 | [diff] [blame] | 5307 | CharUnits sz = getObjCEncodingTypeSize(PType); | 
| Douglas Gregor | a9d8493 | 2011-05-27 01:19:52 +0000 | [diff] [blame] | 5308 | if (sz.isZero()) | 
| Fariborz Jahanian | 271b8d4 | 2012-06-29 22:54:56 +0000 | [diff] [blame] | 5309 | continue; | 
|  | 5310 |  | 
| David Chisnall | 50e4eba | 2010-12-30 14:05:53 +0000 | [diff] [blame] | 5311 | assert (sz.isPositive() && | 
| Douglas Gregor | a9d8493 | 2011-05-27 01:19:52 +0000 | [diff] [blame] | 5312 | "getObjCEncodingForFunctionDecl - Incomplete param type"); | 
| David Chisnall | 50e4eba | 2010-12-30 14:05:53 +0000 | [diff] [blame] | 5313 | ParmOffset += sz; | 
|  | 5314 | } | 
|  | 5315 | S += charUnitsToString(ParmOffset); | 
|  | 5316 | ParmOffset = CharUnits::Zero(); | 
|  | 5317 |  | 
|  | 5318 | // Argument types. | 
| David Majnemer | 59f7792 | 2016-06-24 04:05:48 +0000 | [diff] [blame] | 5319 | for (auto PVDecl : Decl->parameters()) { | 
| David Chisnall | 50e4eba | 2010-12-30 14:05:53 +0000 | [diff] [blame] | 5320 | QualType PType = PVDecl->getOriginalType(); | 
|  | 5321 | if (const ArrayType *AT = | 
|  | 5322 | dyn_cast<ArrayType>(PType->getCanonicalTypeInternal())) { | 
|  | 5323 | // Use array's original type only if it has known number of | 
|  | 5324 | // elements. | 
|  | 5325 | if (!isa<ConstantArrayType>(AT)) | 
|  | 5326 | PType = PVDecl->getType(); | 
|  | 5327 | } else if (PType->isFunctionType()) | 
|  | 5328 | PType = PVDecl->getType(); | 
|  | 5329 | getObjCEncodingForType(PType, S); | 
|  | 5330 | S += charUnitsToString(ParmOffset); | 
|  | 5331 | ParmOffset += getObjCEncodingTypeSize(PType); | 
|  | 5332 | } | 
| Douglas Gregor | a9d8493 | 2011-05-27 01:19:52 +0000 | [diff] [blame] | 5333 |  | 
|  | 5334 | return false; | 
| David Chisnall | 50e4eba | 2010-12-30 14:05:53 +0000 | [diff] [blame] | 5335 | } | 
|  | 5336 |  | 
| Bob Wilson | 5f4e3a7 | 2011-11-30 01:57:58 +0000 | [diff] [blame] | 5337 | /// getObjCEncodingForMethodParameter - Return the encoded type for a single | 
|  | 5338 | /// method parameter or return type. If Extended, include class names and | 
|  | 5339 | /// block object types. | 
|  | 5340 | void ASTContext::getObjCEncodingForMethodParameter(Decl::ObjCDeclQualifier QT, | 
|  | 5341 | QualType T, std::string& S, | 
|  | 5342 | bool Extended) const { | 
|  | 5343 | // Encode type qualifer, 'in', 'inout', etc. for the parameter. | 
|  | 5344 | getObjCEncodingForTypeQualifier(QT, S); | 
|  | 5345 | // Encode parameter type. | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 5346 | getObjCEncodingForTypeImpl(T, S, true, true, nullptr, | 
| Bob Wilson | 5f4e3a7 | 2011-11-30 01:57:58 +0000 | [diff] [blame] | 5347 | true     /*OutermostType*/, | 
|  | 5348 | false    /*EncodingProperty*/, | 
|  | 5349 | false    /*StructField*/, | 
|  | 5350 | Extended /*EncodeBlockParameters*/, | 
|  | 5351 | Extended /*EncodeClassNames*/); | 
|  | 5352 | } | 
|  | 5353 |  | 
| Ted Kremenek | 1b0ea82 | 2008-01-07 19:49:32 +0000 | [diff] [blame] | 5354 | /// getObjCEncodingForMethodDecl - Return the encoded type for this method | 
| Fariborz Jahanian | 797f24c | 2007-10-29 22:57:28 +0000 | [diff] [blame] | 5355 | /// declaration. | 
| Douglas Gregor | a9d8493 | 2011-05-27 01:19:52 +0000 | [diff] [blame] | 5356 | bool ASTContext::getObjCEncodingForMethodDecl(const ObjCMethodDecl *Decl, | 
| Bob Wilson | 5f4e3a7 | 2011-11-30 01:57:58 +0000 | [diff] [blame] | 5357 | std::string& S, | 
|  | 5358 | bool Extended) const { | 
| Daniel Dunbar | 4932b36 | 2008-08-28 04:38:10 +0000 | [diff] [blame] | 5359 | // FIXME: This is not very efficient. | 
| Bob Wilson | 5f4e3a7 | 2011-11-30 01:57:58 +0000 | [diff] [blame] | 5360 | // Encode return type. | 
| Alp Toker | 314cc81 | 2014-01-25 16:55:45 +0000 | [diff] [blame] | 5361 | getObjCEncodingForMethodParameter(Decl->getObjCDeclQualifier(), | 
|  | 5362 | Decl->getReturnType(), S, Extended); | 
| Fariborz Jahanian | 797f24c | 2007-10-29 22:57:28 +0000 | [diff] [blame] | 5363 | // Compute size of all parameters. | 
|  | 5364 | // Start with computing size of a pointer in number of bytes. | 
|  | 5365 | // FIXME: There might(should) be a better way of doing this computation! | 
|  | 5366 | SourceLocation Loc; | 
| Ken Dyck | 4077500 | 2010-01-11 17:06:35 +0000 | [diff] [blame] | 5367 | CharUnits PtrSize = getTypeSizeInChars(VoidPtrTy); | 
| Fariborz Jahanian | 797f24c | 2007-10-29 22:57:28 +0000 | [diff] [blame] | 5368 | // The first two arguments (self and _cmd) are pointers; account for | 
|  | 5369 | // their size. | 
| Ken Dyck | 4077500 | 2010-01-11 17:06:35 +0000 | [diff] [blame] | 5370 | CharUnits ParmOffset = 2 * PtrSize; | 
| Argyrios Kyrtzidis | b8c3aaf | 2011-10-03 06:37:04 +0000 | [diff] [blame] | 5371 | for (ObjCMethodDecl::param_const_iterator PI = Decl->param_begin(), | 
| Fariborz Jahanian | d9235db | 2010-04-08 21:29:11 +0000 | [diff] [blame] | 5372 | E = Decl->sel_param_end(); PI != E; ++PI) { | 
| Chris Lattner | a499715 | 2009-02-20 18:43:26 +0000 | [diff] [blame] | 5373 | QualType PType = (*PI)->getType(); | 
| Ken Dyck | de37a67 | 2010-01-11 19:19:56 +0000 | [diff] [blame] | 5374 | CharUnits sz = getObjCEncodingTypeSize(PType); | 
| Douglas Gregor | a9d8493 | 2011-05-27 01:19:52 +0000 | [diff] [blame] | 5375 | if (sz.isZero()) | 
| Fariborz Jahanian | 271b8d4 | 2012-06-29 22:54:56 +0000 | [diff] [blame] | 5376 | continue; | 
|  | 5377 |  | 
| Ken Dyck | 4077500 | 2010-01-11 17:06:35 +0000 | [diff] [blame] | 5378 | assert (sz.isPositive() && | 
|  | 5379 | "getObjCEncodingForMethodDecl - Incomplete param type"); | 
| Fariborz Jahanian | 797f24c | 2007-10-29 22:57:28 +0000 | [diff] [blame] | 5380 | ParmOffset += sz; | 
|  | 5381 | } | 
| Ken Dyck | 4077500 | 2010-01-11 17:06:35 +0000 | [diff] [blame] | 5382 | S += charUnitsToString(ParmOffset); | 
| Fariborz Jahanian | 797f24c | 2007-10-29 22:57:28 +0000 | [diff] [blame] | 5383 | S += "@0:"; | 
| Ken Dyck | 4077500 | 2010-01-11 17:06:35 +0000 | [diff] [blame] | 5384 | S += charUnitsToString(PtrSize); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5385 |  | 
| Fariborz Jahanian | 797f24c | 2007-10-29 22:57:28 +0000 | [diff] [blame] | 5386 | // Argument types. | 
|  | 5387 | ParmOffset = 2 * PtrSize; | 
| Argyrios Kyrtzidis | b8c3aaf | 2011-10-03 06:37:04 +0000 | [diff] [blame] | 5388 | for (ObjCMethodDecl::param_const_iterator PI = Decl->param_begin(), | 
| Fariborz Jahanian | d9235db | 2010-04-08 21:29:11 +0000 | [diff] [blame] | 5389 | E = Decl->sel_param_end(); PI != E; ++PI) { | 
| Argyrios Kyrtzidis | b8c3aaf | 2011-10-03 06:37:04 +0000 | [diff] [blame] | 5390 | const ParmVarDecl *PVDecl = *PI; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5391 | QualType PType = PVDecl->getOriginalType(); | 
| Fariborz Jahanian | a0befc0 | 2008-12-20 23:29:59 +0000 | [diff] [blame] | 5392 | if (const ArrayType *AT = | 
| Steve Naroff | e4e55d2 | 2009-04-14 00:03:58 +0000 | [diff] [blame] | 5393 | dyn_cast<ArrayType>(PType->getCanonicalTypeInternal())) { | 
|  | 5394 | // Use array's original type only if it has known number of | 
|  | 5395 | // elements. | 
| Steve Naroff | 323827e | 2009-04-14 00:40:09 +0000 | [diff] [blame] | 5396 | if (!isa<ConstantArrayType>(AT)) | 
| Steve Naroff | e4e55d2 | 2009-04-14 00:03:58 +0000 | [diff] [blame] | 5397 | PType = PVDecl->getType(); | 
|  | 5398 | } else if (PType->isFunctionType()) | 
|  | 5399 | PType = PVDecl->getType(); | 
| Bob Wilson | 5f4e3a7 | 2011-11-30 01:57:58 +0000 | [diff] [blame] | 5400 | getObjCEncodingForMethodParameter(PVDecl->getObjCDeclQualifier(), | 
|  | 5401 | PType, S, Extended); | 
| Ken Dyck | 4077500 | 2010-01-11 17:06:35 +0000 | [diff] [blame] | 5402 | S += charUnitsToString(ParmOffset); | 
| Ken Dyck | de37a67 | 2010-01-11 19:19:56 +0000 | [diff] [blame] | 5403 | ParmOffset += getObjCEncodingTypeSize(PType); | 
| Fariborz Jahanian | 797f24c | 2007-10-29 22:57:28 +0000 | [diff] [blame] | 5404 | } | 
| Douglas Gregor | a9d8493 | 2011-05-27 01:19:52 +0000 | [diff] [blame] | 5405 |  | 
|  | 5406 | return false; | 
| Fariborz Jahanian | 797f24c | 2007-10-29 22:57:28 +0000 | [diff] [blame] | 5407 | } | 
|  | 5408 |  | 
| Fariborz Jahanian | dad9630 | 2014-01-22 19:02:20 +0000 | [diff] [blame] | 5409 | ObjCPropertyImplDecl * | 
|  | 5410 | ASTContext::getObjCPropertyImplDeclForPropertyDecl( | 
|  | 5411 | const ObjCPropertyDecl *PD, | 
|  | 5412 | const Decl *Container) const { | 
|  | 5413 | if (!Container) | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 5414 | return nullptr; | 
| Fariborz Jahanian | dad9630 | 2014-01-22 19:02:20 +0000 | [diff] [blame] | 5415 | if (const ObjCCategoryImplDecl *CID = | 
|  | 5416 | dyn_cast<ObjCCategoryImplDecl>(Container)) { | 
| Aaron Ballman | d85eff4 | 2014-03-14 15:02:45 +0000 | [diff] [blame] | 5417 | for (auto *PID : CID->property_impls()) | 
|  | 5418 | if (PID->getPropertyDecl() == PD) | 
|  | 5419 | return PID; | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 5420 | } else { | 
|  | 5421 | const ObjCImplementationDecl *OID=cast<ObjCImplementationDecl>(Container); | 
|  | 5422 | for (auto *PID : OID->property_impls()) | 
|  | 5423 | if (PID->getPropertyDecl() == PD) | 
|  | 5424 | return PID; | 
|  | 5425 | } | 
|  | 5426 | return nullptr; | 
| Fariborz Jahanian | dad9630 | 2014-01-22 19:02:20 +0000 | [diff] [blame] | 5427 | } | 
|  | 5428 |  | 
| Daniel Dunbar | 4932b36 | 2008-08-28 04:38:10 +0000 | [diff] [blame] | 5429 | /// getObjCEncodingForPropertyDecl - Return the encoded type for this | 
| Fariborz Jahanian | 2f85a64 | 2009-01-20 20:04:12 +0000 | [diff] [blame] | 5430 | /// property declaration. If non-NULL, Container must be either an | 
| Daniel Dunbar | 4932b36 | 2008-08-28 04:38:10 +0000 | [diff] [blame] | 5431 | /// ObjCCategoryImplDecl or ObjCImplementationDecl; it should only be | 
|  | 5432 | /// NULL when getting encodings for protocol properties. | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5433 | /// Property attributes are stored as a comma-delimited C string. The simple | 
|  | 5434 | /// attributes readonly and bycopy are encoded as single characters. The | 
|  | 5435 | /// parametrized attributes, getter=name, setter=name, and ivar=name, are | 
|  | 5436 | /// encoded as single characters, followed by an identifier. Property types | 
|  | 5437 | /// are also encoded as a parametrized attribute. The characters used to encode | 
| Fariborz Jahanian | 2f85a64 | 2009-01-20 20:04:12 +0000 | [diff] [blame] | 5438 | /// these attributes are defined by the following enumeration: | 
|  | 5439 | /// @code | 
|  | 5440 | /// enum PropertyAttributes { | 
|  | 5441 | /// kPropertyReadOnly = 'R',   // property is read-only. | 
|  | 5442 | /// kPropertyBycopy = 'C',     // property is a copy of the value last assigned | 
|  | 5443 | /// kPropertyByref = '&',  // property is a reference to the value last assigned | 
|  | 5444 | /// kPropertyDynamic = 'D',    // property is dynamic | 
|  | 5445 | /// kPropertyGetter = 'G',     // followed by getter selector name | 
|  | 5446 | /// kPropertySetter = 'S',     // followed by setter selector name | 
|  | 5447 | /// kPropertyInstanceVariable = 'V'  // followed by instance variable  name | 
| Bob Wilson | 8cf6185 | 2012-03-22 17:48:02 +0000 | [diff] [blame] | 5448 | /// kPropertyType = 'T'              // followed by old-style type encoding. | 
| Fariborz Jahanian | 2f85a64 | 2009-01-20 20:04:12 +0000 | [diff] [blame] | 5449 | /// kPropertyWeak = 'W'              // 'weak' property | 
|  | 5450 | /// kPropertyStrong = 'P'            // property GC'able | 
|  | 5451 | /// kPropertyNonAtomic = 'N'         // property non-atomic | 
|  | 5452 | /// }; | 
|  | 5453 | /// @endcode | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5454 | void ASTContext::getObjCEncodingForPropertyDecl(const ObjCPropertyDecl *PD, | 
| Daniel Dunbar | 4932b36 | 2008-08-28 04:38:10 +0000 | [diff] [blame] | 5455 | const Decl *Container, | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 5456 | std::string& S) const { | 
| Daniel Dunbar | 4932b36 | 2008-08-28 04:38:10 +0000 | [diff] [blame] | 5457 | // Collect information from the property implementation decl(s). | 
|  | 5458 | bool Dynamic = false; | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 5459 | ObjCPropertyImplDecl *SynthesizePID = nullptr; | 
| Daniel Dunbar | 4932b36 | 2008-08-28 04:38:10 +0000 | [diff] [blame] | 5460 |  | 
| Fariborz Jahanian | dad9630 | 2014-01-22 19:02:20 +0000 | [diff] [blame] | 5461 | if (ObjCPropertyImplDecl *PropertyImpDecl = | 
|  | 5462 | getObjCPropertyImplDeclForPropertyDecl(PD, Container)) { | 
|  | 5463 | if (PropertyImpDecl->getPropertyImplementation() == ObjCPropertyImplDecl::Dynamic) | 
|  | 5464 | Dynamic = true; | 
|  | 5465 | else | 
|  | 5466 | SynthesizePID = PropertyImpDecl; | 
| Daniel Dunbar | 4932b36 | 2008-08-28 04:38:10 +0000 | [diff] [blame] | 5467 | } | 
|  | 5468 |  | 
|  | 5469 | // FIXME: This is not very efficient. | 
|  | 5470 | S = "T"; | 
|  | 5471 |  | 
|  | 5472 | // Encode result type. | 
| Fariborz Jahanian | 218c630 | 2009-01-20 19:14:18 +0000 | [diff] [blame] | 5473 | // GCC has some special rules regarding encoding of properties which | 
|  | 5474 | // closely resembles encoding of ivars. | 
| Joe Groff | 98ac7d2 | 2014-07-07 22:25:15 +0000 | [diff] [blame] | 5475 | getObjCEncodingForPropertyType(PD->getType(), S); | 
| Daniel Dunbar | 4932b36 | 2008-08-28 04:38:10 +0000 | [diff] [blame] | 5476 |  | 
|  | 5477 | if (PD->isReadOnly()) { | 
|  | 5478 | S += ",R"; | 
| Nico Weber | 4e8626f | 2013-05-08 23:47:40 +0000 | [diff] [blame] | 5479 | if (PD->getPropertyAttributes() & ObjCPropertyDecl::OBJC_PR_copy) | 
|  | 5480 | S += ",C"; | 
|  | 5481 | if (PD->getPropertyAttributes() & ObjCPropertyDecl::OBJC_PR_retain) | 
|  | 5482 | S += ",&"; | 
| Fariborz Jahanian | 33079ee | 2014-04-02 22:49:42 +0000 | [diff] [blame] | 5483 | if (PD->getPropertyAttributes() & ObjCPropertyDecl::OBJC_PR_weak) | 
|  | 5484 | S += ",W"; | 
| Daniel Dunbar | 4932b36 | 2008-08-28 04:38:10 +0000 | [diff] [blame] | 5485 | } else { | 
|  | 5486 | switch (PD->getSetterKind()) { | 
|  | 5487 | case ObjCPropertyDecl::Assign: break; | 
|  | 5488 | case ObjCPropertyDecl::Copy:   S += ",C"; break; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5489 | case ObjCPropertyDecl::Retain: S += ",&"; break; | 
| Fariborz Jahanian | 70a315c | 2011-08-12 20:47:08 +0000 | [diff] [blame] | 5490 | case ObjCPropertyDecl::Weak:   S += ",W"; break; | 
| Daniel Dunbar | 4932b36 | 2008-08-28 04:38:10 +0000 | [diff] [blame] | 5491 | } | 
|  | 5492 | } | 
|  | 5493 |  | 
|  | 5494 | // It really isn't clear at all what this means, since properties | 
|  | 5495 | // are "dynamic by default". | 
|  | 5496 | if (Dynamic) | 
|  | 5497 | S += ",D"; | 
|  | 5498 |  | 
| Fariborz Jahanian | 218c630 | 2009-01-20 19:14:18 +0000 | [diff] [blame] | 5499 | if (PD->getPropertyAttributes() & ObjCPropertyDecl::OBJC_PR_nonatomic) | 
|  | 5500 | S += ",N"; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5501 |  | 
| Daniel Dunbar | 4932b36 | 2008-08-28 04:38:10 +0000 | [diff] [blame] | 5502 | if (PD->getPropertyAttributes() & ObjCPropertyDecl::OBJC_PR_getter) { | 
|  | 5503 | S += ",G"; | 
| Chris Lattner | e4b9569 | 2008-11-24 03:33:13 +0000 | [diff] [blame] | 5504 | S += PD->getGetterName().getAsString(); | 
| Daniel Dunbar | 4932b36 | 2008-08-28 04:38:10 +0000 | [diff] [blame] | 5505 | } | 
|  | 5506 |  | 
|  | 5507 | if (PD->getPropertyAttributes() & ObjCPropertyDecl::OBJC_PR_setter) { | 
|  | 5508 | S += ",S"; | 
| Chris Lattner | e4b9569 | 2008-11-24 03:33:13 +0000 | [diff] [blame] | 5509 | S += PD->getSetterName().getAsString(); | 
| Daniel Dunbar | 4932b36 | 2008-08-28 04:38:10 +0000 | [diff] [blame] | 5510 | } | 
|  | 5511 |  | 
|  | 5512 | if (SynthesizePID) { | 
|  | 5513 | const ObjCIvarDecl *OID = SynthesizePID->getPropertyIvarDecl(); | 
|  | 5514 | S += ",V"; | 
| Chris Lattner | 1cbaacc | 2008-11-24 04:00:27 +0000 | [diff] [blame] | 5515 | S += OID->getNameAsString(); | 
| Daniel Dunbar | 4932b36 | 2008-08-28 04:38:10 +0000 | [diff] [blame] | 5516 | } | 
|  | 5517 |  | 
|  | 5518 | // FIXME: OBJCGC: weak & strong | 
|  | 5519 | } | 
|  | 5520 |  | 
| Fariborz Jahanian | 0f66a6c | 2008-12-23 19:56:47 +0000 | [diff] [blame] | 5521 | /// getLegacyIntegralTypeEncoding - | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5522 | /// Another legacy compatibility encoding: 32-bit longs are encoded as | 
|  | 5523 | /// 'l' or 'L' , but not always.  For typedefs, we need to use | 
| Fariborz Jahanian | 0f66a6c | 2008-12-23 19:56:47 +0000 | [diff] [blame] | 5524 | /// 'i' or 'I' instead if encoding a struct field, or a pointer! | 
|  | 5525 | /// | 
|  | 5526 | void ASTContext::getLegacyIntegralTypeEncoding (QualType &PointeeTy) const { | 
| Mike Stump | 212005c | 2009-07-22 18:58:19 +0000 | [diff] [blame] | 5527 | if (isa<TypedefType>(PointeeTy.getTypePtr())) { | 
| John McCall | 9dd450b | 2009-09-21 23:43:11 +0000 | [diff] [blame] | 5528 | if (const BuiltinType *BT = PointeeTy->getAs<BuiltinType>()) { | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 5529 | if (BT->getKind() == BuiltinType::ULong && getIntWidth(PointeeTy) == 32) | 
| Fariborz Jahanian | 0f66a6c | 2008-12-23 19:56:47 +0000 | [diff] [blame] | 5530 | PointeeTy = UnsignedIntTy; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5531 | else | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 5532 | if (BT->getKind() == BuiltinType::Long && getIntWidth(PointeeTy) == 32) | 
| Fariborz Jahanian | 0f66a6c | 2008-12-23 19:56:47 +0000 | [diff] [blame] | 5533 | PointeeTy = IntTy; | 
|  | 5534 | } | 
|  | 5535 | } | 
|  | 5536 | } | 
|  | 5537 |  | 
| Fariborz Jahanian | 0a71ad2 | 2008-01-22 22:44:46 +0000 | [diff] [blame] | 5538 | void ASTContext::getObjCEncodingForType(QualType T, std::string& S, | 
| Fariborz Jahanian | 4bf437e | 2014-08-22 23:17:52 +0000 | [diff] [blame] | 5539 | const FieldDecl *Field, | 
|  | 5540 | QualType *NotEncodedT) const { | 
| Daniel Dunbar | 3cd9a29 | 2008-10-17 07:30:50 +0000 | [diff] [blame] | 5541 | // We follow the behavior of gcc, expanding structures which are | 
|  | 5542 | // directly pointed to, and expanding embedded structures. Note that | 
|  | 5543 | // these rules are sufficient to prevent recursive encoding of the | 
|  | 5544 | // same type. | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5545 | getObjCEncodingForTypeImpl(T, S, true, true, Field, | 
| Fariborz Jahanian | 4bf437e | 2014-08-22 23:17:52 +0000 | [diff] [blame] | 5546 | true /* outermost type */, false, false, | 
|  | 5547 | false, false, false, NotEncodedT); | 
| Daniel Dunbar | 3cd9a29 | 2008-10-17 07:30:50 +0000 | [diff] [blame] | 5548 | } | 
|  | 5549 |  | 
| Joe Groff | 98ac7d2 | 2014-07-07 22:25:15 +0000 | [diff] [blame] | 5550 | void ASTContext::getObjCEncodingForPropertyType(QualType T, | 
|  | 5551 | std::string& S) const { | 
|  | 5552 | // Encode result type. | 
|  | 5553 | // GCC has some special rules regarding encoding of properties which | 
|  | 5554 | // closely resembles encoding of ivars. | 
|  | 5555 | getObjCEncodingForTypeImpl(T, S, true, true, nullptr, | 
|  | 5556 | true /* outermost type */, | 
|  | 5557 | true /* encoding property */); | 
|  | 5558 | } | 
|  | 5559 |  | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 5560 | static char getObjCEncodingForPrimitiveKind(const ASTContext *C, | 
|  | 5561 | BuiltinType::Kind kind) { | 
|  | 5562 | switch (kind) { | 
| David Chisnall | b190a2c | 2010-06-04 01:10:52 +0000 | [diff] [blame] | 5563 | case BuiltinType::Void:       return 'v'; | 
|  | 5564 | case BuiltinType::Bool:       return 'B'; | 
|  | 5565 | case BuiltinType::Char_U: | 
|  | 5566 | case BuiltinType::UChar:      return 'C'; | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 5567 | case BuiltinType::Char16: | 
| David Chisnall | b190a2c | 2010-06-04 01:10:52 +0000 | [diff] [blame] | 5568 | case BuiltinType::UShort:     return 'S'; | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 5569 | case BuiltinType::Char32: | 
| David Chisnall | b190a2c | 2010-06-04 01:10:52 +0000 | [diff] [blame] | 5570 | case BuiltinType::UInt:       return 'I'; | 
|  | 5571 | case BuiltinType::ULong: | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 5572 | return C->getTargetInfo().getLongWidth() == 32 ? 'L' : 'Q'; | 
| David Chisnall | b190a2c | 2010-06-04 01:10:52 +0000 | [diff] [blame] | 5573 | case BuiltinType::UInt128:    return 'T'; | 
|  | 5574 | case BuiltinType::ULongLong:  return 'Q'; | 
|  | 5575 | case BuiltinType::Char_S: | 
|  | 5576 | case BuiltinType::SChar:      return 'c'; | 
|  | 5577 | case BuiltinType::Short:      return 's'; | 
| Chris Lattner | ad3467e | 2010-12-25 23:25:43 +0000 | [diff] [blame] | 5578 | case BuiltinType::WChar_S: | 
|  | 5579 | case BuiltinType::WChar_U: | 
| David Chisnall | b190a2c | 2010-06-04 01:10:52 +0000 | [diff] [blame] | 5580 | case BuiltinType::Int:        return 'i'; | 
|  | 5581 | case BuiltinType::Long: | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 5582 | return C->getTargetInfo().getLongWidth() == 32 ? 'l' : 'q'; | 
| David Chisnall | b190a2c | 2010-06-04 01:10:52 +0000 | [diff] [blame] | 5583 | case BuiltinType::LongLong:   return 'q'; | 
|  | 5584 | case BuiltinType::Int128:     return 't'; | 
|  | 5585 | case BuiltinType::Float:      return 'f'; | 
|  | 5586 | case BuiltinType::Double:     return 'd'; | 
| Daniel Dunbar | 7cba5a7 | 2010-10-11 21:13:48 +0000 | [diff] [blame] | 5587 | case BuiltinType::LongDouble: return 'D'; | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 5588 | case BuiltinType::NullPtr:    return '*'; // like char* | 
|  | 5589 |  | 
| Nemanja Ivanovic | bb1ea2d | 2016-05-09 08:52:33 +0000 | [diff] [blame] | 5590 | case BuiltinType::Float128: | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 5591 | case BuiltinType::Half: | 
|  | 5592 | // FIXME: potentially need @encodes for these! | 
|  | 5593 | return ' '; | 
|  | 5594 |  | 
|  | 5595 | case BuiltinType::ObjCId: | 
|  | 5596 | case BuiltinType::ObjCClass: | 
|  | 5597 | case BuiltinType::ObjCSel: | 
|  | 5598 | llvm_unreachable("@encoding ObjC primitive type"); | 
|  | 5599 |  | 
|  | 5600 | // OpenCL and placeholder types don't need @encodings. | 
| Alexey Bader | 954ba21 | 2016-04-08 13:40:33 +0000 | [diff] [blame] | 5601 | #define IMAGE_TYPE(ImgType, Id, SingletonId, Access, Suffix) \ | 
|  | 5602 | case BuiltinType::Id: | 
| Alexey Bader | b62f144 | 2016-04-13 08:33:41 +0000 | [diff] [blame] | 5603 | #include "clang/Basic/OpenCLImageTypes.def" | 
| Guy Benyei | 1b4fb3e | 2013-01-20 12:31:11 +0000 | [diff] [blame] | 5604 | case BuiltinType::OCLEvent: | 
| Alexey Bader | 9c8453f | 2015-09-15 11:18:52 +0000 | [diff] [blame] | 5605 | case BuiltinType::OCLClkEvent: | 
|  | 5606 | case BuiltinType::OCLQueue: | 
|  | 5607 | case BuiltinType::OCLNDRange: | 
|  | 5608 | case BuiltinType::OCLReserveID: | 
| Guy Benyei | 6105419 | 2013-02-07 10:55:47 +0000 | [diff] [blame] | 5609 | case BuiltinType::OCLSampler: | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 5610 | case BuiltinType::Dependent: | 
|  | 5611 | #define BUILTIN_TYPE(KIND, ID) | 
|  | 5612 | #define PLACEHOLDER_TYPE(KIND, ID) \ | 
|  | 5613 | case BuiltinType::KIND: | 
|  | 5614 | #include "clang/AST/BuiltinTypes.def" | 
|  | 5615 | llvm_unreachable("invalid builtin type for @encode"); | 
| David Chisnall | b190a2c | 2010-06-04 01:10:52 +0000 | [diff] [blame] | 5616 | } | 
| David Blaikie | 5a6a020 | 2013-01-09 17:48:41 +0000 | [diff] [blame] | 5617 | llvm_unreachable("invalid BuiltinType::Kind value"); | 
| David Chisnall | b190a2c | 2010-06-04 01:10:52 +0000 | [diff] [blame] | 5618 | } | 
|  | 5619 |  | 
| Douglas Gregor | 8b7d403 | 2011-09-08 17:18:35 +0000 | [diff] [blame] | 5620 | static char ObjCEncodingForEnumType(const ASTContext *C, const EnumType *ET) { | 
|  | 5621 | EnumDecl *Enum = ET->getDecl(); | 
|  | 5622 |  | 
|  | 5623 | // The encoding of an non-fixed enum type is always 'i', regardless of size. | 
|  | 5624 | if (!Enum->isFixed()) | 
|  | 5625 | return 'i'; | 
|  | 5626 |  | 
|  | 5627 | // The encoding of a fixed enum type matches its fixed underlying type. | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 5628 | const BuiltinType *BT = Enum->getIntegerType()->castAs<BuiltinType>(); | 
|  | 5629 | return getObjCEncodingForPrimitiveKind(C, BT->getKind()); | 
| Douglas Gregor | 8b7d403 | 2011-09-08 17:18:35 +0000 | [diff] [blame] | 5630 | } | 
|  | 5631 |  | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 5632 | static void EncodeBitField(const ASTContext *Ctx, std::string& S, | 
| David Chisnall | b190a2c | 2010-06-04 01:10:52 +0000 | [diff] [blame] | 5633 | QualType T, const FieldDecl *FD) { | 
| Richard Smith | caf3390 | 2011-10-10 18:28:20 +0000 | [diff] [blame] | 5634 | assert(FD->isBitField() && "not a bitfield - getObjCEncodingForTypeImpl"); | 
| Fariborz Jahanian | 5c76772 | 2009-01-13 01:18:13 +0000 | [diff] [blame] | 5635 | S += 'b'; | 
| David Chisnall | b190a2c | 2010-06-04 01:10:52 +0000 | [diff] [blame] | 5636 | // The NeXT runtime encodes bit fields as b followed by the number of bits. | 
|  | 5637 | // The GNU runtime requires more information; bitfields are encoded as b, | 
|  | 5638 | // then the offset (in bits) of the first element, then the type of the | 
|  | 5639 | // bitfield, then the size in bits.  For example, in this structure: | 
|  | 5640 | // | 
|  | 5641 | // struct | 
|  | 5642 | // { | 
|  | 5643 | //    int integer; | 
|  | 5644 | //    int flags:2; | 
|  | 5645 | // }; | 
|  | 5646 | // On a 32-bit system, the encoding for flags would be b2 for the NeXT | 
|  | 5647 | // runtime, but b32i2 for the GNU runtime.  The reason for this extra | 
|  | 5648 | // information is not especially sensible, but we're stuck with it for | 
|  | 5649 | // compatibility with GCC, although providing it breaks anything that | 
|  | 5650 | // actually uses runtime introspection and wants to work on both runtimes... | 
| John McCall | 5fb5df9 | 2012-06-20 06:18:46 +0000 | [diff] [blame] | 5651 | if (Ctx->getLangOpts().ObjCRuntime.isGNUFamily()) { | 
| David Chisnall | b190a2c | 2010-06-04 01:10:52 +0000 | [diff] [blame] | 5652 | const RecordDecl *RD = FD->getParent(); | 
|  | 5653 | const ASTRecordLayout &RL = Ctx->getASTRecordLayout(RD); | 
| Eli Friedman | a3c122d | 2011-07-07 01:54:01 +0000 | [diff] [blame] | 5654 | S += llvm::utostr(RL.getFieldOffset(FD->getFieldIndex())); | 
| Douglas Gregor | 8b7d403 | 2011-09-08 17:18:35 +0000 | [diff] [blame] | 5655 | if (const EnumType *ET = T->getAs<EnumType>()) | 
|  | 5656 | S += ObjCEncodingForEnumType(Ctx, ET); | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 5657 | else { | 
|  | 5658 | const BuiltinType *BT = T->castAs<BuiltinType>(); | 
|  | 5659 | S += getObjCEncodingForPrimitiveKind(Ctx, BT->getKind()); | 
|  | 5660 | } | 
| David Chisnall | b190a2c | 2010-06-04 01:10:52 +0000 | [diff] [blame] | 5661 | } | 
| Richard Smith | caf3390 | 2011-10-10 18:28:20 +0000 | [diff] [blame] | 5662 | S += llvm::utostr(FD->getBitWidthValue(*Ctx)); | 
| Fariborz Jahanian | 5c76772 | 2009-01-13 01:18:13 +0000 | [diff] [blame] | 5663 | } | 
|  | 5664 |  | 
| Daniel Dunbar | 07d0785 | 2009-10-18 21:17:35 +0000 | [diff] [blame] | 5665 | // FIXME: Use SmallString for accumulating string. | 
| Daniel Dunbar | 3cd9a29 | 2008-10-17 07:30:50 +0000 | [diff] [blame] | 5666 | void ASTContext::getObjCEncodingForTypeImpl(QualType T, std::string& S, | 
|  | 5667 | bool ExpandPointedToStructures, | 
|  | 5668 | bool ExpandStructures, | 
| Daniel Dunbar | c040ce4 | 2009-04-20 06:37:24 +0000 | [diff] [blame] | 5669 | const FieldDecl *FD, | 
| Fariborz Jahanian | 218c630 | 2009-01-20 19:14:18 +0000 | [diff] [blame] | 5670 | bool OutermostType, | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 5671 | bool EncodingProperty, | 
| Bob Wilson | 5f4e3a7 | 2011-11-30 01:57:58 +0000 | [diff] [blame] | 5672 | bool StructField, | 
|  | 5673 | bool EncodeBlockParameters, | 
| Fariborz Jahanian | d4c1a20 | 2013-02-15 21:14:50 +0000 | [diff] [blame] | 5674 | bool EncodeClassNames, | 
| Fariborz Jahanian | 4bf437e | 2014-08-22 23:17:52 +0000 | [diff] [blame] | 5675 | bool EncodePointerToObjCTypedef, | 
|  | 5676 | QualType *NotEncodedT) const { | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 5677 | CanQualType CT = getCanonicalType(T); | 
|  | 5678 | switch (CT->getTypeClass()) { | 
|  | 5679 | case Type::Builtin: | 
|  | 5680 | case Type::Enum: | 
| Chris Lattner | e7cabb9 | 2009-07-13 00:10:46 +0000 | [diff] [blame] | 5681 | if (FD && FD->isBitField()) | 
| David Chisnall | b190a2c | 2010-06-04 01:10:52 +0000 | [diff] [blame] | 5682 | return EncodeBitField(this, S, T, FD); | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 5683 | if (const BuiltinType *BT = dyn_cast<BuiltinType>(CT)) | 
|  | 5684 | S += getObjCEncodingForPrimitiveKind(this, BT->getKind()); | 
|  | 5685 | else | 
|  | 5686 | S += ObjCEncodingForEnumType(this, cast<EnumType>(CT)); | 
| Chris Lattner | e7cabb9 | 2009-07-13 00:10:46 +0000 | [diff] [blame] | 5687 | return; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5688 |  | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 5689 | case Type::Complex: { | 
|  | 5690 | const ComplexType *CT = T->castAs<ComplexType>(); | 
| Anders Carlsson | 39b2e13 | 2009-04-09 21:55:45 +0000 | [diff] [blame] | 5691 | S += 'j'; | 
| Fariborz Jahanian | 4bf437e | 2014-08-22 23:17:52 +0000 | [diff] [blame] | 5692 | getObjCEncodingForTypeImpl(CT->getElementType(), S, false, false, nullptr); | 
| Chris Lattner | e7cabb9 | 2009-07-13 00:10:46 +0000 | [diff] [blame] | 5693 | return; | 
|  | 5694 | } | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 5695 |  | 
|  | 5696 | case Type::Atomic: { | 
|  | 5697 | const AtomicType *AT = T->castAs<AtomicType>(); | 
|  | 5698 | S += 'A'; | 
| Fariborz Jahanian | 4bf437e | 2014-08-22 23:17:52 +0000 | [diff] [blame] | 5699 | getObjCEncodingForTypeImpl(AT->getValueType(), S, false, false, nullptr); | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 5700 | return; | 
| Fariborz Jahanian | 9ffd706 | 2010-04-13 23:45:47 +0000 | [diff] [blame] | 5701 | } | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 5702 |  | 
|  | 5703 | // encoding for pointer or reference types. | 
|  | 5704 | case Type::Pointer: | 
|  | 5705 | case Type::LValueReference: | 
|  | 5706 | case Type::RValueReference: { | 
|  | 5707 | QualType PointeeTy; | 
|  | 5708 | if (isa<PointerType>(CT)) { | 
|  | 5709 | const PointerType *PT = T->castAs<PointerType>(); | 
|  | 5710 | if (PT->isObjCSelType()) { | 
|  | 5711 | S += ':'; | 
|  | 5712 | return; | 
|  | 5713 | } | 
|  | 5714 | PointeeTy = PT->getPointeeType(); | 
|  | 5715 | } else { | 
|  | 5716 | PointeeTy = T->castAs<ReferenceType>()->getPointeeType(); | 
|  | 5717 | } | 
|  | 5718 |  | 
| Fariborz Jahanian | 0f66a6c | 2008-12-23 19:56:47 +0000 | [diff] [blame] | 5719 | bool isReadOnly = false; | 
|  | 5720 | // For historical/compatibility reasons, the read-only qualifier of the | 
|  | 5721 | // pointee gets emitted _before_ the '^'.  The read-only qualifier of | 
|  | 5722 | // the pointer itself gets ignored, _unless_ we are looking at a typedef! | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5723 | // Also, do not emit the 'r' for anything but the outermost type! | 
| Mike Stump | 212005c | 2009-07-22 18:58:19 +0000 | [diff] [blame] | 5724 | if (isa<TypedefType>(T.getTypePtr())) { | 
| Fariborz Jahanian | 0f66a6c | 2008-12-23 19:56:47 +0000 | [diff] [blame] | 5725 | if (OutermostType && T.isConstQualified()) { | 
|  | 5726 | isReadOnly = true; | 
|  | 5727 | S += 'r'; | 
|  | 5728 | } | 
| Mike Stump | e9c6ffc | 2009-07-31 02:02:20 +0000 | [diff] [blame] | 5729 | } else if (OutermostType) { | 
| Fariborz Jahanian | 0f66a6c | 2008-12-23 19:56:47 +0000 | [diff] [blame] | 5730 | QualType P = PointeeTy; | 
| Ted Kremenek | c23c7e6 | 2009-07-29 21:53:49 +0000 | [diff] [blame] | 5731 | while (P->getAs<PointerType>()) | 
|  | 5732 | P = P->getAs<PointerType>()->getPointeeType(); | 
| Fariborz Jahanian | 0f66a6c | 2008-12-23 19:56:47 +0000 | [diff] [blame] | 5733 | if (P.isConstQualified()) { | 
|  | 5734 | isReadOnly = true; | 
|  | 5735 | S += 'r'; | 
|  | 5736 | } | 
|  | 5737 | } | 
|  | 5738 | if (isReadOnly) { | 
|  | 5739 | // Another legacy compatibility encoding. Some ObjC qualifier and type | 
|  | 5740 | // combinations need to be rearranged. | 
|  | 5741 | // Rewrite "in const" from "nr" to "rn" | 
| Chris Lattner | 0e62c1c | 2011-07-23 10:55:15 +0000 | [diff] [blame] | 5742 | if (StringRef(S).endswith("nr")) | 
| Benjamin Kramer | 2e3197e | 2010-04-27 17:12:11 +0000 | [diff] [blame] | 5743 | S.replace(S.end()-2, S.end(), "rn"); | 
| Fariborz Jahanian | 0f66a6c | 2008-12-23 19:56:47 +0000 | [diff] [blame] | 5744 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5745 |  | 
| Anders Carlsson | d849982 | 2007-10-29 05:01:08 +0000 | [diff] [blame] | 5746 | if (PointeeTy->isCharType()) { | 
|  | 5747 | // char pointer types should be encoded as '*' unless it is a | 
|  | 5748 | // type that has been typedef'd to 'BOOL'. | 
| Anders Carlsson | 18acd44 | 2007-10-29 06:33:42 +0000 | [diff] [blame] | 5749 | if (!isTypeTypedefedAsBOOL(PointeeTy)) { | 
| Anders Carlsson | d849982 | 2007-10-29 05:01:08 +0000 | [diff] [blame] | 5750 | S += '*'; | 
|  | 5751 | return; | 
|  | 5752 | } | 
| Ted Kremenek | c23c7e6 | 2009-07-29 21:53:49 +0000 | [diff] [blame] | 5753 | } else if (const RecordType *RTy = PointeeTy->getAs<RecordType>()) { | 
| Steve Naroff | 3de6b70 | 2009-07-22 17:14:51 +0000 | [diff] [blame] | 5754 | // GCC binary compat: Need to convert "struct objc_class *" to "#". | 
|  | 5755 | if (RTy->getDecl()->getIdentifier() == &Idents.get("objc_class")) { | 
|  | 5756 | S += '#'; | 
|  | 5757 | return; | 
|  | 5758 | } | 
|  | 5759 | // GCC binary compat: Need to convert "struct objc_object *" to "@". | 
|  | 5760 | if (RTy->getDecl()->getIdentifier() == &Idents.get("objc_object")) { | 
|  | 5761 | S += '@'; | 
|  | 5762 | return; | 
|  | 5763 | } | 
|  | 5764 | // fall through... | 
| Anders Carlsson | d849982 | 2007-10-29 05:01:08 +0000 | [diff] [blame] | 5765 | } | 
| Anders Carlsson | d849982 | 2007-10-29 05:01:08 +0000 | [diff] [blame] | 5766 | S += '^'; | 
| Fariborz Jahanian | 0f66a6c | 2008-12-23 19:56:47 +0000 | [diff] [blame] | 5767 | getLegacyIntegralTypeEncoding(PointeeTy); | 
|  | 5768 |  | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5769 | getObjCEncodingForTypeImpl(PointeeTy, S, false, ExpandPointedToStructures, | 
| Fariborz Jahanian | 4bf437e | 2014-08-22 23:17:52 +0000 | [diff] [blame] | 5770 | nullptr, false, false, false, false, false, false, | 
|  | 5771 | NotEncodedT); | 
| Chris Lattner | e7cabb9 | 2009-07-13 00:10:46 +0000 | [diff] [blame] | 5772 | return; | 
|  | 5773 | } | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 5774 |  | 
|  | 5775 | case Type::ConstantArray: | 
|  | 5776 | case Type::IncompleteArray: | 
|  | 5777 | case Type::VariableArray: { | 
|  | 5778 | const ArrayType *AT = cast<ArrayType>(CT); | 
|  | 5779 |  | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 5780 | if (isa<IncompleteArrayType>(AT) && !StructField) { | 
| Anders Carlsson | d05f44b | 2009-02-22 01:38:57 +0000 | [diff] [blame] | 5781 | // Incomplete arrays are encoded as a pointer to the array element. | 
|  | 5782 | S += '^'; | 
|  | 5783 |  | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5784 | getObjCEncodingForTypeImpl(AT->getElementType(), S, | 
| Anders Carlsson | d05f44b | 2009-02-22 01:38:57 +0000 | [diff] [blame] | 5785 | false, ExpandStructures, FD); | 
|  | 5786 | } else { | 
|  | 5787 | S += '['; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5788 |  | 
| Fariborz Jahanian | f0dc11a | 2013-06-04 16:04:37 +0000 | [diff] [blame] | 5789 | if (const ConstantArrayType *CAT = dyn_cast<ConstantArrayType>(AT)) | 
|  | 5790 | S += llvm::utostr(CAT->getSize().getZExtValue()); | 
|  | 5791 | else { | 
| Anders Carlsson | d05f44b | 2009-02-22 01:38:57 +0000 | [diff] [blame] | 5792 | //Variable length arrays are encoded as a regular array with 0 elements. | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 5793 | assert((isa<VariableArrayType>(AT) || isa<IncompleteArrayType>(AT)) && | 
|  | 5794 | "Unknown array type!"); | 
| Anders Carlsson | d05f44b | 2009-02-22 01:38:57 +0000 | [diff] [blame] | 5795 | S += '0'; | 
|  | 5796 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5797 |  | 
|  | 5798 | getObjCEncodingForTypeImpl(AT->getElementType(), S, | 
| Fariborz Jahanian | 4bf437e | 2014-08-22 23:17:52 +0000 | [diff] [blame] | 5799 | false, ExpandStructures, FD, | 
|  | 5800 | false, false, false, false, false, false, | 
|  | 5801 | NotEncodedT); | 
| Anders Carlsson | d05f44b | 2009-02-22 01:38:57 +0000 | [diff] [blame] | 5802 | S += ']'; | 
|  | 5803 | } | 
| Chris Lattner | e7cabb9 | 2009-07-13 00:10:46 +0000 | [diff] [blame] | 5804 | return; | 
|  | 5805 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5806 |  | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 5807 | case Type::FunctionNoProto: | 
|  | 5808 | case Type::FunctionProto: | 
| Anders Carlsson | df4cc61 | 2007-10-30 00:06:20 +0000 | [diff] [blame] | 5809 | S += '?'; | 
| Chris Lattner | e7cabb9 | 2009-07-13 00:10:46 +0000 | [diff] [blame] | 5810 | return; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5811 |  | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 5812 | case Type::Record: { | 
|  | 5813 | RecordDecl *RDecl = cast<RecordType>(CT)->getDecl(); | 
| Daniel Dunbar | ff3c674 | 2008-10-17 16:17:37 +0000 | [diff] [blame] | 5814 | S += RDecl->isUnion() ? '(' : '{'; | 
| Daniel Dunbar | 40cac77 | 2008-10-17 06:22:57 +0000 | [diff] [blame] | 5815 | // Anonymous structures print as '?' | 
|  | 5816 | if (const IdentifierInfo *II = RDecl->getIdentifier()) { | 
|  | 5817 | S += II->getName(); | 
| Fariborz Jahanian | c515820 | 2010-05-07 00:28:49 +0000 | [diff] [blame] | 5818 | if (ClassTemplateSpecializationDecl *Spec | 
|  | 5819 | = dyn_cast<ClassTemplateSpecializationDecl>(RDecl)) { | 
|  | 5820 | const TemplateArgumentList &TemplateArgs = Spec->getTemplateArgs(); | 
| Benjamin Kramer | 9170e91 | 2013-02-22 15:46:01 +0000 | [diff] [blame] | 5821 | llvm::raw_string_ostream OS(S); | 
|  | 5822 | TemplateSpecializationType::PrintTemplateArgumentList(OS, | 
| David Majnemer | 6fbeee3 | 2016-07-07 04:43:07 +0000 | [diff] [blame] | 5823 | TemplateArgs.asArray(), | 
| Douglas Gregor | c0b0728 | 2011-09-27 22:38:19 +0000 | [diff] [blame] | 5824 | (*this).getPrintingPolicy()); | 
| Fariborz Jahanian | c515820 | 2010-05-07 00:28:49 +0000 | [diff] [blame] | 5825 | } | 
| Daniel Dunbar | 40cac77 | 2008-10-17 06:22:57 +0000 | [diff] [blame] | 5826 | } else { | 
|  | 5827 | S += '?'; | 
|  | 5828 | } | 
| Daniel Dunbar | fc1066d | 2008-10-17 20:21:44 +0000 | [diff] [blame] | 5829 | if (ExpandStructures) { | 
| Fariborz Jahanian | 0a71ad2 | 2008-01-22 22:44:46 +0000 | [diff] [blame] | 5830 | S += '='; | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 5831 | if (!RDecl->isUnion()) { | 
| Fariborz Jahanian | 4bf437e | 2014-08-22 23:17:52 +0000 | [diff] [blame] | 5832 | getObjCEncodingForStructureImpl(RDecl, S, FD, true, NotEncodedT); | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 5833 | } else { | 
| Aaron Ballman | e8a8bae | 2014-03-08 20:12:42 +0000 | [diff] [blame] | 5834 | for (const auto *Field : RDecl->fields()) { | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 5835 | if (FD) { | 
|  | 5836 | S += '"'; | 
|  | 5837 | S += Field->getNameAsString(); | 
|  | 5838 | S += '"'; | 
|  | 5839 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5840 |  | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 5841 | // Special case bit-fields. | 
|  | 5842 | if (Field->isBitField()) { | 
|  | 5843 | getObjCEncodingForTypeImpl(Field->getType(), S, false, true, | 
| Aaron Ballman | e8a8bae | 2014-03-08 20:12:42 +0000 | [diff] [blame] | 5844 | Field); | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 5845 | } else { | 
|  | 5846 | QualType qt = Field->getType(); | 
|  | 5847 | getLegacyIntegralTypeEncoding(qt); | 
|  | 5848 | getObjCEncodingForTypeImpl(qt, S, false, true, | 
|  | 5849 | FD, /*OutermostType*/false, | 
|  | 5850 | /*EncodingProperty*/false, | 
| Fariborz Jahanian | 4bf437e | 2014-08-22 23:17:52 +0000 | [diff] [blame] | 5851 | /*StructField*/true, | 
|  | 5852 | false, false, false, NotEncodedT); | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 5853 | } | 
| Daniel Dunbar | ff3c674 | 2008-10-17 16:17:37 +0000 | [diff] [blame] | 5854 | } | 
| Fariborz Jahanian | 0a71ad2 | 2008-01-22 22:44:46 +0000 | [diff] [blame] | 5855 | } | 
| Fariborz Jahanian | bc92fd7 | 2007-11-13 23:21:38 +0000 | [diff] [blame] | 5856 | } | 
| Daniel Dunbar | ff3c674 | 2008-10-17 16:17:37 +0000 | [diff] [blame] | 5857 | S += RDecl->isUnion() ? ')' : '}'; | 
| Chris Lattner | e7cabb9 | 2009-07-13 00:10:46 +0000 | [diff] [blame] | 5858 | return; | 
|  | 5859 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5860 |  | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 5861 | case Type::BlockPointer: { | 
|  | 5862 | const BlockPointerType *BT = T->castAs<BlockPointerType>(); | 
| Steve Naroff | 49140cb | 2009-02-02 18:24:29 +0000 | [diff] [blame] | 5863 | S += "@?"; // Unlike a pointer-to-function, which is "^?". | 
| Bob Wilson | 5f4e3a7 | 2011-11-30 01:57:58 +0000 | [diff] [blame] | 5864 | if (EncodeBlockParameters) { | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 5865 | const FunctionType *FT = BT->getPointeeType()->castAs<FunctionType>(); | 
| Bob Wilson | 5f4e3a7 | 2011-11-30 01:57:58 +0000 | [diff] [blame] | 5866 |  | 
|  | 5867 | S += '<'; | 
|  | 5868 | // Block return type | 
| Alp Toker | 314cc81 | 2014-01-25 16:55:45 +0000 | [diff] [blame] | 5869 | getObjCEncodingForTypeImpl( | 
|  | 5870 | FT->getReturnType(), S, ExpandPointedToStructures, ExpandStructures, | 
|  | 5871 | FD, false /* OutermostType */, EncodingProperty, | 
| Fariborz Jahanian | 4bf437e | 2014-08-22 23:17:52 +0000 | [diff] [blame] | 5872 | false /* StructField */, EncodeBlockParameters, EncodeClassNames, false, | 
|  | 5873 | NotEncodedT); | 
| Bob Wilson | 5f4e3a7 | 2011-11-30 01:57:58 +0000 | [diff] [blame] | 5874 | // Block self | 
|  | 5875 | S += "@?"; | 
|  | 5876 | // Block parameters | 
|  | 5877 | if (const FunctionProtoType *FPT = dyn_cast<FunctionProtoType>(FT)) { | 
| Aaron Ballman | 40bd0aa | 2014-03-17 15:23:01 +0000 | [diff] [blame] | 5878 | for (const auto &I : FPT->param_types()) | 
|  | 5879 | getObjCEncodingForTypeImpl( | 
|  | 5880 | I, S, ExpandPointedToStructures, ExpandStructures, FD, | 
|  | 5881 | false /* OutermostType */, EncodingProperty, | 
| Fariborz Jahanian | 4bf437e | 2014-08-22 23:17:52 +0000 | [diff] [blame] | 5882 | false /* StructField */, EncodeBlockParameters, EncodeClassNames, | 
|  | 5883 | false, NotEncodedT); | 
| Bob Wilson | 5f4e3a7 | 2011-11-30 01:57:58 +0000 | [diff] [blame] | 5884 | } | 
|  | 5885 | S += '>'; | 
|  | 5886 | } | 
| Chris Lattner | e7cabb9 | 2009-07-13 00:10:46 +0000 | [diff] [blame] | 5887 | return; | 
|  | 5888 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5889 |  | 
| Fariborz Jahanian | 33fa962 | 2014-01-28 20:41:15 +0000 | [diff] [blame] | 5890 | case Type::ObjCObject: { | 
|  | 5891 | // hack to match legacy encoding of *id and *Class | 
|  | 5892 | QualType Ty = getObjCObjectPointerType(CT); | 
|  | 5893 | if (Ty->isObjCIdType()) { | 
|  | 5894 | S += "{objc_object=}"; | 
|  | 5895 | return; | 
|  | 5896 | } | 
|  | 5897 | else if (Ty->isObjCClassType()) { | 
|  | 5898 | S += "{objc_class=}"; | 
|  | 5899 | return; | 
|  | 5900 | } | 
|  | 5901 | } | 
|  | 5902 |  | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 5903 | case Type::ObjCInterface: { | 
|  | 5904 | // Ignore protocol qualifiers when mangling at this level. | 
| Fariborz Jahanian | 1d35f12 | 2008-12-19 23:34:38 +0000 | [diff] [blame] | 5905 | // @encode(class_name) | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 5906 | ObjCInterfaceDecl *OI = T->castAs<ObjCObjectType>()->getInterface(); | 
| Fariborz Jahanian | 1d35f12 | 2008-12-19 23:34:38 +0000 | [diff] [blame] | 5907 | S += '{'; | 
| Douglas Gregor | b32684e | 2015-06-16 21:04:55 +0000 | [diff] [blame] | 5908 | S += OI->getObjCRuntimeNameAsString(); | 
| Akira Hatanaka | fd0fb20 | 2016-08-17 19:42:22 +0000 | [diff] [blame] | 5909 | if (ExpandStructures) { | 
|  | 5910 | S += '='; | 
|  | 5911 | SmallVector<const ObjCIvarDecl*, 32> Ivars; | 
|  | 5912 | DeepCollectObjCIvars(OI, true, Ivars); | 
|  | 5913 | for (unsigned i = 0, e = Ivars.size(); i != e; ++i) { | 
|  | 5914 | const FieldDecl *Field = cast<FieldDecl>(Ivars[i]); | 
|  | 5915 | if (Field->isBitField()) | 
|  | 5916 | getObjCEncodingForTypeImpl(Field->getType(), S, false, true, Field); | 
|  | 5917 | else | 
|  | 5918 | getObjCEncodingForTypeImpl(Field->getType(), S, false, true, FD, | 
|  | 5919 | false, false, false, false, false, | 
|  | 5920 | EncodePointerToObjCTypedef, | 
|  | 5921 | NotEncodedT); | 
|  | 5922 | } | 
| Fariborz Jahanian | 1d35f12 | 2008-12-19 23:34:38 +0000 | [diff] [blame] | 5923 | } | 
|  | 5924 | S += '}'; | 
| Chris Lattner | e7cabb9 | 2009-07-13 00:10:46 +0000 | [diff] [blame] | 5925 | return; | 
| Fariborz Jahanian | 1d35f12 | 2008-12-19 23:34:38 +0000 | [diff] [blame] | 5926 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5927 |  | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 5928 | case Type::ObjCObjectPointer: { | 
|  | 5929 | const ObjCObjectPointerType *OPT = T->castAs<ObjCObjectPointerType>(); | 
| Steve Naroff | 7cae42b | 2009-07-10 23:34:53 +0000 | [diff] [blame] | 5930 | if (OPT->isObjCIdType()) { | 
|  | 5931 | S += '@'; | 
|  | 5932 | return; | 
| Chris Lattner | e7cabb9 | 2009-07-13 00:10:46 +0000 | [diff] [blame] | 5933 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5934 |  | 
| Steve Naroff | f0c8611 | 2009-10-28 22:03:49 +0000 | [diff] [blame] | 5935 | if (OPT->isObjCClassType() || OPT->isObjCQualifiedClassType()) { | 
|  | 5936 | // FIXME: Consider if we need to output qualifiers for 'Class<p>'. | 
|  | 5937 | // Since this is a binary compatibility issue, need to consult with runtime | 
|  | 5938 | // folks. Fortunately, this is a *very* obsure construct. | 
| Steve Naroff | 7cae42b | 2009-07-10 23:34:53 +0000 | [diff] [blame] | 5939 | S += '#'; | 
|  | 5940 | return; | 
| Chris Lattner | e7cabb9 | 2009-07-13 00:10:46 +0000 | [diff] [blame] | 5941 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5942 |  | 
| Chris Lattner | e7cabb9 | 2009-07-13 00:10:46 +0000 | [diff] [blame] | 5943 | if (OPT->isObjCQualifiedIdType()) { | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5944 | getObjCEncodingForTypeImpl(getObjCIdType(), S, | 
| Steve Naroff | 7cae42b | 2009-07-10 23:34:53 +0000 | [diff] [blame] | 5945 | ExpandPointedToStructures, | 
|  | 5946 | ExpandStructures, FD); | 
| Bob Wilson | 5f4e3a7 | 2011-11-30 01:57:58 +0000 | [diff] [blame] | 5947 | if (FD || EncodingProperty || EncodeClassNames) { | 
| Steve Naroff | 7cae42b | 2009-07-10 23:34:53 +0000 | [diff] [blame] | 5948 | // Note that we do extended encoding of protocol qualifer list | 
|  | 5949 | // Only when doing ivar or property encoding. | 
| Steve Naroff | 7cae42b | 2009-07-10 23:34:53 +0000 | [diff] [blame] | 5950 | S += '"'; | 
| Aaron Ballman | 8373146 | 2014-03-17 16:14:00 +0000 | [diff] [blame] | 5951 | for (const auto *I : OPT->quals()) { | 
| Steve Naroff | 7cae42b | 2009-07-10 23:34:53 +0000 | [diff] [blame] | 5952 | S += '<'; | 
| Douglas Gregor | b32684e | 2015-06-16 21:04:55 +0000 | [diff] [blame] | 5953 | S += I->getObjCRuntimeNameAsString(); | 
| Steve Naroff | 7cae42b | 2009-07-10 23:34:53 +0000 | [diff] [blame] | 5954 | S += '>'; | 
|  | 5955 | } | 
|  | 5956 | S += '"'; | 
|  | 5957 | } | 
|  | 5958 | return; | 
| Chris Lattner | e7cabb9 | 2009-07-13 00:10:46 +0000 | [diff] [blame] | 5959 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5960 |  | 
| Chris Lattner | e7cabb9 | 2009-07-13 00:10:46 +0000 | [diff] [blame] | 5961 | QualType PointeeTy = OPT->getPointeeType(); | 
|  | 5962 | if (!EncodingProperty && | 
| Fariborz Jahanian | d4c1a20 | 2013-02-15 21:14:50 +0000 | [diff] [blame] | 5963 | isa<TypedefType>(PointeeTy.getTypePtr()) && | 
|  | 5964 | !EncodePointerToObjCTypedef) { | 
| Chris Lattner | e7cabb9 | 2009-07-13 00:10:46 +0000 | [diff] [blame] | 5965 | // Another historical/compatibility reason. | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5966 | // We encode the underlying type which comes out as | 
| Chris Lattner | e7cabb9 | 2009-07-13 00:10:46 +0000 | [diff] [blame] | 5967 | // {...}; | 
|  | 5968 | S += '^'; | 
| Fariborz Jahanian | 88890e7 | 2013-07-12 16:19:11 +0000 | [diff] [blame] | 5969 | if (FD && OPT->getInterfaceDecl()) { | 
| Fariborz Jahanian | c682ef5 | 2013-07-12 16:41:56 +0000 | [diff] [blame] | 5970 | // Prevent recursive encoding of fields in some rare cases. | 
| Fariborz Jahanian | 88890e7 | 2013-07-12 16:19:11 +0000 | [diff] [blame] | 5971 | ObjCInterfaceDecl *OI = OPT->getInterfaceDecl(); | 
|  | 5972 | SmallVector<const ObjCIvarDecl*, 32> Ivars; | 
|  | 5973 | DeepCollectObjCIvars(OI, true, Ivars); | 
|  | 5974 | for (unsigned i = 0, e = Ivars.size(); i != e; ++i) { | 
|  | 5975 | if (cast<FieldDecl>(Ivars[i]) == FD) { | 
|  | 5976 | S += '{'; | 
| Douglas Gregor | b32684e | 2015-06-16 21:04:55 +0000 | [diff] [blame] | 5977 | S += OI->getObjCRuntimeNameAsString(); | 
| Fariborz Jahanian | 88890e7 | 2013-07-12 16:19:11 +0000 | [diff] [blame] | 5978 | S += '}'; | 
|  | 5979 | return; | 
|  | 5980 | } | 
|  | 5981 | } | 
|  | 5982 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 5983 | getObjCEncodingForTypeImpl(PointeeTy, S, | 
|  | 5984 | false, ExpandPointedToStructures, | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 5985 | nullptr, | 
| Fariborz Jahanian | d4c1a20 | 2013-02-15 21:14:50 +0000 | [diff] [blame] | 5986 | false, false, false, false, false, | 
|  | 5987 | /*EncodePointerToObjCTypedef*/true); | 
| Steve Naroff | 7cae42b | 2009-07-10 23:34:53 +0000 | [diff] [blame] | 5988 | return; | 
|  | 5989 | } | 
| Chris Lattner | e7cabb9 | 2009-07-13 00:10:46 +0000 | [diff] [blame] | 5990 |  | 
|  | 5991 | S += '@'; | 
| Bob Wilson | 5f4e3a7 | 2011-11-30 01:57:58 +0000 | [diff] [blame] | 5992 | if (OPT->getInterfaceDecl() && | 
|  | 5993 | (FD || EncodingProperty || EncodeClassNames)) { | 
| Chris Lattner | e7cabb9 | 2009-07-13 00:10:46 +0000 | [diff] [blame] | 5994 | S += '"'; | 
| Douglas Gregor | b32684e | 2015-06-16 21:04:55 +0000 | [diff] [blame] | 5995 | S += OPT->getInterfaceDecl()->getObjCRuntimeNameAsString(); | 
| Aaron Ballman | 8373146 | 2014-03-17 16:14:00 +0000 | [diff] [blame] | 5996 | for (const auto *I : OPT->quals()) { | 
| Chris Lattner | e7cabb9 | 2009-07-13 00:10:46 +0000 | [diff] [blame] | 5997 | S += '<'; | 
| Douglas Gregor | b32684e | 2015-06-16 21:04:55 +0000 | [diff] [blame] | 5998 | S += I->getObjCRuntimeNameAsString(); | 
| Chris Lattner | e7cabb9 | 2009-07-13 00:10:46 +0000 | [diff] [blame] | 5999 | S += '>'; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 6000 | } | 
| Chris Lattner | e7cabb9 | 2009-07-13 00:10:46 +0000 | [diff] [blame] | 6001 | S += '"'; | 
|  | 6002 | } | 
|  | 6003 | return; | 
|  | 6004 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 6005 |  | 
| John McCall | a9e6e8d | 2010-05-17 23:56:34 +0000 | [diff] [blame] | 6006 | // gcc just blithely ignores member pointers. | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 6007 | // FIXME: we shoul do better than that.  'M' is available. | 
|  | 6008 | case Type::MemberPointer: | 
| Fariborz Jahanian | 4bf437e | 2014-08-22 23:17:52 +0000 | [diff] [blame] | 6009 | // This matches gcc's encoding, even though technically it is insufficient. | 
|  | 6010 | //FIXME. We should do a better job than gcc. | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 6011 | case Type::Vector: | 
|  | 6012 | case Type::ExtVector: | 
| Fariborz Jahanian | 4bf437e | 2014-08-22 23:17:52 +0000 | [diff] [blame] | 6013 | // Until we have a coherent encoding of these three types, issue warning. | 
|  | 6014 | { if (NotEncodedT) | 
|  | 6015 | *NotEncodedT = T; | 
|  | 6016 | return; | 
|  | 6017 | } | 
|  | 6018 |  | 
|  | 6019 | // We could see an undeduced auto type here during error recovery. | 
|  | 6020 | // Just ignore it. | 
| Richard Smith | 27d807c | 2013-04-30 13:56:41 +0000 | [diff] [blame] | 6021 | case Type::Auto: | 
| Richard Smith | 27d807c | 2013-04-30 13:56:41 +0000 | [diff] [blame] | 6022 | return; | 
|  | 6023 |  | 
| Xiuli Pan | 9c14e28 | 2016-01-09 12:53:17 +0000 | [diff] [blame] | 6024 | case Type::Pipe: | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 6025 | #define ABSTRACT_TYPE(KIND, BASE) | 
|  | 6026 | #define TYPE(KIND, BASE) | 
|  | 6027 | #define DEPENDENT_TYPE(KIND, BASE) \ | 
|  | 6028 | case Type::KIND: | 
|  | 6029 | #define NON_CANONICAL_TYPE(KIND, BASE) \ | 
|  | 6030 | case Type::KIND: | 
|  | 6031 | #define NON_CANONICAL_UNLESS_DEPENDENT_TYPE(KIND, BASE) \ | 
|  | 6032 | case Type::KIND: | 
|  | 6033 | #include "clang/AST/TypeNodes.def" | 
|  | 6034 | llvm_unreachable("@encode for dependent type!"); | 
| Fariborz Jahanian | e0587be | 2010-10-07 21:25:25 +0000 | [diff] [blame] | 6035 | } | 
| John McCall | 393a78d | 2012-12-20 02:45:14 +0000 | [diff] [blame] | 6036 | llvm_unreachable("bad type kind!"); | 
| Anders Carlsson | d849982 | 2007-10-29 05:01:08 +0000 | [diff] [blame] | 6037 | } | 
|  | 6038 |  | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 6039 | void ASTContext::getObjCEncodingForStructureImpl(RecordDecl *RDecl, | 
|  | 6040 | std::string &S, | 
|  | 6041 | const FieldDecl *FD, | 
| Fariborz Jahanian | 4bf437e | 2014-08-22 23:17:52 +0000 | [diff] [blame] | 6042 | bool includeVBases, | 
|  | 6043 | QualType *NotEncodedT) const { | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 6044 | assert(RDecl && "Expected non-null RecordDecl"); | 
|  | 6045 | assert(!RDecl->isUnion() && "Should not be called for unions"); | 
| Argyrios Kyrtzidis | 785705b | 2016-01-16 00:20:02 +0000 | [diff] [blame] | 6046 | if (!RDecl->getDefinition() || RDecl->getDefinition()->isInvalidDecl()) | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 6047 | return; | 
|  | 6048 |  | 
|  | 6049 | CXXRecordDecl *CXXRec = dyn_cast<CXXRecordDecl>(RDecl); | 
|  | 6050 | std::multimap<uint64_t, NamedDecl *> FieldOrBaseOffsets; | 
|  | 6051 | const ASTRecordLayout &layout = getASTRecordLayout(RDecl); | 
|  | 6052 |  | 
|  | 6053 | if (CXXRec) { | 
| Aaron Ballman | 574705e | 2014-03-13 15:41:46 +0000 | [diff] [blame] | 6054 | for (const auto &BI : CXXRec->bases()) { | 
|  | 6055 | if (!BI.isVirtual()) { | 
|  | 6056 | CXXRecordDecl *base = BI.getType()->getAsCXXRecordDecl(); | 
| Argyrios Kyrtzidis | 95a76f3 | 2011-06-17 23:19:38 +0000 | [diff] [blame] | 6057 | if (base->isEmpty()) | 
|  | 6058 | continue; | 
| Benjamin Kramer | 2ef3031 | 2012-07-04 18:45:14 +0000 | [diff] [blame] | 6059 | uint64_t offs = toBits(layout.getBaseClassOffset(base)); | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 6060 | FieldOrBaseOffsets.insert(FieldOrBaseOffsets.upper_bound(offs), | 
|  | 6061 | std::make_pair(offs, base)); | 
|  | 6062 | } | 
|  | 6063 | } | 
|  | 6064 | } | 
|  | 6065 |  | 
|  | 6066 | unsigned i = 0; | 
| Hans Wennborg | a302cd9 | 2014-08-21 16:06:57 +0000 | [diff] [blame] | 6067 | for (auto *Field : RDecl->fields()) { | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 6068 | uint64_t offs = layout.getFieldOffset(i); | 
|  | 6069 | FieldOrBaseOffsets.insert(FieldOrBaseOffsets.upper_bound(offs), | 
| Hans Wennborg | a302cd9 | 2014-08-21 16:06:57 +0000 | [diff] [blame] | 6070 | std::make_pair(offs, Field)); | 
|  | 6071 | ++i; | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 6072 | } | 
|  | 6073 |  | 
|  | 6074 | if (CXXRec && includeVBases) { | 
| Aaron Ballman | 445a939 | 2014-03-13 16:15:17 +0000 | [diff] [blame] | 6075 | for (const auto &BI : CXXRec->vbases()) { | 
|  | 6076 | CXXRecordDecl *base = BI.getType()->getAsCXXRecordDecl(); | 
| Argyrios Kyrtzidis | 95a76f3 | 2011-06-17 23:19:38 +0000 | [diff] [blame] | 6077 | if (base->isEmpty()) | 
|  | 6078 | continue; | 
| Benjamin Kramer | 2ef3031 | 2012-07-04 18:45:14 +0000 | [diff] [blame] | 6079 | uint64_t offs = toBits(layout.getVBaseClassOffset(base)); | 
| Eli Friedman | 1d24af8 | 2013-09-18 01:59:16 +0000 | [diff] [blame] | 6080 | if (offs >= uint64_t(toBits(layout.getNonVirtualSize())) && | 
|  | 6081 | FieldOrBaseOffsets.find(offs) == FieldOrBaseOffsets.end()) | 
| Argyrios Kyrtzidis | 81c0b5c | 2011-09-26 18:14:24 +0000 | [diff] [blame] | 6082 | FieldOrBaseOffsets.insert(FieldOrBaseOffsets.end(), | 
|  | 6083 | std::make_pair(offs, base)); | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 6084 | } | 
|  | 6085 | } | 
|  | 6086 |  | 
|  | 6087 | CharUnits size; | 
|  | 6088 | if (CXXRec) { | 
|  | 6089 | size = includeVBases ? layout.getSize() : layout.getNonVirtualSize(); | 
|  | 6090 | } else { | 
|  | 6091 | size = layout.getSize(); | 
|  | 6092 | } | 
|  | 6093 |  | 
| Fariborz Jahanian | f1a66de | 2014-01-07 01:02:50 +0000 | [diff] [blame] | 6094 | #ifndef NDEBUG | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 6095 | uint64_t CurOffs = 0; | 
| Fariborz Jahanian | f1a66de | 2014-01-07 01:02:50 +0000 | [diff] [blame] | 6096 | #endif | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 6097 | std::multimap<uint64_t, NamedDecl *>::iterator | 
|  | 6098 | CurLayObj = FieldOrBaseOffsets.begin(); | 
|  | 6099 |  | 
| Douglas Gregor | f5d6c74 | 2012-04-27 22:30:01 +0000 | [diff] [blame] | 6100 | if (CXXRec && CXXRec->isDynamicClass() && | 
|  | 6101 | (CurLayObj == FieldOrBaseOffsets.end() || CurLayObj->first != 0)) { | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 6102 | if (FD) { | 
|  | 6103 | S += "\"_vptr$"; | 
|  | 6104 | std::string recname = CXXRec->getNameAsString(); | 
|  | 6105 | if (recname.empty()) recname = "?"; | 
|  | 6106 | S += recname; | 
|  | 6107 | S += '"'; | 
|  | 6108 | } | 
|  | 6109 | S += "^^?"; | 
| Fariborz Jahanian | f1a66de | 2014-01-07 01:02:50 +0000 | [diff] [blame] | 6110 | #ifndef NDEBUG | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 6111 | CurOffs += getTypeSize(VoidPtrTy); | 
| Fariborz Jahanian | f1a66de | 2014-01-07 01:02:50 +0000 | [diff] [blame] | 6112 | #endif | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 6113 | } | 
|  | 6114 |  | 
|  | 6115 | if (!RDecl->hasFlexibleArrayMember()) { | 
|  | 6116 | // Mark the end of the structure. | 
|  | 6117 | uint64_t offs = toBits(size); | 
|  | 6118 | FieldOrBaseOffsets.insert(FieldOrBaseOffsets.upper_bound(offs), | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 6119 | std::make_pair(offs, nullptr)); | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 6120 | } | 
|  | 6121 |  | 
|  | 6122 | for (; CurLayObj != FieldOrBaseOffsets.end(); ++CurLayObj) { | 
| Fariborz Jahanian | f1a66de | 2014-01-07 01:02:50 +0000 | [diff] [blame] | 6123 | #ifndef NDEBUG | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 6124 | assert(CurOffs <= CurLayObj->first); | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 6125 | if (CurOffs < CurLayObj->first) { | 
|  | 6126 | uint64_t padding = CurLayObj->first - CurOffs; | 
|  | 6127 | // FIXME: There doesn't seem to be a way to indicate in the encoding that | 
|  | 6128 | // packing/alignment of members is different that normal, in which case | 
|  | 6129 | // the encoding will be out-of-sync with the real layout. | 
|  | 6130 | // If the runtime switches to just consider the size of types without | 
|  | 6131 | // taking into account alignment, we could make padding explicit in the | 
|  | 6132 | // encoding (e.g. using arrays of chars). The encoding strings would be | 
|  | 6133 | // longer then though. | 
|  | 6134 | CurOffs += padding; | 
|  | 6135 | } | 
| Fariborz Jahanian | f1a66de | 2014-01-07 01:02:50 +0000 | [diff] [blame] | 6136 | #endif | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 6137 |  | 
|  | 6138 | NamedDecl *dcl = CurLayObj->second; | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 6139 | if (!dcl) | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 6140 | break; // reached end of structure. | 
|  | 6141 |  | 
|  | 6142 | if (CXXRecordDecl *base = dyn_cast<CXXRecordDecl>(dcl)) { | 
|  | 6143 | // We expand the bases without their virtual bases since those are going | 
|  | 6144 | // in the initial structure. Note that this differs from gcc which | 
|  | 6145 | // expands virtual bases each time one is encountered in the hierarchy, | 
|  | 6146 | // making the encoding type bigger than it really is. | 
| Fariborz Jahanian | 4bf437e | 2014-08-22 23:17:52 +0000 | [diff] [blame] | 6147 | getObjCEncodingForStructureImpl(base, S, FD, /*includeVBases*/false, | 
|  | 6148 | NotEncodedT); | 
| Argyrios Kyrtzidis | 95a76f3 | 2011-06-17 23:19:38 +0000 | [diff] [blame] | 6149 | assert(!base->isEmpty()); | 
| Fariborz Jahanian | f1a66de | 2014-01-07 01:02:50 +0000 | [diff] [blame] | 6150 | #ifndef NDEBUG | 
| Argyrios Kyrtzidis | 95a76f3 | 2011-06-17 23:19:38 +0000 | [diff] [blame] | 6151 | CurOffs += toBits(getASTRecordLayout(base).getNonVirtualSize()); | 
| Fariborz Jahanian | f1a66de | 2014-01-07 01:02:50 +0000 | [diff] [blame] | 6152 | #endif | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 6153 | } else { | 
|  | 6154 | FieldDecl *field = cast<FieldDecl>(dcl); | 
|  | 6155 | if (FD) { | 
|  | 6156 | S += '"'; | 
|  | 6157 | S += field->getNameAsString(); | 
|  | 6158 | S += '"'; | 
|  | 6159 | } | 
|  | 6160 |  | 
|  | 6161 | if (field->isBitField()) { | 
|  | 6162 | EncodeBitField(this, S, field->getType(), field); | 
| Fariborz Jahanian | f1a66de | 2014-01-07 01:02:50 +0000 | [diff] [blame] | 6163 | #ifndef NDEBUG | 
| Richard Smith | caf3390 | 2011-10-10 18:28:20 +0000 | [diff] [blame] | 6164 | CurOffs += field->getBitWidthValue(*this); | 
| Fariborz Jahanian | f1a66de | 2014-01-07 01:02:50 +0000 | [diff] [blame] | 6165 | #endif | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 6166 | } else { | 
|  | 6167 | QualType qt = field->getType(); | 
|  | 6168 | getLegacyIntegralTypeEncoding(qt); | 
|  | 6169 | getObjCEncodingForTypeImpl(qt, S, false, true, FD, | 
|  | 6170 | /*OutermostType*/false, | 
|  | 6171 | /*EncodingProperty*/false, | 
| Fariborz Jahanian | 4bf437e | 2014-08-22 23:17:52 +0000 | [diff] [blame] | 6172 | /*StructField*/true, | 
|  | 6173 | false, false, false, NotEncodedT); | 
| Fariborz Jahanian | f1a66de | 2014-01-07 01:02:50 +0000 | [diff] [blame] | 6174 | #ifndef NDEBUG | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 6175 | CurOffs += getTypeSize(field->getType()); | 
| Fariborz Jahanian | f1a66de | 2014-01-07 01:02:50 +0000 | [diff] [blame] | 6176 | #endif | 
| Argyrios Kyrtzidis | 49b35de | 2011-05-17 00:46:38 +0000 | [diff] [blame] | 6177 | } | 
|  | 6178 | } | 
|  | 6179 | } | 
|  | 6180 | } | 
|  | 6181 |  | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 6182 | void ASTContext::getObjCEncodingForTypeQualifier(Decl::ObjCDeclQualifier QT, | 
| Fariborz Jahanian | ac73ff8 | 2007-11-01 17:18:37 +0000 | [diff] [blame] | 6183 | std::string& S) const { | 
|  | 6184 | if (QT & Decl::OBJC_TQ_In) | 
|  | 6185 | S += 'n'; | 
|  | 6186 | if (QT & Decl::OBJC_TQ_Inout) | 
|  | 6187 | S += 'N'; | 
|  | 6188 | if (QT & Decl::OBJC_TQ_Out) | 
|  | 6189 | S += 'o'; | 
|  | 6190 | if (QT & Decl::OBJC_TQ_Bycopy) | 
|  | 6191 | S += 'O'; | 
|  | 6192 | if (QT & Decl::OBJC_TQ_Byref) | 
|  | 6193 | S += 'R'; | 
|  | 6194 | if (QT & Decl::OBJC_TQ_Oneway) | 
|  | 6195 | S += 'V'; | 
|  | 6196 | } | 
|  | 6197 |  | 
| Douglas Gregor | 3ea7269 | 2011-08-12 05:46:01 +0000 | [diff] [blame] | 6198 | TypedefDecl *ASTContext::getObjCIdDecl() const { | 
|  | 6199 | if (!ObjCIdDecl) { | 
| Douglas Gregor | e9d95f1 | 2015-07-07 03:57:35 +0000 | [diff] [blame] | 6200 | QualType T = getObjCObjectType(ObjCBuiltinIdTy, { }, { }); | 
| Douglas Gregor | 3ea7269 | 2011-08-12 05:46:01 +0000 | [diff] [blame] | 6201 | T = getObjCObjectPointerType(T); | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 6202 | ObjCIdDecl = buildImplicitTypedef(T, "id"); | 
| Douglas Gregor | 3ea7269 | 2011-08-12 05:46:01 +0000 | [diff] [blame] | 6203 | } | 
| Douglas Gregor | 3ea7269 | 2011-08-12 05:46:01 +0000 | [diff] [blame] | 6204 | return ObjCIdDecl; | 
| Steve Naroff | 66e9f33 | 2007-10-15 14:41:52 +0000 | [diff] [blame] | 6205 | } | 
|  | 6206 |  | 
| Douglas Gregor | 52e0280 | 2011-08-12 06:17:30 +0000 | [diff] [blame] | 6207 | TypedefDecl *ASTContext::getObjCSelDecl() const { | 
|  | 6208 | if (!ObjCSelDecl) { | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 6209 | QualType T = getPointerType(ObjCBuiltinSelTy); | 
|  | 6210 | ObjCSelDecl = buildImplicitTypedef(T, "SEL"); | 
| Douglas Gregor | 52e0280 | 2011-08-12 06:17:30 +0000 | [diff] [blame] | 6211 | } | 
|  | 6212 | return ObjCSelDecl; | 
| Fariborz Jahanian | 4bef462 | 2007-10-16 20:40:23 +0000 | [diff] [blame] | 6213 | } | 
|  | 6214 |  | 
| Douglas Gregor | 0a58618 | 2011-08-12 05:59:41 +0000 | [diff] [blame] | 6215 | TypedefDecl *ASTContext::getObjCClassDecl() const { | 
|  | 6216 | if (!ObjCClassDecl) { | 
| Douglas Gregor | e9d95f1 | 2015-07-07 03:57:35 +0000 | [diff] [blame] | 6217 | QualType T = getObjCObjectType(ObjCBuiltinClassTy, { }, { }); | 
| Douglas Gregor | 0a58618 | 2011-08-12 05:59:41 +0000 | [diff] [blame] | 6218 | T = getObjCObjectPointerType(T); | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 6219 | ObjCClassDecl = buildImplicitTypedef(T, "Class"); | 
| Douglas Gregor | 0a58618 | 2011-08-12 05:59:41 +0000 | [diff] [blame] | 6220 | } | 
| Douglas Gregor | 0a58618 | 2011-08-12 05:59:41 +0000 | [diff] [blame] | 6221 | return ObjCClassDecl; | 
| Anders Carlsson | f56a7ae | 2007-10-31 02:53:19 +0000 | [diff] [blame] | 6222 | } | 
|  | 6223 |  | 
| Douglas Gregor | d53ae83 | 2012-01-17 18:09:05 +0000 | [diff] [blame] | 6224 | ObjCInterfaceDecl *ASTContext::getObjCProtocolDecl() const { | 
|  | 6225 | if (!ObjCProtocolClassDecl) { | 
|  | 6226 | ObjCProtocolClassDecl | 
|  | 6227 | = ObjCInterfaceDecl::Create(*this, getTranslationUnitDecl(), | 
|  | 6228 | SourceLocation(), | 
|  | 6229 | &Idents.get("Protocol"), | 
| Douglas Gregor | 85f3f95 | 2015-07-07 03:57:15 +0000 | [diff] [blame] | 6230 | /*typeParamList=*/nullptr, | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 6231 | /*PrevDecl=*/nullptr, | 
| Douglas Gregor | d53ae83 | 2012-01-17 18:09:05 +0000 | [diff] [blame] | 6232 | SourceLocation(), true); | 
|  | 6233 | } | 
|  | 6234 |  | 
|  | 6235 | return ObjCProtocolClassDecl; | 
|  | 6236 | } | 
|  | 6237 |  | 
| Meador Inge | 5d3fb22 | 2012-06-16 03:34:49 +0000 | [diff] [blame] | 6238 | //===----------------------------------------------------------------------===// | 
|  | 6239 | // __builtin_va_list Construction Functions | 
|  | 6240 | //===----------------------------------------------------------------------===// | 
|  | 6241 |  | 
| Charles Davis | c7d5c94 | 2015-09-17 20:55:33 +0000 | [diff] [blame] | 6242 | static TypedefDecl *CreateCharPtrNamedVaListDecl(const ASTContext *Context, | 
|  | 6243 | StringRef Name) { | 
|  | 6244 | // typedef char* __builtin[_ms]_va_list; | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 6245 | QualType T = Context->getPointerType(Context->CharTy); | 
| Charles Davis | c7d5c94 | 2015-09-17 20:55:33 +0000 | [diff] [blame] | 6246 | return Context->buildImplicitTypedef(T, Name); | 
|  | 6247 | } | 
|  | 6248 |  | 
|  | 6249 | static TypedefDecl *CreateMSVaListDecl(const ASTContext *Context) { | 
|  | 6250 | return CreateCharPtrNamedVaListDecl(Context, "__builtin_ms_va_list"); | 
|  | 6251 | } | 
|  | 6252 |  | 
|  | 6253 | static TypedefDecl *CreateCharPtrBuiltinVaListDecl(const ASTContext *Context) { | 
|  | 6254 | return CreateCharPtrNamedVaListDecl(Context, "__builtin_va_list"); | 
| Meador Inge | 5d3fb22 | 2012-06-16 03:34:49 +0000 | [diff] [blame] | 6255 | } | 
|  | 6256 |  | 
|  | 6257 | static TypedefDecl *CreateVoidPtrBuiltinVaListDecl(const ASTContext *Context) { | 
|  | 6258 | // typedef void* __builtin_va_list; | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 6259 | QualType T = Context->getPointerType(Context->VoidTy); | 
|  | 6260 | return Context->buildImplicitTypedef(T, "__builtin_va_list"); | 
| Meador Inge | 5d3fb22 | 2012-06-16 03:34:49 +0000 | [diff] [blame] | 6261 | } | 
|  | 6262 |  | 
| Tim Northover | 9bb857a | 2013-01-31 12:13:10 +0000 | [diff] [blame] | 6263 | static TypedefDecl * | 
|  | 6264 | CreateAArch64ABIBuiltinVaListDecl(const ASTContext *Context) { | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 6265 | // struct __va_list | 
| Alp Toker | 2dea15b | 2013-12-17 01:22:38 +0000 | [diff] [blame] | 6266 | RecordDecl *VaListTagDecl = Context->buildImplicitRecord("__va_list"); | 
| Tim Northover | 9bb857a | 2013-01-31 12:13:10 +0000 | [diff] [blame] | 6267 | if (Context->getLangOpts().CPlusPlus) { | 
|  | 6268 | // namespace std { struct __va_list { | 
|  | 6269 | NamespaceDecl *NS; | 
|  | 6270 | NS = NamespaceDecl::Create(const_cast<ASTContext &>(*Context), | 
|  | 6271 | Context->getTranslationUnitDecl(), | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 6272 | /*Inline*/ false, SourceLocation(), | 
| Tim Northover | 9bb857a | 2013-01-31 12:13:10 +0000 | [diff] [blame] | 6273 | SourceLocation(), &Context->Idents.get("std"), | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 6274 | /*PrevDecl*/ nullptr); | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 6275 | NS->setImplicit(); | 
| Tim Northover | 9bb857a | 2013-01-31 12:13:10 +0000 | [diff] [blame] | 6276 | VaListTagDecl->setDeclContext(NS); | 
| Tim Northover | 9bb857a | 2013-01-31 12:13:10 +0000 | [diff] [blame] | 6277 | } | 
|  | 6278 |  | 
|  | 6279 | VaListTagDecl->startDefinition(); | 
|  | 6280 |  | 
|  | 6281 | const size_t NumFields = 5; | 
|  | 6282 | QualType FieldTypes[NumFields]; | 
|  | 6283 | const char *FieldNames[NumFields]; | 
|  | 6284 |  | 
|  | 6285 | // void *__stack; | 
|  | 6286 | FieldTypes[0] = Context->getPointerType(Context->VoidTy); | 
|  | 6287 | FieldNames[0] = "__stack"; | 
|  | 6288 |  | 
|  | 6289 | // void *__gr_top; | 
|  | 6290 | FieldTypes[1] = Context->getPointerType(Context->VoidTy); | 
|  | 6291 | FieldNames[1] = "__gr_top"; | 
|  | 6292 |  | 
|  | 6293 | // void *__vr_top; | 
|  | 6294 | FieldTypes[2] = Context->getPointerType(Context->VoidTy); | 
|  | 6295 | FieldNames[2] = "__vr_top"; | 
|  | 6296 |  | 
|  | 6297 | // int __gr_offs; | 
|  | 6298 | FieldTypes[3] = Context->IntTy; | 
|  | 6299 | FieldNames[3] = "__gr_offs"; | 
|  | 6300 |  | 
|  | 6301 | // int __vr_offs; | 
|  | 6302 | FieldTypes[4] = Context->IntTy; | 
|  | 6303 | FieldNames[4] = "__vr_offs"; | 
|  | 6304 |  | 
|  | 6305 | // Create fields | 
|  | 6306 | for (unsigned i = 0; i < NumFields; ++i) { | 
|  | 6307 | FieldDecl *Field = FieldDecl::Create(const_cast<ASTContext &>(*Context), | 
|  | 6308 | VaListTagDecl, | 
|  | 6309 | SourceLocation(), | 
|  | 6310 | SourceLocation(), | 
|  | 6311 | &Context->Idents.get(FieldNames[i]), | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 6312 | FieldTypes[i], /*TInfo=*/nullptr, | 
|  | 6313 | /*BitWidth=*/nullptr, | 
| Tim Northover | 9bb857a | 2013-01-31 12:13:10 +0000 | [diff] [blame] | 6314 | /*Mutable=*/false, | 
|  | 6315 | ICIS_NoInit); | 
|  | 6316 | Field->setAccess(AS_public); | 
|  | 6317 | VaListTagDecl->addDecl(Field); | 
|  | 6318 | } | 
|  | 6319 | VaListTagDecl->completeDefinition(); | 
| Richard Smith | 9b88a4c | 2015-07-27 05:40:23 +0000 | [diff] [blame] | 6320 | Context->VaListTagDecl = VaListTagDecl; | 
| Tim Northover | 9bb857a | 2013-01-31 12:13:10 +0000 | [diff] [blame] | 6321 | QualType VaListTagType = Context->getRecordType(VaListTagDecl); | 
| Tim Northover | 9bb857a | 2013-01-31 12:13:10 +0000 | [diff] [blame] | 6322 |  | 
|  | 6323 | // } __builtin_va_list; | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 6324 | return Context->buildImplicitTypedef(VaListTagType, "__builtin_va_list"); | 
| Tim Northover | 9bb857a | 2013-01-31 12:13:10 +0000 | [diff] [blame] | 6325 | } | 
|  | 6326 |  | 
| Meador Inge | 5d3fb22 | 2012-06-16 03:34:49 +0000 | [diff] [blame] | 6327 | static TypedefDecl *CreatePowerABIBuiltinVaListDecl(const ASTContext *Context) { | 
|  | 6328 | // typedef struct __va_list_tag { | 
|  | 6329 | RecordDecl *VaListTagDecl; | 
|  | 6330 |  | 
| Alp Toker | 2dea15b | 2013-12-17 01:22:38 +0000 | [diff] [blame] | 6331 | VaListTagDecl = Context->buildImplicitRecord("__va_list_tag"); | 
| Meador Inge | 5d3fb22 | 2012-06-16 03:34:49 +0000 | [diff] [blame] | 6332 | VaListTagDecl->startDefinition(); | 
|  | 6333 |  | 
|  | 6334 | const size_t NumFields = 5; | 
|  | 6335 | QualType FieldTypes[NumFields]; | 
|  | 6336 | const char *FieldNames[NumFields]; | 
|  | 6337 |  | 
|  | 6338 | //   unsigned char gpr; | 
|  | 6339 | FieldTypes[0] = Context->UnsignedCharTy; | 
|  | 6340 | FieldNames[0] = "gpr"; | 
|  | 6341 |  | 
|  | 6342 | //   unsigned char fpr; | 
|  | 6343 | FieldTypes[1] = Context->UnsignedCharTy; | 
|  | 6344 | FieldNames[1] = "fpr"; | 
|  | 6345 |  | 
|  | 6346 | //   unsigned short reserved; | 
|  | 6347 | FieldTypes[2] = Context->UnsignedShortTy; | 
|  | 6348 | FieldNames[2] = "reserved"; | 
|  | 6349 |  | 
|  | 6350 | //   void* overflow_arg_area; | 
|  | 6351 | FieldTypes[3] = Context->getPointerType(Context->VoidTy); | 
|  | 6352 | FieldNames[3] = "overflow_arg_area"; | 
|  | 6353 |  | 
|  | 6354 | //   void* reg_save_area; | 
|  | 6355 | FieldTypes[4] = Context->getPointerType(Context->VoidTy); | 
|  | 6356 | FieldNames[4] = "reg_save_area"; | 
|  | 6357 |  | 
|  | 6358 | // Create fields | 
|  | 6359 | for (unsigned i = 0; i < NumFields; ++i) { | 
|  | 6360 | FieldDecl *Field = FieldDecl::Create(*Context, VaListTagDecl, | 
|  | 6361 | SourceLocation(), | 
|  | 6362 | SourceLocation(), | 
|  | 6363 | &Context->Idents.get(FieldNames[i]), | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 6364 | FieldTypes[i], /*TInfo=*/nullptr, | 
|  | 6365 | /*BitWidth=*/nullptr, | 
| Meador Inge | 5d3fb22 | 2012-06-16 03:34:49 +0000 | [diff] [blame] | 6366 | /*Mutable=*/false, | 
|  | 6367 | ICIS_NoInit); | 
|  | 6368 | Field->setAccess(AS_public); | 
|  | 6369 | VaListTagDecl->addDecl(Field); | 
|  | 6370 | } | 
|  | 6371 | VaListTagDecl->completeDefinition(); | 
| Richard Smith | 9b88a4c | 2015-07-27 05:40:23 +0000 | [diff] [blame] | 6372 | Context->VaListTagDecl = VaListTagDecl; | 
| Meador Inge | 5d3fb22 | 2012-06-16 03:34:49 +0000 | [diff] [blame] | 6373 | QualType VaListTagType = Context->getRecordType(VaListTagDecl); | 
|  | 6374 |  | 
|  | 6375 | // } __va_list_tag; | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 6376 | TypedefDecl *VaListTagTypedefDecl = | 
|  | 6377 | Context->buildImplicitTypedef(VaListTagType, "__va_list_tag"); | 
|  | 6378 |  | 
| Meador Inge | 5d3fb22 | 2012-06-16 03:34:49 +0000 | [diff] [blame] | 6379 | QualType VaListTagTypedefType = | 
|  | 6380 | Context->getTypedefType(VaListTagTypedefDecl); | 
|  | 6381 |  | 
|  | 6382 | // typedef __va_list_tag __builtin_va_list[1]; | 
|  | 6383 | llvm::APInt Size(Context->getTypeSize(Context->getSizeType()), 1); | 
|  | 6384 | QualType VaListTagArrayType | 
|  | 6385 | = Context->getConstantArrayType(VaListTagTypedefType, | 
|  | 6386 | Size, ArrayType::Normal, 0); | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 6387 | return Context->buildImplicitTypedef(VaListTagArrayType, "__builtin_va_list"); | 
| Meador Inge | 5d3fb22 | 2012-06-16 03:34:49 +0000 | [diff] [blame] | 6388 | } | 
|  | 6389 |  | 
|  | 6390 | static TypedefDecl * | 
|  | 6391 | CreateX86_64ABIBuiltinVaListDecl(const ASTContext *Context) { | 
| Richard Smith | 9b88a4c | 2015-07-27 05:40:23 +0000 | [diff] [blame] | 6392 | // struct __va_list_tag { | 
| Meador Inge | 5d3fb22 | 2012-06-16 03:34:49 +0000 | [diff] [blame] | 6393 | RecordDecl *VaListTagDecl; | 
| Alp Toker | 2dea15b | 2013-12-17 01:22:38 +0000 | [diff] [blame] | 6394 | VaListTagDecl = Context->buildImplicitRecord("__va_list_tag"); | 
| Meador Inge | 5d3fb22 | 2012-06-16 03:34:49 +0000 | [diff] [blame] | 6395 | VaListTagDecl->startDefinition(); | 
|  | 6396 |  | 
|  | 6397 | const size_t NumFields = 4; | 
|  | 6398 | QualType FieldTypes[NumFields]; | 
|  | 6399 | const char *FieldNames[NumFields]; | 
|  | 6400 |  | 
|  | 6401 | //   unsigned gp_offset; | 
|  | 6402 | FieldTypes[0] = Context->UnsignedIntTy; | 
|  | 6403 | FieldNames[0] = "gp_offset"; | 
|  | 6404 |  | 
|  | 6405 | //   unsigned fp_offset; | 
|  | 6406 | FieldTypes[1] = Context->UnsignedIntTy; | 
|  | 6407 | FieldNames[1] = "fp_offset"; | 
|  | 6408 |  | 
|  | 6409 | //   void* overflow_arg_area; | 
|  | 6410 | FieldTypes[2] = Context->getPointerType(Context->VoidTy); | 
|  | 6411 | FieldNames[2] = "overflow_arg_area"; | 
|  | 6412 |  | 
|  | 6413 | //   void* reg_save_area; | 
|  | 6414 | FieldTypes[3] = Context->getPointerType(Context->VoidTy); | 
|  | 6415 | FieldNames[3] = "reg_save_area"; | 
|  | 6416 |  | 
|  | 6417 | // Create fields | 
|  | 6418 | for (unsigned i = 0; i < NumFields; ++i) { | 
|  | 6419 | FieldDecl *Field = FieldDecl::Create(const_cast<ASTContext &>(*Context), | 
|  | 6420 | VaListTagDecl, | 
|  | 6421 | SourceLocation(), | 
|  | 6422 | SourceLocation(), | 
|  | 6423 | &Context->Idents.get(FieldNames[i]), | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 6424 | FieldTypes[i], /*TInfo=*/nullptr, | 
|  | 6425 | /*BitWidth=*/nullptr, | 
| Meador Inge | 5d3fb22 | 2012-06-16 03:34:49 +0000 | [diff] [blame] | 6426 | /*Mutable=*/false, | 
|  | 6427 | ICIS_NoInit); | 
|  | 6428 | Field->setAccess(AS_public); | 
|  | 6429 | VaListTagDecl->addDecl(Field); | 
|  | 6430 | } | 
|  | 6431 | VaListTagDecl->completeDefinition(); | 
| Richard Smith | 9b88a4c | 2015-07-27 05:40:23 +0000 | [diff] [blame] | 6432 | Context->VaListTagDecl = VaListTagDecl; | 
| Meador Inge | 5d3fb22 | 2012-06-16 03:34:49 +0000 | [diff] [blame] | 6433 | QualType VaListTagType = Context->getRecordType(VaListTagDecl); | 
|  | 6434 |  | 
| Richard Smith | 9b88a4c | 2015-07-27 05:40:23 +0000 | [diff] [blame] | 6435 | // }; | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 6436 |  | 
| Richard Smith | 9b88a4c | 2015-07-27 05:40:23 +0000 | [diff] [blame] | 6437 | // typedef struct __va_list_tag __builtin_va_list[1]; | 
| Meador Inge | 5d3fb22 | 2012-06-16 03:34:49 +0000 | [diff] [blame] | 6438 | llvm::APInt Size(Context->getTypeSize(Context->getSizeType()), 1); | 
| Richard Smith | 9b88a4c | 2015-07-27 05:40:23 +0000 | [diff] [blame] | 6439 | QualType VaListTagArrayType = | 
|  | 6440 | Context->getConstantArrayType(VaListTagType, Size, ArrayType::Normal, 0); | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 6441 | return Context->buildImplicitTypedef(VaListTagArrayType, "__builtin_va_list"); | 
| Meador Inge | 5d3fb22 | 2012-06-16 03:34:49 +0000 | [diff] [blame] | 6442 | } | 
|  | 6443 |  | 
|  | 6444 | static TypedefDecl *CreatePNaClABIBuiltinVaListDecl(const ASTContext *Context) { | 
|  | 6445 | // typedef int __builtin_va_list[4]; | 
|  | 6446 | llvm::APInt Size(Context->getTypeSize(Context->getSizeType()), 4); | 
|  | 6447 | QualType IntArrayType | 
|  | 6448 | = Context->getConstantArrayType(Context->IntTy, | 
|  | 6449 | Size, ArrayType::Normal, 0); | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 6450 | return Context->buildImplicitTypedef(IntArrayType, "__builtin_va_list"); | 
| Meador Inge | 5d3fb22 | 2012-06-16 03:34:49 +0000 | [diff] [blame] | 6451 | } | 
|  | 6452 |  | 
| Logan Chien | 57086ce | 2012-10-10 06:56:20 +0000 | [diff] [blame] | 6453 | static TypedefDecl * | 
|  | 6454 | CreateAAPCSABIBuiltinVaListDecl(const ASTContext *Context) { | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 6455 | // struct __va_list | 
| Alp Toker | 2dea15b | 2013-12-17 01:22:38 +0000 | [diff] [blame] | 6456 | RecordDecl *VaListDecl = Context->buildImplicitRecord("__va_list"); | 
| Logan Chien | 57086ce | 2012-10-10 06:56:20 +0000 | [diff] [blame] | 6457 | if (Context->getLangOpts().CPlusPlus) { | 
|  | 6458 | // namespace std { struct __va_list { | 
|  | 6459 | NamespaceDecl *NS; | 
|  | 6460 | NS = NamespaceDecl::Create(const_cast<ASTContext &>(*Context), | 
|  | 6461 | Context->getTranslationUnitDecl(), | 
|  | 6462 | /*Inline*/false, SourceLocation(), | 
|  | 6463 | SourceLocation(), &Context->Idents.get("std"), | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 6464 | /*PrevDecl*/ nullptr); | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 6465 | NS->setImplicit(); | 
| Logan Chien | 57086ce | 2012-10-10 06:56:20 +0000 | [diff] [blame] | 6466 | VaListDecl->setDeclContext(NS); | 
| Logan Chien | 57086ce | 2012-10-10 06:56:20 +0000 | [diff] [blame] | 6467 | } | 
|  | 6468 |  | 
|  | 6469 | VaListDecl->startDefinition(); | 
|  | 6470 |  | 
|  | 6471 | // void * __ap; | 
|  | 6472 | FieldDecl *Field = FieldDecl::Create(const_cast<ASTContext &>(*Context), | 
|  | 6473 | VaListDecl, | 
|  | 6474 | SourceLocation(), | 
|  | 6475 | SourceLocation(), | 
|  | 6476 | &Context->Idents.get("__ap"), | 
|  | 6477 | Context->getPointerType(Context->VoidTy), | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 6478 | /*TInfo=*/nullptr, | 
|  | 6479 | /*BitWidth=*/nullptr, | 
| Logan Chien | 57086ce | 2012-10-10 06:56:20 +0000 | [diff] [blame] | 6480 | /*Mutable=*/false, | 
|  | 6481 | ICIS_NoInit); | 
|  | 6482 | Field->setAccess(AS_public); | 
|  | 6483 | VaListDecl->addDecl(Field); | 
|  | 6484 |  | 
|  | 6485 | // }; | 
|  | 6486 | VaListDecl->completeDefinition(); | 
| Oleg Ranevskyy | b88d247 | 2016-03-30 21:30:30 +0000 | [diff] [blame] | 6487 | Context->VaListTagDecl = VaListDecl; | 
| Logan Chien | 57086ce | 2012-10-10 06:56:20 +0000 | [diff] [blame] | 6488 |  | 
|  | 6489 | // typedef struct __va_list __builtin_va_list; | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 6490 | QualType T = Context->getRecordType(VaListDecl); | 
|  | 6491 | return Context->buildImplicitTypedef(T, "__builtin_va_list"); | 
| Logan Chien | 57086ce | 2012-10-10 06:56:20 +0000 | [diff] [blame] | 6492 | } | 
|  | 6493 |  | 
| Ulrich Weigand | 4744507 | 2013-05-06 16:26:41 +0000 | [diff] [blame] | 6494 | static TypedefDecl * | 
|  | 6495 | CreateSystemZBuiltinVaListDecl(const ASTContext *Context) { | 
| Richard Smith | 9b88a4c | 2015-07-27 05:40:23 +0000 | [diff] [blame] | 6496 | // struct __va_list_tag { | 
| Ulrich Weigand | 4744507 | 2013-05-06 16:26:41 +0000 | [diff] [blame] | 6497 | RecordDecl *VaListTagDecl; | 
| Alp Toker | 2dea15b | 2013-12-17 01:22:38 +0000 | [diff] [blame] | 6498 | VaListTagDecl = Context->buildImplicitRecord("__va_list_tag"); | 
| Ulrich Weigand | 4744507 | 2013-05-06 16:26:41 +0000 | [diff] [blame] | 6499 | VaListTagDecl->startDefinition(); | 
|  | 6500 |  | 
|  | 6501 | const size_t NumFields = 4; | 
|  | 6502 | QualType FieldTypes[NumFields]; | 
|  | 6503 | const char *FieldNames[NumFields]; | 
|  | 6504 |  | 
|  | 6505 | //   long __gpr; | 
|  | 6506 | FieldTypes[0] = Context->LongTy; | 
|  | 6507 | FieldNames[0] = "__gpr"; | 
|  | 6508 |  | 
|  | 6509 | //   long __fpr; | 
|  | 6510 | FieldTypes[1] = Context->LongTy; | 
|  | 6511 | FieldNames[1] = "__fpr"; | 
|  | 6512 |  | 
|  | 6513 | //   void *__overflow_arg_area; | 
|  | 6514 | FieldTypes[2] = Context->getPointerType(Context->VoidTy); | 
|  | 6515 | FieldNames[2] = "__overflow_arg_area"; | 
|  | 6516 |  | 
|  | 6517 | //   void *__reg_save_area; | 
|  | 6518 | FieldTypes[3] = Context->getPointerType(Context->VoidTy); | 
|  | 6519 | FieldNames[3] = "__reg_save_area"; | 
|  | 6520 |  | 
|  | 6521 | // Create fields | 
|  | 6522 | for (unsigned i = 0; i < NumFields; ++i) { | 
|  | 6523 | FieldDecl *Field = FieldDecl::Create(const_cast<ASTContext &>(*Context), | 
|  | 6524 | VaListTagDecl, | 
|  | 6525 | SourceLocation(), | 
|  | 6526 | SourceLocation(), | 
|  | 6527 | &Context->Idents.get(FieldNames[i]), | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 6528 | FieldTypes[i], /*TInfo=*/nullptr, | 
|  | 6529 | /*BitWidth=*/nullptr, | 
| Ulrich Weigand | 4744507 | 2013-05-06 16:26:41 +0000 | [diff] [blame] | 6530 | /*Mutable=*/false, | 
|  | 6531 | ICIS_NoInit); | 
|  | 6532 | Field->setAccess(AS_public); | 
|  | 6533 | VaListTagDecl->addDecl(Field); | 
|  | 6534 | } | 
|  | 6535 | VaListTagDecl->completeDefinition(); | 
| Richard Smith | 9b88a4c | 2015-07-27 05:40:23 +0000 | [diff] [blame] | 6536 | Context->VaListTagDecl = VaListTagDecl; | 
| Ulrich Weigand | 4744507 | 2013-05-06 16:26:41 +0000 | [diff] [blame] | 6537 | QualType VaListTagType = Context->getRecordType(VaListTagDecl); | 
| Ulrich Weigand | 4744507 | 2013-05-06 16:26:41 +0000 | [diff] [blame] | 6538 |  | 
| Richard Smith | 9b88a4c | 2015-07-27 05:40:23 +0000 | [diff] [blame] | 6539 | // }; | 
| Ulrich Weigand | 4744507 | 2013-05-06 16:26:41 +0000 | [diff] [blame] | 6540 |  | 
|  | 6541 | // typedef __va_list_tag __builtin_va_list[1]; | 
|  | 6542 | llvm::APInt Size(Context->getTypeSize(Context->getSizeType()), 1); | 
| Richard Smith | 9b88a4c | 2015-07-27 05:40:23 +0000 | [diff] [blame] | 6543 | QualType VaListTagArrayType = | 
|  | 6544 | Context->getConstantArrayType(VaListTagType, Size, ArrayType::Normal, 0); | 
| Ulrich Weigand | 4744507 | 2013-05-06 16:26:41 +0000 | [diff] [blame] | 6545 |  | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 6546 | return Context->buildImplicitTypedef(VaListTagArrayType, "__builtin_va_list"); | 
| Ulrich Weigand | 4744507 | 2013-05-06 16:26:41 +0000 | [diff] [blame] | 6547 | } | 
|  | 6548 |  | 
| Meador Inge | 5d3fb22 | 2012-06-16 03:34:49 +0000 | [diff] [blame] | 6549 | static TypedefDecl *CreateVaListDecl(const ASTContext *Context, | 
|  | 6550 | TargetInfo::BuiltinVaListKind Kind) { | 
|  | 6551 | switch (Kind) { | 
|  | 6552 | case TargetInfo::CharPtrBuiltinVaList: | 
|  | 6553 | return CreateCharPtrBuiltinVaListDecl(Context); | 
|  | 6554 | case TargetInfo::VoidPtrBuiltinVaList: | 
|  | 6555 | return CreateVoidPtrBuiltinVaListDecl(Context); | 
| Tim Northover | 9bb857a | 2013-01-31 12:13:10 +0000 | [diff] [blame] | 6556 | case TargetInfo::AArch64ABIBuiltinVaList: | 
|  | 6557 | return CreateAArch64ABIBuiltinVaListDecl(Context); | 
| Meador Inge | 5d3fb22 | 2012-06-16 03:34:49 +0000 | [diff] [blame] | 6558 | case TargetInfo::PowerABIBuiltinVaList: | 
|  | 6559 | return CreatePowerABIBuiltinVaListDecl(Context); | 
|  | 6560 | case TargetInfo::X86_64ABIBuiltinVaList: | 
|  | 6561 | return CreateX86_64ABIBuiltinVaListDecl(Context); | 
|  | 6562 | case TargetInfo::PNaClABIBuiltinVaList: | 
|  | 6563 | return CreatePNaClABIBuiltinVaListDecl(Context); | 
| Logan Chien | 57086ce | 2012-10-10 06:56:20 +0000 | [diff] [blame] | 6564 | case TargetInfo::AAPCSABIBuiltinVaList: | 
|  | 6565 | return CreateAAPCSABIBuiltinVaListDecl(Context); | 
| Ulrich Weigand | 4744507 | 2013-05-06 16:26:41 +0000 | [diff] [blame] | 6566 | case TargetInfo::SystemZBuiltinVaList: | 
|  | 6567 | return CreateSystemZBuiltinVaListDecl(Context); | 
| Meador Inge | 5d3fb22 | 2012-06-16 03:34:49 +0000 | [diff] [blame] | 6568 | } | 
|  | 6569 |  | 
|  | 6570 | llvm_unreachable("Unhandled __builtin_va_list type kind"); | 
|  | 6571 | } | 
|  | 6572 |  | 
|  | 6573 | TypedefDecl *ASTContext::getBuiltinVaListDecl() const { | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 6574 | if (!BuiltinVaListDecl) { | 
| Meador Inge | 5d3fb22 | 2012-06-16 03:34:49 +0000 | [diff] [blame] | 6575 | BuiltinVaListDecl = CreateVaListDecl(this, Target->getBuiltinVaListKind()); | 
| Alp Toker | 56b5cc9 | 2013-12-15 10:36:26 +0000 | [diff] [blame] | 6576 | assert(BuiltinVaListDecl->isImplicit()); | 
|  | 6577 | } | 
| Meador Inge | 5d3fb22 | 2012-06-16 03:34:49 +0000 | [diff] [blame] | 6578 |  | 
|  | 6579 | return BuiltinVaListDecl; | 
|  | 6580 | } | 
|  | 6581 |  | 
| Richard Smith | 9b88a4c | 2015-07-27 05:40:23 +0000 | [diff] [blame] | 6582 | Decl *ASTContext::getVaListTagDecl() const { | 
|  | 6583 | // Force the creation of VaListTagDecl by building the __builtin_va_list | 
| Meador Inge | cfb6090 | 2012-07-01 15:57:25 +0000 | [diff] [blame] | 6584 | // declaration. | 
| Richard Smith | 9b88a4c | 2015-07-27 05:40:23 +0000 | [diff] [blame] | 6585 | if (!VaListTagDecl) | 
|  | 6586 | (void)getBuiltinVaListDecl(); | 
| Meador Inge | cfb6090 | 2012-07-01 15:57:25 +0000 | [diff] [blame] | 6587 |  | 
| Richard Smith | 9b88a4c | 2015-07-27 05:40:23 +0000 | [diff] [blame] | 6588 | return VaListTagDecl; | 
| Meador Inge | cfb6090 | 2012-07-01 15:57:25 +0000 | [diff] [blame] | 6589 | } | 
|  | 6590 |  | 
| Charles Davis | c7d5c94 | 2015-09-17 20:55:33 +0000 | [diff] [blame] | 6591 | TypedefDecl *ASTContext::getBuiltinMSVaListDecl() const { | 
|  | 6592 | if (!BuiltinMSVaListDecl) | 
|  | 6593 | BuiltinMSVaListDecl = CreateMSVaListDecl(this); | 
|  | 6594 |  | 
|  | 6595 | return BuiltinMSVaListDecl; | 
|  | 6596 | } | 
|  | 6597 |  | 
| Ted Kremenek | 1b0ea82 | 2008-01-07 19:49:32 +0000 | [diff] [blame] | 6598 | void ASTContext::setObjCConstantStringInterface(ObjCInterfaceDecl *Decl) { | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 6599 | assert(ObjCConstantStringType.isNull() && | 
| Steve Naroff | f73b784 | 2007-10-15 23:35:17 +0000 | [diff] [blame] | 6600 | "'NSConstantString' type already set!"); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 6601 |  | 
| Ted Kremenek | 1b0ea82 | 2008-01-07 19:49:32 +0000 | [diff] [blame] | 6602 | ObjCConstantStringType = getObjCInterfaceType(Decl); | 
| Steve Naroff | f73b784 | 2007-10-15 23:35:17 +0000 | [diff] [blame] | 6603 | } | 
|  | 6604 |  | 
| John McCall | d28ae27 | 2009-12-02 08:04:21 +0000 | [diff] [blame] | 6605 | /// \brief Retrieve the template name that corresponds to a non-empty | 
|  | 6606 | /// lookup. | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 6607 | TemplateName | 
|  | 6608 | ASTContext::getOverloadedTemplateName(UnresolvedSetIterator Begin, | 
|  | 6609 | UnresolvedSetIterator End) const { | 
| John McCall | d28ae27 | 2009-12-02 08:04:21 +0000 | [diff] [blame] | 6610 | unsigned size = End - Begin; | 
|  | 6611 | assert(size > 1 && "set is not overloaded!"); | 
|  | 6612 |  | 
|  | 6613 | void *memory = Allocate(sizeof(OverloadedTemplateStorage) + | 
|  | 6614 | size * sizeof(FunctionTemplateDecl*)); | 
|  | 6615 | OverloadedTemplateStorage *OT = new(memory) OverloadedTemplateStorage(size); | 
|  | 6616 |  | 
|  | 6617 | NamedDecl **Storage = OT->getStorage(); | 
| John McCall | ad37125 | 2010-01-20 00:46:10 +0000 | [diff] [blame] | 6618 | for (UnresolvedSetIterator I = Begin; I != End; ++I) { | 
| John McCall | d28ae27 | 2009-12-02 08:04:21 +0000 | [diff] [blame] | 6619 | NamedDecl *D = *I; | 
|  | 6620 | assert(isa<FunctionTemplateDecl>(D) || | 
|  | 6621 | (isa<UsingShadowDecl>(D) && | 
|  | 6622 | isa<FunctionTemplateDecl>(D->getUnderlyingDecl()))); | 
|  | 6623 | *Storage++ = D; | 
|  | 6624 | } | 
|  | 6625 |  | 
|  | 6626 | return TemplateName(OT); | 
|  | 6627 | } | 
|  | 6628 |  | 
| Douglas Gregor | dc572a3 | 2009-03-30 22:58:21 +0000 | [diff] [blame] | 6629 | /// \brief Retrieve the template name that represents a qualified | 
|  | 6630 | /// template name such as \c std::vector. | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 6631 | TemplateName | 
|  | 6632 | ASTContext::getQualifiedTemplateName(NestedNameSpecifier *NNS, | 
|  | 6633 | bool TemplateKeyword, | 
|  | 6634 | TemplateDecl *Template) const { | 
| Douglas Gregor | e29139c | 2011-03-03 17:04:51 +0000 | [diff] [blame] | 6635 | assert(NNS && "Missing nested-name-specifier in qualified template name"); | 
|  | 6636 |  | 
| Douglas Gregor | c42075a | 2010-02-04 18:10:26 +0000 | [diff] [blame] | 6637 | // FIXME: Canonicalization? | 
| Douglas Gregor | dc572a3 | 2009-03-30 22:58:21 +0000 | [diff] [blame] | 6638 | llvm::FoldingSetNodeID ID; | 
|  | 6639 | QualifiedTemplateName::Profile(ID, NNS, TemplateKeyword, Template); | 
|  | 6640 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 6641 | void *InsertPos = nullptr; | 
| Douglas Gregor | dc572a3 | 2009-03-30 22:58:21 +0000 | [diff] [blame] | 6642 | QualifiedTemplateName *QTN = | 
|  | 6643 | QualifiedTemplateNames.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 6644 | if (!QTN) { | 
| Richard Smith | a5e6976 | 2012-08-16 01:19:31 +0000 | [diff] [blame] | 6645 | QTN = new (*this, llvm::alignOf<QualifiedTemplateName>()) | 
|  | 6646 | QualifiedTemplateName(NNS, TemplateKeyword, Template); | 
| Douglas Gregor | dc572a3 | 2009-03-30 22:58:21 +0000 | [diff] [blame] | 6647 | QualifiedTemplateNames.InsertNode(QTN, InsertPos); | 
|  | 6648 | } | 
|  | 6649 |  | 
|  | 6650 | return TemplateName(QTN); | 
|  | 6651 | } | 
|  | 6652 |  | 
|  | 6653 | /// \brief Retrieve the template name that represents a dependent | 
|  | 6654 | /// template name such as \c MetaFun::template apply. | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 6655 | TemplateName | 
|  | 6656 | ASTContext::getDependentTemplateName(NestedNameSpecifier *NNS, | 
|  | 6657 | const IdentifierInfo *Name) const { | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 6658 | assert((!NNS || NNS->isDependent()) && | 
| Douglas Gregor | 308047d | 2009-09-09 00:23:06 +0000 | [diff] [blame] | 6659 | "Nested name specifier must be dependent"); | 
| Douglas Gregor | dc572a3 | 2009-03-30 22:58:21 +0000 | [diff] [blame] | 6660 |  | 
|  | 6661 | llvm::FoldingSetNodeID ID; | 
|  | 6662 | DependentTemplateName::Profile(ID, NNS, Name); | 
|  | 6663 |  | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 6664 | void *InsertPos = nullptr; | 
| Douglas Gregor | dc572a3 | 2009-03-30 22:58:21 +0000 | [diff] [blame] | 6665 | DependentTemplateName *QTN = | 
|  | 6666 | DependentTemplateNames.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 6667 |  | 
|  | 6668 | if (QTN) | 
|  | 6669 | return TemplateName(QTN); | 
|  | 6670 |  | 
|  | 6671 | NestedNameSpecifier *CanonNNS = getCanonicalNestedNameSpecifier(NNS); | 
|  | 6672 | if (CanonNNS == NNS) { | 
| Richard Smith | a5e6976 | 2012-08-16 01:19:31 +0000 | [diff] [blame] | 6673 | QTN = new (*this, llvm::alignOf<DependentTemplateName>()) | 
|  | 6674 | DependentTemplateName(NNS, Name); | 
| Douglas Gregor | dc572a3 | 2009-03-30 22:58:21 +0000 | [diff] [blame] | 6675 | } else { | 
|  | 6676 | TemplateName Canon = getDependentTemplateName(CanonNNS, Name); | 
| Richard Smith | a5e6976 | 2012-08-16 01:19:31 +0000 | [diff] [blame] | 6677 | QTN = new (*this, llvm::alignOf<DependentTemplateName>()) | 
|  | 6678 | DependentTemplateName(NNS, Name, Canon); | 
| Douglas Gregor | c42075a | 2010-02-04 18:10:26 +0000 | [diff] [blame] | 6679 | DependentTemplateName *CheckQTN = | 
|  | 6680 | DependentTemplateNames.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 6681 | assert(!CheckQTN && "Dependent type name canonicalization broken"); | 
|  | 6682 | (void)CheckQTN; | 
| Douglas Gregor | dc572a3 | 2009-03-30 22:58:21 +0000 | [diff] [blame] | 6683 | } | 
|  | 6684 |  | 
|  | 6685 | DependentTemplateNames.InsertNode(QTN, InsertPos); | 
|  | 6686 | return TemplateName(QTN); | 
|  | 6687 | } | 
|  | 6688 |  | 
| Douglas Gregor | 71395fa | 2009-11-04 00:56:37 +0000 | [diff] [blame] | 6689 | /// \brief Retrieve the template name that represents a dependent | 
|  | 6690 | /// template name such as \c MetaFun::template operator+. | 
|  | 6691 | TemplateName | 
|  | 6692 | ASTContext::getDependentTemplateName(NestedNameSpecifier *NNS, | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 6693 | OverloadedOperatorKind Operator) const { | 
| Douglas Gregor | 71395fa | 2009-11-04 00:56:37 +0000 | [diff] [blame] | 6694 | assert((!NNS || NNS->isDependent()) && | 
|  | 6695 | "Nested name specifier must be dependent"); | 
|  | 6696 |  | 
|  | 6697 | llvm::FoldingSetNodeID ID; | 
|  | 6698 | DependentTemplateName::Profile(ID, NNS, Operator); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 6699 |  | 
|  | 6700 | void *InsertPos = nullptr; | 
| Douglas Gregor | c42075a | 2010-02-04 18:10:26 +0000 | [diff] [blame] | 6701 | DependentTemplateName *QTN | 
|  | 6702 | = DependentTemplateNames.FindNodeOrInsertPos(ID, InsertPos); | 
| Douglas Gregor | 71395fa | 2009-11-04 00:56:37 +0000 | [diff] [blame] | 6703 |  | 
|  | 6704 | if (QTN) | 
|  | 6705 | return TemplateName(QTN); | 
|  | 6706 |  | 
|  | 6707 | NestedNameSpecifier *CanonNNS = getCanonicalNestedNameSpecifier(NNS); | 
|  | 6708 | if (CanonNNS == NNS) { | 
| Richard Smith | a5e6976 | 2012-08-16 01:19:31 +0000 | [diff] [blame] | 6709 | QTN = new (*this, llvm::alignOf<DependentTemplateName>()) | 
|  | 6710 | DependentTemplateName(NNS, Operator); | 
| Douglas Gregor | 71395fa | 2009-11-04 00:56:37 +0000 | [diff] [blame] | 6711 | } else { | 
|  | 6712 | TemplateName Canon = getDependentTemplateName(CanonNNS, Operator); | 
| Richard Smith | a5e6976 | 2012-08-16 01:19:31 +0000 | [diff] [blame] | 6713 | QTN = new (*this, llvm::alignOf<DependentTemplateName>()) | 
|  | 6714 | DependentTemplateName(NNS, Operator, Canon); | 
| Douglas Gregor | c42075a | 2010-02-04 18:10:26 +0000 | [diff] [blame] | 6715 |  | 
|  | 6716 | DependentTemplateName *CheckQTN | 
|  | 6717 | = DependentTemplateNames.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 6718 | assert(!CheckQTN && "Dependent template name canonicalization broken"); | 
|  | 6719 | (void)CheckQTN; | 
| Douglas Gregor | 71395fa | 2009-11-04 00:56:37 +0000 | [diff] [blame] | 6720 | } | 
|  | 6721 |  | 
|  | 6722 | DependentTemplateNames.InsertNode(QTN, InsertPos); | 
|  | 6723 | return TemplateName(QTN); | 
|  | 6724 | } | 
|  | 6725 |  | 
| Douglas Gregor | 5590be0 | 2011-01-15 06:45:20 +0000 | [diff] [blame] | 6726 | TemplateName | 
| John McCall | d9dfe3a | 2011-06-30 08:33:18 +0000 | [diff] [blame] | 6727 | ASTContext::getSubstTemplateTemplateParm(TemplateTemplateParmDecl *param, | 
|  | 6728 | TemplateName replacement) const { | 
|  | 6729 | llvm::FoldingSetNodeID ID; | 
|  | 6730 | SubstTemplateTemplateParmStorage::Profile(ID, param, replacement); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 6731 |  | 
|  | 6732 | void *insertPos = nullptr; | 
| John McCall | d9dfe3a | 2011-06-30 08:33:18 +0000 | [diff] [blame] | 6733 | SubstTemplateTemplateParmStorage *subst | 
|  | 6734 | = SubstTemplateTemplateParms.FindNodeOrInsertPos(ID, insertPos); | 
|  | 6735 |  | 
|  | 6736 | if (!subst) { | 
|  | 6737 | subst = new (*this) SubstTemplateTemplateParmStorage(param, replacement); | 
|  | 6738 | SubstTemplateTemplateParms.InsertNode(subst, insertPos); | 
|  | 6739 | } | 
|  | 6740 |  | 
|  | 6741 | return TemplateName(subst); | 
|  | 6742 | } | 
|  | 6743 |  | 
|  | 6744 | TemplateName | 
| Douglas Gregor | 5590be0 | 2011-01-15 06:45:20 +0000 | [diff] [blame] | 6745 | ASTContext::getSubstTemplateTemplateParmPack(TemplateTemplateParmDecl *Param, | 
|  | 6746 | const TemplateArgument &ArgPack) const { | 
|  | 6747 | ASTContext &Self = const_cast<ASTContext &>(*this); | 
|  | 6748 | llvm::FoldingSetNodeID ID; | 
|  | 6749 | SubstTemplateTemplateParmPackStorage::Profile(ID, Self, Param, ArgPack); | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 6750 |  | 
|  | 6751 | void *InsertPos = nullptr; | 
| Douglas Gregor | 5590be0 | 2011-01-15 06:45:20 +0000 | [diff] [blame] | 6752 | SubstTemplateTemplateParmPackStorage *Subst | 
|  | 6753 | = SubstTemplateTemplateParmPacks.FindNodeOrInsertPos(ID, InsertPos); | 
|  | 6754 |  | 
|  | 6755 | if (!Subst) { | 
| John McCall | d9dfe3a | 2011-06-30 08:33:18 +0000 | [diff] [blame] | 6756 | Subst = new (*this) SubstTemplateTemplateParmPackStorage(Param, | 
| Douglas Gregor | 5590be0 | 2011-01-15 06:45:20 +0000 | [diff] [blame] | 6757 | ArgPack.pack_size(), | 
|  | 6758 | ArgPack.pack_begin()); | 
|  | 6759 | SubstTemplateTemplateParmPacks.InsertNode(Subst, InsertPos); | 
|  | 6760 | } | 
|  | 6761 |  | 
|  | 6762 | return TemplateName(Subst); | 
|  | 6763 | } | 
|  | 6764 |  | 
| Douglas Gregor | 8af6e6d | 2008-11-03 14:12:49 +0000 | [diff] [blame] | 6765 | /// getFromTargetType - Given one of the integer types provided by | 
| Douglas Gregor | ab13857 | 2008-11-03 15:57:00 +0000 | [diff] [blame] | 6766 | /// TargetInfo, produce the corresponding type. The unsigned @p Type | 
|  | 6767 | /// is actually a value of type @c TargetInfo::IntType. | 
| John McCall | 48f2d58 | 2009-10-23 23:03:21 +0000 | [diff] [blame] | 6768 | CanQualType ASTContext::getFromTargetType(unsigned Type) const { | 
| Douglas Gregor | 8af6e6d | 2008-11-03 14:12:49 +0000 | [diff] [blame] | 6769 | switch (Type) { | 
| John McCall | 48f2d58 | 2009-10-23 23:03:21 +0000 | [diff] [blame] | 6770 | case TargetInfo::NoInt: return CanQualType(); | 
| Stepan Dyatkovskiy | 5a63792 | 2013-09-05 11:23:21 +0000 | [diff] [blame] | 6771 | case TargetInfo::SignedChar: return SignedCharTy; | 
|  | 6772 | case TargetInfo::UnsignedChar: return UnsignedCharTy; | 
| Douglas Gregor | 8af6e6d | 2008-11-03 14:12:49 +0000 | [diff] [blame] | 6773 | case TargetInfo::SignedShort: return ShortTy; | 
|  | 6774 | case TargetInfo::UnsignedShort: return UnsignedShortTy; | 
|  | 6775 | case TargetInfo::SignedInt: return IntTy; | 
|  | 6776 | case TargetInfo::UnsignedInt: return UnsignedIntTy; | 
|  | 6777 | case TargetInfo::SignedLong: return LongTy; | 
|  | 6778 | case TargetInfo::UnsignedLong: return UnsignedLongTy; | 
|  | 6779 | case TargetInfo::SignedLongLong: return LongLongTy; | 
|  | 6780 | case TargetInfo::UnsignedLongLong: return UnsignedLongLongTy; | 
|  | 6781 | } | 
|  | 6782 |  | 
| David Blaikie | 83d382b | 2011-09-23 05:06:16 +0000 | [diff] [blame] | 6783 | llvm_unreachable("Unhandled TargetInfo::IntType value"); | 
| Douglas Gregor | 8af6e6d | 2008-11-03 14:12:49 +0000 | [diff] [blame] | 6784 | } | 
| Ted Kremenek | 77c51b2 | 2008-07-24 23:58:27 +0000 | [diff] [blame] | 6785 |  | 
|  | 6786 | //===----------------------------------------------------------------------===// | 
|  | 6787 | //                        Type Predicates. | 
|  | 6788 | //===----------------------------------------------------------------------===// | 
|  | 6789 |  | 
| Fariborz Jahanian | 9620769 | 2009-02-18 21:49:28 +0000 | [diff] [blame] | 6790 | /// getObjCGCAttr - Returns one of GCNone, Weak or Strong objc's | 
|  | 6791 | /// garbage collection attribute. | 
|  | 6792 | /// | 
| John McCall | 553d45a | 2011-01-12 00:34:59 +0000 | [diff] [blame] | 6793 | Qualifiers::GC ASTContext::getObjCGCAttrKind(QualType Ty) const { | 
| David Blaikie | bbafb8a | 2012-03-11 07:00:24 +0000 | [diff] [blame] | 6794 | if (getLangOpts().getGC() == LangOptions::NonGC) | 
| John McCall | 553d45a | 2011-01-12 00:34:59 +0000 | [diff] [blame] | 6795 | return Qualifiers::GCNone; | 
|  | 6796 |  | 
| David Blaikie | bbafb8a | 2012-03-11 07:00:24 +0000 | [diff] [blame] | 6797 | assert(getLangOpts().ObjC1); | 
| John McCall | 553d45a | 2011-01-12 00:34:59 +0000 | [diff] [blame] | 6798 | Qualifiers::GC GCAttrs = Ty.getObjCGCAttr(); | 
|  | 6799 |  | 
|  | 6800 | // Default behaviour under objective-C's gc is for ObjC pointers | 
|  | 6801 | // (or pointers to them) be treated as though they were declared | 
|  | 6802 | // as __strong. | 
|  | 6803 | if (GCAttrs == Qualifiers::GCNone) { | 
|  | 6804 | if (Ty->isObjCObjectPointerType() || Ty->isBlockPointerType()) | 
|  | 6805 | return Qualifiers::Strong; | 
|  | 6806 | else if (Ty->isPointerType()) | 
|  | 6807 | return getObjCGCAttrKind(Ty->getAs<PointerType>()->getPointeeType()); | 
|  | 6808 | } else { | 
|  | 6809 | // It's not valid to set GC attributes on anything that isn't a | 
|  | 6810 | // pointer. | 
|  | 6811 | #ifndef NDEBUG | 
|  | 6812 | QualType CT = Ty->getCanonicalTypeInternal(); | 
|  | 6813 | while (const ArrayType *AT = dyn_cast<ArrayType>(CT)) | 
|  | 6814 | CT = AT->getElementType(); | 
|  | 6815 | assert(CT->isAnyPointerType() || CT->isBlockPointerType()); | 
|  | 6816 | #endif | 
| Fariborz Jahanian | 9620769 | 2009-02-18 21:49:28 +0000 | [diff] [blame] | 6817 | } | 
| Chris Lattner | d60183d | 2009-02-18 22:53:11 +0000 | [diff] [blame] | 6818 | return GCAttrs; | 
| Fariborz Jahanian | 9620769 | 2009-02-18 21:49:28 +0000 | [diff] [blame] | 6819 | } | 
|  | 6820 |  | 
| Chris Lattner | 49af6a4 | 2008-04-07 06:51:04 +0000 | [diff] [blame] | 6821 | //===----------------------------------------------------------------------===// | 
|  | 6822 | //                        Type Compatibility Testing | 
|  | 6823 | //===----------------------------------------------------------------------===// | 
| Chris Lattner | b338a6b | 2007-11-01 05:03:41 +0000 | [diff] [blame] | 6824 |  | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 6825 | /// areCompatVectorTypes - Return true if the two specified vector types are | 
| Chris Lattner | 49af6a4 | 2008-04-07 06:51:04 +0000 | [diff] [blame] | 6826 | /// compatible. | 
|  | 6827 | static bool areCompatVectorTypes(const VectorType *LHS, | 
|  | 6828 | const VectorType *RHS) { | 
| John McCall | b692a09 | 2009-10-22 20:10:53 +0000 | [diff] [blame] | 6829 | assert(LHS->isCanonicalUnqualified() && RHS->isCanonicalUnqualified()); | 
| Chris Lattner | 49af6a4 | 2008-04-07 06:51:04 +0000 | [diff] [blame] | 6830 | return LHS->getElementType() == RHS->getElementType() && | 
| Chris Lattner | 465fa32 | 2008-10-05 17:34:18 +0000 | [diff] [blame] | 6831 | LHS->getNumElements() == RHS->getNumElements(); | 
| Chris Lattner | 49af6a4 | 2008-04-07 06:51:04 +0000 | [diff] [blame] | 6832 | } | 
|  | 6833 |  | 
| Douglas Gregor | 59e8b3b | 2010-08-06 10:14:59 +0000 | [diff] [blame] | 6834 | bool ASTContext::areCompatibleVectorTypes(QualType FirstVec, | 
|  | 6835 | QualType SecondVec) { | 
|  | 6836 | assert(FirstVec->isVectorType() && "FirstVec should be a vector type"); | 
|  | 6837 | assert(SecondVec->isVectorType() && "SecondVec should be a vector type"); | 
|  | 6838 |  | 
|  | 6839 | if (hasSameUnqualifiedType(FirstVec, SecondVec)) | 
|  | 6840 | return true; | 
|  | 6841 |  | 
| Bob Wilson | e6aeebb | 2010-11-12 17:24:54 +0000 | [diff] [blame] | 6842 | // Treat Neon vector types and most AltiVec vector types as if they are the | 
|  | 6843 | // equivalent GCC vector types. | 
| Douglas Gregor | 59e8b3b | 2010-08-06 10:14:59 +0000 | [diff] [blame] | 6844 | const VectorType *First = FirstVec->getAs<VectorType>(); | 
|  | 6845 | const VectorType *Second = SecondVec->getAs<VectorType>(); | 
| Bob Wilson | e6aeebb | 2010-11-12 17:24:54 +0000 | [diff] [blame] | 6846 | if (First->getNumElements() == Second->getNumElements() && | 
| Douglas Gregor | 59e8b3b | 2010-08-06 10:14:59 +0000 | [diff] [blame] | 6847 | hasSameType(First->getElementType(), Second->getElementType()) && | 
| Bob Wilson | e6aeebb | 2010-11-12 17:24:54 +0000 | [diff] [blame] | 6848 | First->getVectorKind() != VectorType::AltiVecPixel && | 
|  | 6849 | First->getVectorKind() != VectorType::AltiVecBool && | 
|  | 6850 | Second->getVectorKind() != VectorType::AltiVecPixel && | 
|  | 6851 | Second->getVectorKind() != VectorType::AltiVecBool) | 
| Douglas Gregor | 59e8b3b | 2010-08-06 10:14:59 +0000 | [diff] [blame] | 6852 | return true; | 
|  | 6853 |  | 
|  | 6854 | return false; | 
|  | 6855 | } | 
|  | 6856 |  | 
| Steve Naroff | 8e6aee5 | 2009-07-23 01:01:38 +0000 | [diff] [blame] | 6857 | //===----------------------------------------------------------------------===// | 
|  | 6858 | // ObjCQualifiedIdTypesAreCompatible - Compatibility testing for qualified id's. | 
|  | 6859 | //===----------------------------------------------------------------------===// | 
|  | 6860 |  | 
|  | 6861 | /// ProtocolCompatibleWithProtocol - return 'true' if 'lProto' is in the | 
|  | 6862 | /// inheritance hierarchy of 'rProto'. | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 6863 | bool | 
|  | 6864 | ASTContext::ProtocolCompatibleWithProtocol(ObjCProtocolDecl *lProto, | 
|  | 6865 | ObjCProtocolDecl *rProto) const { | 
| Douglas Gregor | 33b2429 | 2012-01-01 18:09:12 +0000 | [diff] [blame] | 6866 | if (declaresSameEntity(lProto, rProto)) | 
| Steve Naroff | 8e6aee5 | 2009-07-23 01:01:38 +0000 | [diff] [blame] | 6867 | return true; | 
| Aaron Ballman | 0f6e64d | 2014-03-13 22:58:06 +0000 | [diff] [blame] | 6868 | for (auto *PI : rProto->protocols()) | 
|  | 6869 | if (ProtocolCompatibleWithProtocol(lProto, PI)) | 
| Steve Naroff | 8e6aee5 | 2009-07-23 01:01:38 +0000 | [diff] [blame] | 6870 | return true; | 
|  | 6871 | return false; | 
|  | 6872 | } | 
|  | 6873 |  | 
| Dmitri Gribenko | 4364fcf | 2012-08-28 02:49:14 +0000 | [diff] [blame] | 6874 | /// ObjCQualifiedClassTypesAreCompatible - compare  Class<pr,...> and | 
|  | 6875 | /// Class<pr1, ...>. | 
| Fariborz Jahanian | 3c7ebc3 | 2010-07-19 22:02:22 +0000 | [diff] [blame] | 6876 | bool ASTContext::ObjCQualifiedClassTypesAreCompatible(QualType lhs, | 
|  | 6877 | QualType rhs) { | 
|  | 6878 | const ObjCObjectPointerType *lhsQID = lhs->getAs<ObjCObjectPointerType>(); | 
|  | 6879 | const ObjCObjectPointerType *rhsOPT = rhs->getAs<ObjCObjectPointerType>(); | 
|  | 6880 | assert ((lhsQID && rhsOPT) && "ObjCQualifiedClassTypesAreCompatible"); | 
|  | 6881 |  | 
| Aaron Ballman | 8373146 | 2014-03-17 16:14:00 +0000 | [diff] [blame] | 6882 | for (auto *lhsProto : lhsQID->quals()) { | 
| Fariborz Jahanian | 3c7ebc3 | 2010-07-19 22:02:22 +0000 | [diff] [blame] | 6883 | bool match = false; | 
| Aaron Ballman | 8373146 | 2014-03-17 16:14:00 +0000 | [diff] [blame] | 6884 | for (auto *rhsProto : rhsOPT->quals()) { | 
| Fariborz Jahanian | 3c7ebc3 | 2010-07-19 22:02:22 +0000 | [diff] [blame] | 6885 | if (ProtocolCompatibleWithProtocol(lhsProto, rhsProto)) { | 
|  | 6886 | match = true; | 
|  | 6887 | break; | 
|  | 6888 | } | 
|  | 6889 | } | 
|  | 6890 | if (!match) | 
|  | 6891 | return false; | 
|  | 6892 | } | 
|  | 6893 | return true; | 
|  | 6894 | } | 
|  | 6895 |  | 
| Steve Naroff | 8e6aee5 | 2009-07-23 01:01:38 +0000 | [diff] [blame] | 6896 | /// ObjCQualifiedIdTypesAreCompatible - We know that one of lhs/rhs is an | 
|  | 6897 | /// ObjCQualifiedIDType. | 
|  | 6898 | bool ASTContext::ObjCQualifiedIdTypesAreCompatible(QualType lhs, QualType rhs, | 
|  | 6899 | bool compare) { | 
|  | 6900 | // Allow id<P..> and an 'id' or void* type in all cases. | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 6901 | if (lhs->isVoidPointerType() || | 
| Steve Naroff | 8e6aee5 | 2009-07-23 01:01:38 +0000 | [diff] [blame] | 6902 | lhs->isObjCIdType() || lhs->isObjCClassType()) | 
|  | 6903 | return true; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 6904 | else if (rhs->isVoidPointerType() || | 
| Steve Naroff | 8e6aee5 | 2009-07-23 01:01:38 +0000 | [diff] [blame] | 6905 | rhs->isObjCIdType() || rhs->isObjCClassType()) | 
|  | 6906 | return true; | 
|  | 6907 |  | 
|  | 6908 | if (const ObjCObjectPointerType *lhsQID = lhs->getAsObjCQualifiedIdType()) { | 
| John McCall | 9dd450b | 2009-09-21 23:43:11 +0000 | [diff] [blame] | 6909 | const ObjCObjectPointerType *rhsOPT = rhs->getAs<ObjCObjectPointerType>(); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 6910 |  | 
| Steve Naroff | 8e6aee5 | 2009-07-23 01:01:38 +0000 | [diff] [blame] | 6911 | if (!rhsOPT) return false; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 6912 |  | 
| Steve Naroff | 8e6aee5 | 2009-07-23 01:01:38 +0000 | [diff] [blame] | 6913 | if (rhsOPT->qual_empty()) { | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 6914 | // If the RHS is a unqualified interface pointer "NSString*", | 
| Steve Naroff | 8e6aee5 | 2009-07-23 01:01:38 +0000 | [diff] [blame] | 6915 | // make sure we check the class hierarchy. | 
|  | 6916 | if (ObjCInterfaceDecl *rhsID = rhsOPT->getInterfaceDecl()) { | 
| Aaron Ballman | 8373146 | 2014-03-17 16:14:00 +0000 | [diff] [blame] | 6917 | for (auto *I : lhsQID->quals()) { | 
| Steve Naroff | 8e6aee5 | 2009-07-23 01:01:38 +0000 | [diff] [blame] | 6918 | // when comparing an id<P> on lhs with a static type on rhs, | 
|  | 6919 | // see if static class implements all of id's protocols, directly or | 
|  | 6920 | // through its super class and categories. | 
| Aaron Ballman | 8373146 | 2014-03-17 16:14:00 +0000 | [diff] [blame] | 6921 | if (!rhsID->ClassImplementsProtocol(I, true)) | 
| Steve Naroff | 8e6aee5 | 2009-07-23 01:01:38 +0000 | [diff] [blame] | 6922 | return false; | 
|  | 6923 | } | 
|  | 6924 | } | 
|  | 6925 | // If there are no qualifiers and no interface, we have an 'id'. | 
|  | 6926 | return true; | 
|  | 6927 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 6928 | // Both the right and left sides have qualifiers. | 
| Aaron Ballman | 8373146 | 2014-03-17 16:14:00 +0000 | [diff] [blame] | 6929 | for (auto *lhsProto : lhsQID->quals()) { | 
| Steve Naroff | 8e6aee5 | 2009-07-23 01:01:38 +0000 | [diff] [blame] | 6930 | bool match = false; | 
|  | 6931 |  | 
|  | 6932 | // when comparing an id<P> on lhs with a static type on rhs, | 
|  | 6933 | // see if static class implements all of id's protocols, directly or | 
|  | 6934 | // through its super class and categories. | 
| Aaron Ballman | 8373146 | 2014-03-17 16:14:00 +0000 | [diff] [blame] | 6935 | for (auto *rhsProto : rhsOPT->quals()) { | 
| Steve Naroff | 8e6aee5 | 2009-07-23 01:01:38 +0000 | [diff] [blame] | 6936 | if (ProtocolCompatibleWithProtocol(lhsProto, rhsProto) || | 
|  | 6937 | (compare && ProtocolCompatibleWithProtocol(rhsProto, lhsProto))) { | 
|  | 6938 | match = true; | 
|  | 6939 | break; | 
|  | 6940 | } | 
|  | 6941 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 6942 | // If the RHS is a qualified interface pointer "NSString<P>*", | 
| Steve Naroff | 8e6aee5 | 2009-07-23 01:01:38 +0000 | [diff] [blame] | 6943 | // make sure we check the class hierarchy. | 
|  | 6944 | if (ObjCInterfaceDecl *rhsID = rhsOPT->getInterfaceDecl()) { | 
| Aaron Ballman | 8373146 | 2014-03-17 16:14:00 +0000 | [diff] [blame] | 6945 | for (auto *I : lhsQID->quals()) { | 
| Steve Naroff | 8e6aee5 | 2009-07-23 01:01:38 +0000 | [diff] [blame] | 6946 | // when comparing an id<P> on lhs with a static type on rhs, | 
|  | 6947 | // see if static class implements all of id's protocols, directly or | 
|  | 6948 | // through its super class and categories. | 
| Aaron Ballman | 8373146 | 2014-03-17 16:14:00 +0000 | [diff] [blame] | 6949 | if (rhsID->ClassImplementsProtocol(I, true)) { | 
| Steve Naroff | 8e6aee5 | 2009-07-23 01:01:38 +0000 | [diff] [blame] | 6950 | match = true; | 
|  | 6951 | break; | 
|  | 6952 | } | 
|  | 6953 | } | 
|  | 6954 | } | 
|  | 6955 | if (!match) | 
|  | 6956 | return false; | 
|  | 6957 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 6958 |  | 
| Steve Naroff | 8e6aee5 | 2009-07-23 01:01:38 +0000 | [diff] [blame] | 6959 | return true; | 
|  | 6960 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 6961 |  | 
| Steve Naroff | 8e6aee5 | 2009-07-23 01:01:38 +0000 | [diff] [blame] | 6962 | const ObjCObjectPointerType *rhsQID = rhs->getAsObjCQualifiedIdType(); | 
|  | 6963 | assert(rhsQID && "One of the LHS/RHS should be id<x>"); | 
|  | 6964 |  | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 6965 | if (const ObjCObjectPointerType *lhsOPT = | 
| Steve Naroff | 8e6aee5 | 2009-07-23 01:01:38 +0000 | [diff] [blame] | 6966 | lhs->getAsObjCInterfacePointerType()) { | 
| Fariborz Jahanian | eb7714c | 2010-11-01 20:47:16 +0000 | [diff] [blame] | 6967 | // If both the right and left sides have qualifiers. | 
| Aaron Ballman | 8373146 | 2014-03-17 16:14:00 +0000 | [diff] [blame] | 6968 | for (auto *lhsProto : lhsOPT->quals()) { | 
| Steve Naroff | 8e6aee5 | 2009-07-23 01:01:38 +0000 | [diff] [blame] | 6969 | bool match = false; | 
|  | 6970 |  | 
| Fariborz Jahanian | eb7714c | 2010-11-01 20:47:16 +0000 | [diff] [blame] | 6971 | // when comparing an id<P> on rhs with a static type on lhs, | 
| Steve Naroff | 8e6aee5 | 2009-07-23 01:01:38 +0000 | [diff] [blame] | 6972 | // see if static class implements all of id's protocols, directly or | 
|  | 6973 | // through its super class and categories. | 
| Fariborz Jahanian | eb7714c | 2010-11-01 20:47:16 +0000 | [diff] [blame] | 6974 | // First, lhs protocols in the qualifier list must be found, direct | 
|  | 6975 | // or indirect in rhs's qualifier list or it is a mismatch. | 
| Aaron Ballman | 8373146 | 2014-03-17 16:14:00 +0000 | [diff] [blame] | 6976 | for (auto *rhsProto : rhsQID->quals()) { | 
| Steve Naroff | 8e6aee5 | 2009-07-23 01:01:38 +0000 | [diff] [blame] | 6977 | if (ProtocolCompatibleWithProtocol(lhsProto, rhsProto) || | 
|  | 6978 | (compare && ProtocolCompatibleWithProtocol(rhsProto, lhsProto))) { | 
|  | 6979 | match = true; | 
|  | 6980 | break; | 
|  | 6981 | } | 
|  | 6982 | } | 
|  | 6983 | if (!match) | 
|  | 6984 | return false; | 
|  | 6985 | } | 
| Fariborz Jahanian | eb7714c | 2010-11-01 20:47:16 +0000 | [diff] [blame] | 6986 |  | 
|  | 6987 | // Static class's protocols, or its super class or category protocols | 
|  | 6988 | // must be found, direct or indirect in rhs's qualifier list or it is a mismatch. | 
|  | 6989 | if (ObjCInterfaceDecl *lhsID = lhsOPT->getInterfaceDecl()) { | 
|  | 6990 | llvm::SmallPtrSet<ObjCProtocolDecl *, 8> LHSInheritedProtocols; | 
|  | 6991 | CollectInheritedProtocols(lhsID, LHSInheritedProtocols); | 
|  | 6992 | // This is rather dubious but matches gcc's behavior. If lhs has | 
|  | 6993 | // no type qualifier and its class has no static protocol(s) | 
|  | 6994 | // assume that it is mismatch. | 
|  | 6995 | if (LHSInheritedProtocols.empty() && lhsOPT->qual_empty()) | 
|  | 6996 | return false; | 
| Aaron Ballman | 8373146 | 2014-03-17 16:14:00 +0000 | [diff] [blame] | 6997 | for (auto *lhsProto : LHSInheritedProtocols) { | 
| Fariborz Jahanian | eb7714c | 2010-11-01 20:47:16 +0000 | [diff] [blame] | 6998 | bool match = false; | 
| Aaron Ballman | 8373146 | 2014-03-17 16:14:00 +0000 | [diff] [blame] | 6999 | for (auto *rhsProto : rhsQID->quals()) { | 
| Fariborz Jahanian | eb7714c | 2010-11-01 20:47:16 +0000 | [diff] [blame] | 7000 | if (ProtocolCompatibleWithProtocol(lhsProto, rhsProto) || | 
|  | 7001 | (compare && ProtocolCompatibleWithProtocol(rhsProto, lhsProto))) { | 
|  | 7002 | match = true; | 
|  | 7003 | break; | 
|  | 7004 | } | 
|  | 7005 | } | 
|  | 7006 | if (!match) | 
|  | 7007 | return false; | 
|  | 7008 | } | 
|  | 7009 | } | 
| Steve Naroff | 8e6aee5 | 2009-07-23 01:01:38 +0000 | [diff] [blame] | 7010 | return true; | 
|  | 7011 | } | 
|  | 7012 | return false; | 
|  | 7013 | } | 
|  | 7014 |  | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7015 | /// canAssignObjCInterfaces - Return true if the two interface types are | 
| Chris Lattner | 49af6a4 | 2008-04-07 06:51:04 +0000 | [diff] [blame] | 7016 | /// compatible for assignment from RHS to LHS.  This handles validation of any | 
|  | 7017 | /// protocol qualifiers on the LHS or RHS. | 
|  | 7018 | /// | 
| Steve Naroff | 7cae42b | 2009-07-10 23:34:53 +0000 | [diff] [blame] | 7019 | bool ASTContext::canAssignObjCInterfaces(const ObjCObjectPointerType *LHSOPT, | 
|  | 7020 | const ObjCObjectPointerType *RHSOPT) { | 
| John McCall | 8b07ec2 | 2010-05-15 11:32:37 +0000 | [diff] [blame] | 7021 | const ObjCObjectType* LHS = LHSOPT->getObjectType(); | 
|  | 7022 | const ObjCObjectType* RHS = RHSOPT->getObjectType(); | 
|  | 7023 |  | 
| Steve Naroff | 1329fa0 | 2009-07-15 18:40:39 +0000 | [diff] [blame] | 7024 | // If either type represents the built-in 'id' or 'Class' types, return true. | 
| John McCall | 8b07ec2 | 2010-05-15 11:32:37 +0000 | [diff] [blame] | 7025 | if (LHS->isObjCUnqualifiedIdOrClass() || | 
|  | 7026 | RHS->isObjCUnqualifiedIdOrClass()) | 
| Steve Naroff | 7cae42b | 2009-07-10 23:34:53 +0000 | [diff] [blame] | 7027 | return true; | 
|  | 7028 |  | 
| Douglas Gregor | ab209d8 | 2015-07-07 03:58:42 +0000 | [diff] [blame] | 7029 | // Function object that propagates a successful result or handles | 
|  | 7030 | // __kindof types. | 
|  | 7031 | auto finish = [&](bool succeeded) -> bool { | 
|  | 7032 | if (succeeded) | 
|  | 7033 | return true; | 
|  | 7034 |  | 
|  | 7035 | if (!RHS->isKindOfType()) | 
|  | 7036 | return false; | 
|  | 7037 |  | 
|  | 7038 | // Strip off __kindof and protocol qualifiers, then check whether | 
|  | 7039 | // we can assign the other way. | 
|  | 7040 | return canAssignObjCInterfaces(RHSOPT->stripObjCKindOfTypeAndQuals(*this), | 
|  | 7041 | LHSOPT->stripObjCKindOfTypeAndQuals(*this)); | 
|  | 7042 | }; | 
|  | 7043 |  | 
|  | 7044 | if (LHS->isObjCQualifiedId() || RHS->isObjCQualifiedId()) { | 
|  | 7045 | return finish(ObjCQualifiedIdTypesAreCompatible(QualType(LHSOPT,0), | 
|  | 7046 | QualType(RHSOPT,0), | 
|  | 7047 | false)); | 
|  | 7048 | } | 
| Fariborz Jahanian | 3c7ebc3 | 2010-07-19 22:02:22 +0000 | [diff] [blame] | 7049 |  | 
| Douglas Gregor | ab209d8 | 2015-07-07 03:58:42 +0000 | [diff] [blame] | 7050 | if (LHS->isObjCQualifiedClass() && RHS->isObjCQualifiedClass()) { | 
|  | 7051 | return finish(ObjCQualifiedClassTypesAreCompatible(QualType(LHSOPT,0), | 
|  | 7052 | QualType(RHSOPT,0))); | 
|  | 7053 | } | 
| Fariborz Jahanian | 3c7ebc3 | 2010-07-19 22:02:22 +0000 | [diff] [blame] | 7054 |  | 
| John McCall | 8b07ec2 | 2010-05-15 11:32:37 +0000 | [diff] [blame] | 7055 | // If we have 2 user-defined types, fall into that path. | 
| Douglas Gregor | ab209d8 | 2015-07-07 03:58:42 +0000 | [diff] [blame] | 7056 | if (LHS->getInterface() && RHS->getInterface()) { | 
|  | 7057 | return finish(canAssignObjCInterfaces(LHS, RHS)); | 
|  | 7058 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 7059 |  | 
| Steve Naroff | 8e6aee5 | 2009-07-23 01:01:38 +0000 | [diff] [blame] | 7060 | return false; | 
| Steve Naroff | 7cae42b | 2009-07-10 23:34:53 +0000 | [diff] [blame] | 7061 | } | 
|  | 7062 |  | 
| Fariborz Jahanian | b8b0ea3 | 2010-03-17 00:20:01 +0000 | [diff] [blame] | 7063 | /// canAssignObjCInterfacesInBlockPointer - This routine is specifically written | 
| Chris Lattner | 57540c5 | 2011-04-15 05:22:18 +0000 | [diff] [blame] | 7064 | /// for providing type-safety for objective-c pointers used to pass/return | 
| Fariborz Jahanian | b8b0ea3 | 2010-03-17 00:20:01 +0000 | [diff] [blame] | 7065 | /// arguments in block literals. When passed as arguments, passing 'A*' where | 
|  | 7066 | /// 'id' is expected is not OK. Passing 'Sub *" where 'Super *" is expected is | 
|  | 7067 | /// not OK. For the return type, the opposite is not OK. | 
|  | 7068 | bool ASTContext::canAssignObjCInterfacesInBlockPointer( | 
|  | 7069 | const ObjCObjectPointerType *LHSOPT, | 
| Fariborz Jahanian | 90186f8 | 2011-03-14 16:07:00 +0000 | [diff] [blame] | 7070 | const ObjCObjectPointerType *RHSOPT, | 
|  | 7071 | bool BlockReturnType) { | 
| Douglas Gregor | ab209d8 | 2015-07-07 03:58:42 +0000 | [diff] [blame] | 7072 |  | 
|  | 7073 | // Function object that propagates a successful result or handles | 
|  | 7074 | // __kindof types. | 
|  | 7075 | auto finish = [&](bool succeeded) -> bool { | 
|  | 7076 | if (succeeded) | 
|  | 7077 | return true; | 
|  | 7078 |  | 
|  | 7079 | const ObjCObjectPointerType *Expected = BlockReturnType ? RHSOPT : LHSOPT; | 
|  | 7080 | if (!Expected->isKindOfType()) | 
|  | 7081 | return false; | 
|  | 7082 |  | 
|  | 7083 | // Strip off __kindof and protocol qualifiers, then check whether | 
|  | 7084 | // we can assign the other way. | 
|  | 7085 | return canAssignObjCInterfacesInBlockPointer( | 
|  | 7086 | RHSOPT->stripObjCKindOfTypeAndQuals(*this), | 
|  | 7087 | LHSOPT->stripObjCKindOfTypeAndQuals(*this), | 
|  | 7088 | BlockReturnType); | 
|  | 7089 | }; | 
|  | 7090 |  | 
| Fariborz Jahanian | 440a683 | 2010-04-06 17:23:39 +0000 | [diff] [blame] | 7091 | if (RHSOPT->isObjCBuiltinType() || LHSOPT->isObjCIdType()) | 
| Fariborz Jahanian | b8b0ea3 | 2010-03-17 00:20:01 +0000 | [diff] [blame] | 7092 | return true; | 
|  | 7093 |  | 
|  | 7094 | if (LHSOPT->isObjCBuiltinType()) { | 
| Douglas Gregor | ab209d8 | 2015-07-07 03:58:42 +0000 | [diff] [blame] | 7095 | return finish(RHSOPT->isObjCBuiltinType() || | 
|  | 7096 | RHSOPT->isObjCQualifiedIdType()); | 
| Fariborz Jahanian | b8b0ea3 | 2010-03-17 00:20:01 +0000 | [diff] [blame] | 7097 | } | 
|  | 7098 |  | 
| Fariborz Jahanian | 440a683 | 2010-04-06 17:23:39 +0000 | [diff] [blame] | 7099 | if (LHSOPT->isObjCQualifiedIdType() || RHSOPT->isObjCQualifiedIdType()) | 
| Douglas Gregor | ab209d8 | 2015-07-07 03:58:42 +0000 | [diff] [blame] | 7100 | return finish(ObjCQualifiedIdTypesAreCompatible(QualType(LHSOPT,0), | 
|  | 7101 | QualType(RHSOPT,0), | 
|  | 7102 | false)); | 
| Fariborz Jahanian | b8b0ea3 | 2010-03-17 00:20:01 +0000 | [diff] [blame] | 7103 |  | 
|  | 7104 | const ObjCInterfaceType* LHS = LHSOPT->getInterfaceType(); | 
|  | 7105 | const ObjCInterfaceType* RHS = RHSOPT->getInterfaceType(); | 
|  | 7106 | if (LHS && RHS)  { // We have 2 user-defined types. | 
|  | 7107 | if (LHS != RHS) { | 
|  | 7108 | if (LHS->getDecl()->isSuperClassOf(RHS->getDecl())) | 
| Douglas Gregor | ab209d8 | 2015-07-07 03:58:42 +0000 | [diff] [blame] | 7109 | return finish(BlockReturnType); | 
| Fariborz Jahanian | b8b0ea3 | 2010-03-17 00:20:01 +0000 | [diff] [blame] | 7110 | if (RHS->getDecl()->isSuperClassOf(LHS->getDecl())) | 
| Douglas Gregor | ab209d8 | 2015-07-07 03:58:42 +0000 | [diff] [blame] | 7111 | return finish(!BlockReturnType); | 
| Fariborz Jahanian | b8b0ea3 | 2010-03-17 00:20:01 +0000 | [diff] [blame] | 7112 | } | 
|  | 7113 | else | 
|  | 7114 | return true; | 
|  | 7115 | } | 
|  | 7116 | return false; | 
|  | 7117 | } | 
|  | 7118 |  | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7119 | /// Comparison routine for Objective-C protocols to be used with | 
|  | 7120 | /// llvm::array_pod_sort. | 
|  | 7121 | static int compareObjCProtocolsByName(ObjCProtocolDecl * const *lhs, | 
|  | 7122 | ObjCProtocolDecl * const *rhs) { | 
|  | 7123 | return (*lhs)->getName().compare((*rhs)->getName()); | 
|  | 7124 |  | 
|  | 7125 | } | 
|  | 7126 |  | 
| Fariborz Jahanian | 6c5a8e2 | 2009-10-30 01:13:23 +0000 | [diff] [blame] | 7127 | /// getIntersectionOfProtocols - This routine finds the intersection of set | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7128 | /// of protocols inherited from two distinct objective-c pointer objects with | 
|  | 7129 | /// the given common base. | 
| Fariborz Jahanian | 6c5a8e2 | 2009-10-30 01:13:23 +0000 | [diff] [blame] | 7130 | /// It is used to build composite qualifier list of the composite type of | 
|  | 7131 | /// the conditional expression involving two objective-c pointer objects. | 
|  | 7132 | static | 
|  | 7133 | void getIntersectionOfProtocols(ASTContext &Context, | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7134 | const ObjCInterfaceDecl *CommonBase, | 
| Fariborz Jahanian | 6c5a8e2 | 2009-10-30 01:13:23 +0000 | [diff] [blame] | 7135 | const ObjCObjectPointerType *LHSOPT, | 
|  | 7136 | const ObjCObjectPointerType *RHSOPT, | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7137 | SmallVectorImpl<ObjCProtocolDecl *> &IntersectionSet) { | 
| Fariborz Jahanian | 6c5a8e2 | 2009-10-30 01:13:23 +0000 | [diff] [blame] | 7138 |  | 
| John McCall | 8b07ec2 | 2010-05-15 11:32:37 +0000 | [diff] [blame] | 7139 | const ObjCObjectType* LHS = LHSOPT->getObjectType(); | 
|  | 7140 | const ObjCObjectType* RHS = RHSOPT->getObjectType(); | 
|  | 7141 | assert(LHS->getInterface() && "LHS must have an interface base"); | 
|  | 7142 | assert(RHS->getInterface() && "RHS must have an interface base"); | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7143 |  | 
|  | 7144 | // Add all of the protocols for the LHS. | 
|  | 7145 | llvm::SmallPtrSet<ObjCProtocolDecl *, 8> LHSProtocolSet; | 
|  | 7146 |  | 
|  | 7147 | // Start with the protocol qualifiers. | 
|  | 7148 | for (auto proto : LHS->quals()) { | 
|  | 7149 | Context.CollectInheritedProtocols(proto, LHSProtocolSet); | 
| Fariborz Jahanian | 6c5a8e2 | 2009-10-30 01:13:23 +0000 | [diff] [blame] | 7150 | } | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7151 |  | 
|  | 7152 | // Also add the protocols associated with the LHS interface. | 
|  | 7153 | Context.CollectInheritedProtocols(LHS->getInterface(), LHSProtocolSet); | 
|  | 7154 |  | 
|  | 7155 | // Add all of the protocls for the RHS. | 
|  | 7156 | llvm::SmallPtrSet<ObjCProtocolDecl *, 8> RHSProtocolSet; | 
|  | 7157 |  | 
|  | 7158 | // Start with the protocol qualifiers. | 
|  | 7159 | for (auto proto : RHS->quals()) { | 
|  | 7160 | Context.CollectInheritedProtocols(proto, RHSProtocolSet); | 
| Fariborz Jahanian | 6c5a8e2 | 2009-10-30 01:13:23 +0000 | [diff] [blame] | 7161 | } | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7162 |  | 
|  | 7163 | // Also add the protocols associated with the RHS interface. | 
|  | 7164 | Context.CollectInheritedProtocols(RHS->getInterface(), RHSProtocolSet); | 
|  | 7165 |  | 
|  | 7166 | // Compute the intersection of the collected protocol sets. | 
|  | 7167 | for (auto proto : LHSProtocolSet) { | 
|  | 7168 | if (RHSProtocolSet.count(proto)) | 
|  | 7169 | IntersectionSet.push_back(proto); | 
|  | 7170 | } | 
|  | 7171 |  | 
|  | 7172 | // Compute the set of protocols that is implied by either the common type or | 
|  | 7173 | // the protocols within the intersection. | 
|  | 7174 | llvm::SmallPtrSet<ObjCProtocolDecl *, 8> ImpliedProtocols; | 
|  | 7175 | Context.CollectInheritedProtocols(CommonBase, ImpliedProtocols); | 
|  | 7176 |  | 
|  | 7177 | // Remove any implied protocols from the list of inherited protocols. | 
|  | 7178 | if (!ImpliedProtocols.empty()) { | 
|  | 7179 | IntersectionSet.erase( | 
|  | 7180 | std::remove_if(IntersectionSet.begin(), | 
|  | 7181 | IntersectionSet.end(), | 
|  | 7182 | [&](ObjCProtocolDecl *proto) -> bool { | 
|  | 7183 | return ImpliedProtocols.count(proto) > 0; | 
|  | 7184 | }), | 
|  | 7185 | IntersectionSet.end()); | 
|  | 7186 | } | 
|  | 7187 |  | 
|  | 7188 | // Sort the remaining protocols by name. | 
|  | 7189 | llvm::array_pod_sort(IntersectionSet.begin(), IntersectionSet.end(), | 
|  | 7190 | compareObjCProtocolsByName); | 
| Fariborz Jahanian | 6c5a8e2 | 2009-10-30 01:13:23 +0000 | [diff] [blame] | 7191 | } | 
|  | 7192 |  | 
| Douglas Gregor | 1ac1b63 | 2015-07-07 03:58:54 +0000 | [diff] [blame] | 7193 | /// Determine whether the first type is a subtype of the second. | 
|  | 7194 | static bool canAssignObjCObjectTypes(ASTContext &ctx, QualType lhs, | 
|  | 7195 | QualType rhs) { | 
|  | 7196 | // Common case: two object pointers. | 
|  | 7197 | const ObjCObjectPointerType *lhsOPT = lhs->getAs<ObjCObjectPointerType>(); | 
|  | 7198 | const ObjCObjectPointerType *rhsOPT = rhs->getAs<ObjCObjectPointerType>(); | 
|  | 7199 | if (lhsOPT && rhsOPT) | 
|  | 7200 | return ctx.canAssignObjCInterfaces(lhsOPT, rhsOPT); | 
|  | 7201 |  | 
|  | 7202 | // Two block pointers. | 
|  | 7203 | const BlockPointerType *lhsBlock = lhs->getAs<BlockPointerType>(); | 
|  | 7204 | const BlockPointerType *rhsBlock = rhs->getAs<BlockPointerType>(); | 
|  | 7205 | if (lhsBlock && rhsBlock) | 
|  | 7206 | return ctx.typesAreBlockPointerCompatible(lhs, rhs); | 
|  | 7207 |  | 
|  | 7208 | // If either is an unqualified 'id' and the other is a block, it's | 
|  | 7209 | // acceptable. | 
|  | 7210 | if ((lhsOPT && lhsOPT->isObjCIdType() && rhsBlock) || | 
|  | 7211 | (rhsOPT && rhsOPT->isObjCIdType() && lhsBlock)) | 
|  | 7212 | return true; | 
|  | 7213 |  | 
|  | 7214 | return false; | 
|  | 7215 | } | 
|  | 7216 |  | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7217 | // Check that the given Objective-C type argument lists are equivalent. | 
| Douglas Gregor | 1ac1b63 | 2015-07-07 03:58:54 +0000 | [diff] [blame] | 7218 | static bool sameObjCTypeArgs(ASTContext &ctx, | 
|  | 7219 | const ObjCInterfaceDecl *iface, | 
|  | 7220 | ArrayRef<QualType> lhsArgs, | 
| Douglas Gregor | ab209d8 | 2015-07-07 03:58:42 +0000 | [diff] [blame] | 7221 | ArrayRef<QualType> rhsArgs, | 
|  | 7222 | bool stripKindOf) { | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7223 | if (lhsArgs.size() != rhsArgs.size()) | 
|  | 7224 | return false; | 
|  | 7225 |  | 
| Douglas Gregor | 1ac1b63 | 2015-07-07 03:58:54 +0000 | [diff] [blame] | 7226 | ObjCTypeParamList *typeParams = iface->getTypeParamList(); | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7227 | for (unsigned i = 0, n = lhsArgs.size(); i != n; ++i) { | 
| Douglas Gregor | 1ac1b63 | 2015-07-07 03:58:54 +0000 | [diff] [blame] | 7228 | if (ctx.hasSameType(lhsArgs[i], rhsArgs[i])) | 
|  | 7229 | continue; | 
|  | 7230 |  | 
|  | 7231 | switch (typeParams->begin()[i]->getVariance()) { | 
|  | 7232 | case ObjCTypeParamVariance::Invariant: | 
| Douglas Gregor | ab209d8 | 2015-07-07 03:58:42 +0000 | [diff] [blame] | 7233 | if (!stripKindOf || | 
|  | 7234 | !ctx.hasSameType(lhsArgs[i].stripObjCKindOfType(ctx), | 
|  | 7235 | rhsArgs[i].stripObjCKindOfType(ctx))) { | 
|  | 7236 | return false; | 
|  | 7237 | } | 
| Douglas Gregor | 1ac1b63 | 2015-07-07 03:58:54 +0000 | [diff] [blame] | 7238 | break; | 
|  | 7239 |  | 
|  | 7240 | case ObjCTypeParamVariance::Covariant: | 
|  | 7241 | if (!canAssignObjCObjectTypes(ctx, lhsArgs[i], rhsArgs[i])) | 
|  | 7242 | return false; | 
|  | 7243 | break; | 
|  | 7244 |  | 
|  | 7245 | case ObjCTypeParamVariance::Contravariant: | 
|  | 7246 | if (!canAssignObjCObjectTypes(ctx, rhsArgs[i], lhsArgs[i])) | 
|  | 7247 | return false; | 
|  | 7248 | break; | 
| Douglas Gregor | ab209d8 | 2015-07-07 03:58:42 +0000 | [diff] [blame] | 7249 | } | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7250 | } | 
|  | 7251 |  | 
|  | 7252 | return true; | 
|  | 7253 | } | 
|  | 7254 |  | 
| Fariborz Jahanian | ef8b8ce | 2009-10-27 23:02:38 +0000 | [diff] [blame] | 7255 | QualType ASTContext::areCommonBaseCompatible( | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7256 | const ObjCObjectPointerType *Lptr, | 
|  | 7257 | const ObjCObjectPointerType *Rptr) { | 
| John McCall | 8b07ec2 | 2010-05-15 11:32:37 +0000 | [diff] [blame] | 7258 | const ObjCObjectType *LHS = Lptr->getObjectType(); | 
|  | 7259 | const ObjCObjectType *RHS = Rptr->getObjectType(); | 
|  | 7260 | const ObjCInterfaceDecl* LDecl = LHS->getInterface(); | 
|  | 7261 | const ObjCInterfaceDecl* RDecl = RHS->getInterface(); | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7262 |  | 
|  | 7263 | if (!LDecl || !RDecl) | 
| Fariborz Jahanian | ef8b8ce | 2009-10-27 23:02:38 +0000 | [diff] [blame] | 7264 | return QualType(); | 
| Douglas Gregor | e83b956 | 2015-07-07 03:57:53 +0000 | [diff] [blame] | 7265 |  | 
| Manman Ren | c46f7d1 | 2016-05-06 19:35:02 +0000 | [diff] [blame] | 7266 | // When either LHS or RHS is a kindof type, we should return a kindof type. | 
|  | 7267 | // For example, for common base of kindof(ASub1) and kindof(ASub2), we return | 
|  | 7268 | // kindof(A). | 
|  | 7269 | bool anyKindOf = LHS->isKindOfType() || RHS->isKindOfType(); | 
|  | 7270 |  | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7271 | // Follow the left-hand side up the class hierarchy until we either hit a | 
|  | 7272 | // root or find the RHS. Record the ancestors in case we don't find it. | 
|  | 7273 | llvm::SmallDenseMap<const ObjCInterfaceDecl *, const ObjCObjectType *, 4> | 
|  | 7274 | LHSAncestors; | 
|  | 7275 | while (true) { | 
|  | 7276 | // Record this ancestor. We'll need this if the common type isn't in the | 
|  | 7277 | // path from the LHS to the root. | 
|  | 7278 | LHSAncestors[LHS->getInterface()->getCanonicalDecl()] = LHS; | 
| Douglas Gregor | e83b956 | 2015-07-07 03:57:53 +0000 | [diff] [blame] | 7279 |  | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7280 | if (declaresSameEntity(LHS->getInterface(), RDecl)) { | 
|  | 7281 | // Get the type arguments. | 
|  | 7282 | ArrayRef<QualType> LHSTypeArgs = LHS->getTypeArgsAsWritten(); | 
|  | 7283 | bool anyChanges = false; | 
|  | 7284 | if (LHS->isSpecialized() && RHS->isSpecialized()) { | 
|  | 7285 | // Both have type arguments, compare them. | 
| Douglas Gregor | 1ac1b63 | 2015-07-07 03:58:54 +0000 | [diff] [blame] | 7286 | if (!sameObjCTypeArgs(*this, LHS->getInterface(), | 
|  | 7287 | LHS->getTypeArgs(), RHS->getTypeArgs(), | 
| Douglas Gregor | ab209d8 | 2015-07-07 03:58:42 +0000 | [diff] [blame] | 7288 | /*stripKindOf=*/true)) | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7289 | return QualType(); | 
|  | 7290 | } else if (LHS->isSpecialized() != RHS->isSpecialized()) { | 
|  | 7291 | // If only one has type arguments, the result will not have type | 
|  | 7292 | // arguments. | 
|  | 7293 | LHSTypeArgs = { }; | 
|  | 7294 | anyChanges = true; | 
|  | 7295 | } | 
|  | 7296 |  | 
|  | 7297 | // Compute the intersection of protocols. | 
| Chris Lattner | 0e62c1c | 2011-07-23 10:55:15 +0000 | [diff] [blame] | 7298 | SmallVector<ObjCProtocolDecl *, 8> Protocols; | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7299 | getIntersectionOfProtocols(*this, LHS->getInterface(), Lptr, Rptr, | 
|  | 7300 | Protocols); | 
| John McCall | 8b07ec2 | 2010-05-15 11:32:37 +0000 | [diff] [blame] | 7301 | if (!Protocols.empty()) | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7302 | anyChanges = true; | 
|  | 7303 |  | 
|  | 7304 | // If anything in the LHS will have changed, build a new result type. | 
| Manman Ren | c46f7d1 | 2016-05-06 19:35:02 +0000 | [diff] [blame] | 7305 | // If we need to return a kindof type but LHS is not a kindof type, we | 
|  | 7306 | // build a new result type. | 
|  | 7307 | if (anyChanges || LHS->isKindOfType() != anyKindOf) { | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7308 | QualType Result = getObjCInterfaceType(LHS->getInterface()); | 
| Douglas Gregor | ab209d8 | 2015-07-07 03:58:42 +0000 | [diff] [blame] | 7309 | Result = getObjCObjectType(Result, LHSTypeArgs, Protocols, | 
| Manman Ren | c46f7d1 | 2016-05-06 19:35:02 +0000 | [diff] [blame] | 7310 | anyKindOf || LHS->isKindOfType()); | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7311 | return getObjCObjectPointerType(Result); | 
|  | 7312 | } | 
|  | 7313 |  | 
|  | 7314 | return getObjCObjectPointerType(QualType(LHS, 0)); | 
| Fariborz Jahanian | 6c5a8e2 | 2009-10-30 01:13:23 +0000 | [diff] [blame] | 7315 | } | 
| Douglas Gregor | e83b956 | 2015-07-07 03:57:53 +0000 | [diff] [blame] | 7316 |  | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7317 | // Find the superclass. | 
| Douglas Gregor | e83b956 | 2015-07-07 03:57:53 +0000 | [diff] [blame] | 7318 | QualType LHSSuperType = LHS->getSuperClassType(); | 
|  | 7319 | if (LHSSuperType.isNull()) | 
|  | 7320 | break; | 
|  | 7321 |  | 
|  | 7322 | LHS = LHSSuperType->castAs<ObjCObjectType>(); | 
|  | 7323 | } | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7324 |  | 
|  | 7325 | // We didn't find anything by following the LHS to its root; now check | 
|  | 7326 | // the RHS against the cached set of ancestors. | 
|  | 7327 | while (true) { | 
|  | 7328 | auto KnownLHS = LHSAncestors.find(RHS->getInterface()->getCanonicalDecl()); | 
|  | 7329 | if (KnownLHS != LHSAncestors.end()) { | 
|  | 7330 | LHS = KnownLHS->second; | 
|  | 7331 |  | 
|  | 7332 | // Get the type arguments. | 
|  | 7333 | ArrayRef<QualType> RHSTypeArgs = RHS->getTypeArgsAsWritten(); | 
|  | 7334 | bool anyChanges = false; | 
|  | 7335 | if (LHS->isSpecialized() && RHS->isSpecialized()) { | 
|  | 7336 | // Both have type arguments, compare them. | 
| Douglas Gregor | 1ac1b63 | 2015-07-07 03:58:54 +0000 | [diff] [blame] | 7337 | if (!sameObjCTypeArgs(*this, LHS->getInterface(), | 
|  | 7338 | LHS->getTypeArgs(), RHS->getTypeArgs(), | 
| Douglas Gregor | ab209d8 | 2015-07-07 03:58:42 +0000 | [diff] [blame] | 7339 | /*stripKindOf=*/true)) | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7340 | return QualType(); | 
|  | 7341 | } else if (LHS->isSpecialized() != RHS->isSpecialized()) { | 
|  | 7342 | // If only one has type arguments, the result will not have type | 
|  | 7343 | // arguments. | 
|  | 7344 | RHSTypeArgs = { }; | 
|  | 7345 | anyChanges = true; | 
|  | 7346 | } | 
|  | 7347 |  | 
|  | 7348 | // Compute the intersection of protocols. | 
|  | 7349 | SmallVector<ObjCProtocolDecl *, 8> Protocols; | 
|  | 7350 | getIntersectionOfProtocols(*this, RHS->getInterface(), Lptr, Rptr, | 
|  | 7351 | Protocols); | 
|  | 7352 | if (!Protocols.empty()) | 
|  | 7353 | anyChanges = true; | 
|  | 7354 |  | 
| Manman Ren | c46f7d1 | 2016-05-06 19:35:02 +0000 | [diff] [blame] | 7355 | // If we need to return a kindof type but RHS is not a kindof type, we | 
|  | 7356 | // build a new result type. | 
|  | 7357 | if (anyChanges || RHS->isKindOfType() != anyKindOf) { | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7358 | QualType Result = getObjCInterfaceType(RHS->getInterface()); | 
| Douglas Gregor | ab209d8 | 2015-07-07 03:58:42 +0000 | [diff] [blame] | 7359 | Result = getObjCObjectType(Result, RHSTypeArgs, Protocols, | 
| Manman Ren | c46f7d1 | 2016-05-06 19:35:02 +0000 | [diff] [blame] | 7360 | anyKindOf || RHS->isKindOfType()); | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7361 | return getObjCObjectPointerType(Result); | 
|  | 7362 | } | 
|  | 7363 |  | 
|  | 7364 | return getObjCObjectPointerType(QualType(RHS, 0)); | 
|  | 7365 | } | 
|  | 7366 |  | 
|  | 7367 | // Find the superclass of the RHS. | 
|  | 7368 | QualType RHSSuperType = RHS->getSuperClassType(); | 
|  | 7369 | if (RHSSuperType.isNull()) | 
|  | 7370 | break; | 
|  | 7371 |  | 
|  | 7372 | RHS = RHSSuperType->castAs<ObjCObjectType>(); | 
|  | 7373 | } | 
|  | 7374 |  | 
| Fariborz Jahanian | ef8b8ce | 2009-10-27 23:02:38 +0000 | [diff] [blame] | 7375 | return QualType(); | 
|  | 7376 | } | 
|  | 7377 |  | 
| John McCall | 8b07ec2 | 2010-05-15 11:32:37 +0000 | [diff] [blame] | 7378 | bool ASTContext::canAssignObjCInterfaces(const ObjCObjectType *LHS, | 
|  | 7379 | const ObjCObjectType *RHS) { | 
|  | 7380 | assert(LHS->getInterface() && "LHS is not an interface type"); | 
|  | 7381 | assert(RHS->getInterface() && "RHS is not an interface type"); | 
|  | 7382 |  | 
| Chris Lattner | 49af6a4 | 2008-04-07 06:51:04 +0000 | [diff] [blame] | 7383 | // Verify that the base decls are compatible: the RHS must be a subclass of | 
|  | 7384 | // the LHS. | 
| Douglas Gregor | e83b956 | 2015-07-07 03:57:53 +0000 | [diff] [blame] | 7385 | ObjCInterfaceDecl *LHSInterface = LHS->getInterface(); | 
|  | 7386 | bool IsSuperClass = LHSInterface->isSuperClassOf(RHS->getInterface()); | 
|  | 7387 | if (!IsSuperClass) | 
| Chris Lattner | 49af6a4 | 2008-04-07 06:51:04 +0000 | [diff] [blame] | 7388 | return false; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 7389 |  | 
| Douglas Gregor | e83b956 | 2015-07-07 03:57:53 +0000 | [diff] [blame] | 7390 | // If the LHS has protocol qualifiers, determine whether all of them are | 
|  | 7391 | // satisfied by the RHS (i.e., the RHS has a superset of the protocols in the | 
|  | 7392 | // LHS). | 
|  | 7393 | if (LHS->getNumProtocols() > 0) { | 
| Fariborz Jahanian | 12f7ef3 | 2014-10-13 21:07:45 +0000 | [diff] [blame] | 7394 | // OK if conversion of LHS to SuperClass results in narrowing of types | 
|  | 7395 | // ; i.e., SuperClass may implement at least one of the protocols | 
|  | 7396 | // in LHS's protocol list. Example, SuperObj<P1> = lhs<P1,P2> is ok. | 
|  | 7397 | // But not SuperObj<P1,P2,P3> = lhs<P1,P2>. | 
|  | 7398 | llvm::SmallPtrSet<ObjCProtocolDecl *, 8> SuperClassInheritedProtocols; | 
|  | 7399 | CollectInheritedProtocols(RHS->getInterface(), SuperClassInheritedProtocols); | 
|  | 7400 | // Also, if RHS has explicit quelifiers, include them for comparing with LHS's | 
|  | 7401 | // qualifiers. | 
|  | 7402 | for (auto *RHSPI : RHS->quals()) | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7403 | CollectInheritedProtocols(RHSPI, SuperClassInheritedProtocols); | 
| Fariborz Jahanian | 12f7ef3 | 2014-10-13 21:07:45 +0000 | [diff] [blame] | 7404 | // If there is no protocols associated with RHS, it is not a match. | 
|  | 7405 | if (SuperClassInheritedProtocols.empty()) | 
| Steve Naroff | 114aecb | 2009-03-01 16:12:44 +0000 | [diff] [blame] | 7406 | return false; | 
| Fariborz Jahanian | 12f7ef3 | 2014-10-13 21:07:45 +0000 | [diff] [blame] | 7407 |  | 
|  | 7408 | for (const auto *LHSProto : LHS->quals()) { | 
|  | 7409 | bool SuperImplementsProtocol = false; | 
|  | 7410 | for (auto *SuperClassProto : SuperClassInheritedProtocols) | 
|  | 7411 | if (SuperClassProto->lookupProtocolNamed(LHSProto->getIdentifier())) { | 
|  | 7412 | SuperImplementsProtocol = true; | 
|  | 7413 | break; | 
|  | 7414 | } | 
|  | 7415 | if (!SuperImplementsProtocol) | 
|  | 7416 | return false; | 
|  | 7417 | } | 
| Chris Lattner | 49af6a4 | 2008-04-07 06:51:04 +0000 | [diff] [blame] | 7418 | } | 
| Douglas Gregor | e83b956 | 2015-07-07 03:57:53 +0000 | [diff] [blame] | 7419 |  | 
|  | 7420 | // If the LHS is specialized, we may need to check type arguments. | 
|  | 7421 | if (LHS->isSpecialized()) { | 
|  | 7422 | // Follow the superclass chain until we've matched the LHS class in the | 
|  | 7423 | // hierarchy. This substitutes type arguments through. | 
|  | 7424 | const ObjCObjectType *RHSSuper = RHS; | 
|  | 7425 | while (!declaresSameEntity(RHSSuper->getInterface(), LHSInterface)) | 
|  | 7426 | RHSSuper = RHSSuper->getSuperClassType()->castAs<ObjCObjectType>(); | 
|  | 7427 |  | 
|  | 7428 | // If the RHS is specializd, compare type arguments. | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7429 | if (RHSSuper->isSpecialized() && | 
| Douglas Gregor | 1ac1b63 | 2015-07-07 03:58:54 +0000 | [diff] [blame] | 7430 | !sameObjCTypeArgs(*this, LHS->getInterface(), | 
|  | 7431 | LHS->getTypeArgs(), RHSSuper->getTypeArgs(), | 
| Douglas Gregor | ab209d8 | 2015-07-07 03:58:42 +0000 | [diff] [blame] | 7432 | /*stripKindOf=*/true)) { | 
| Douglas Gregor | c5e07f5 | 2015-07-07 03:58:01 +0000 | [diff] [blame] | 7433 | return false; | 
| Douglas Gregor | e83b956 | 2015-07-07 03:57:53 +0000 | [diff] [blame] | 7434 | } | 
|  | 7435 | } | 
|  | 7436 |  | 
|  | 7437 | return true; | 
| Chris Lattner | 49af6a4 | 2008-04-07 06:51:04 +0000 | [diff] [blame] | 7438 | } | 
|  | 7439 |  | 
| Steve Naroff | b760515 | 2009-02-12 17:52:19 +0000 | [diff] [blame] | 7440 | bool ASTContext::areComparableObjCPointerTypes(QualType LHS, QualType RHS) { | 
|  | 7441 | // get the "pointed to" types | 
| John McCall | 9dd450b | 2009-09-21 23:43:11 +0000 | [diff] [blame] | 7442 | const ObjCObjectPointerType *LHSOPT = LHS->getAs<ObjCObjectPointerType>(); | 
|  | 7443 | const ObjCObjectPointerType *RHSOPT = RHS->getAs<ObjCObjectPointerType>(); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 7444 |  | 
| Steve Naroff | 7cae42b | 2009-07-10 23:34:53 +0000 | [diff] [blame] | 7445 | if (!LHSOPT || !RHSOPT) | 
| Steve Naroff | b760515 | 2009-02-12 17:52:19 +0000 | [diff] [blame] | 7446 | return false; | 
| Steve Naroff | 7cae42b | 2009-07-10 23:34:53 +0000 | [diff] [blame] | 7447 |  | 
|  | 7448 | return canAssignObjCInterfaces(LHSOPT, RHSOPT) || | 
|  | 7449 | canAssignObjCInterfaces(RHSOPT, LHSOPT); | 
| Steve Naroff | b760515 | 2009-02-12 17:52:19 +0000 | [diff] [blame] | 7450 | } | 
|  | 7451 |  | 
| Douglas Gregor | 8b2d2fe | 2010-08-07 11:51:51 +0000 | [diff] [blame] | 7452 | bool ASTContext::canBindObjCObjectType(QualType To, QualType From) { | 
|  | 7453 | return canAssignObjCInterfaces( | 
|  | 7454 | getObjCObjectPointerType(To)->getAs<ObjCObjectPointerType>(), | 
|  | 7455 | getObjCObjectPointerType(From)->getAs<ObjCObjectPointerType>()); | 
|  | 7456 | } | 
|  | 7457 |  | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 7458 | /// typesAreCompatible - C99 6.7.3p9: For two qualified types to be compatible, | 
| Steve Naroff | 32e44c0 | 2007-10-15 20:41:53 +0000 | [diff] [blame] | 7459 | /// both shall have the identically qualified version of a compatible type. | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 7460 | /// C99 6.2.7p1: Two types have compatible types if their types are the | 
| Steve Naroff | 32e44c0 | 2007-10-15 20:41:53 +0000 | [diff] [blame] | 7461 | /// same. See 6.7.[2,3,5] for additional rules. | 
| Douglas Gregor | 17ea3f5 | 2010-07-29 15:18:02 +0000 | [diff] [blame] | 7462 | bool ASTContext::typesAreCompatible(QualType LHS, QualType RHS, | 
|  | 7463 | bool CompareUnqualified) { | 
| David Blaikie | bbafb8a | 2012-03-11 07:00:24 +0000 | [diff] [blame] | 7464 | if (getLangOpts().CPlusPlus) | 
| Douglas Gregor | 21e771e | 2010-02-03 21:02:30 +0000 | [diff] [blame] | 7465 | return hasSameType(LHS, RHS); | 
|  | 7466 |  | 
| Douglas Gregor | 17ea3f5 | 2010-07-29 15:18:02 +0000 | [diff] [blame] | 7467 | return !mergeTypes(LHS, RHS, false, CompareUnqualified).isNull(); | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7468 | } | 
|  | 7469 |  | 
| Fariborz Jahanian | c0f6af2 | 2011-07-12 22:05:16 +0000 | [diff] [blame] | 7470 | bool ASTContext::propertyTypesAreCompatible(QualType LHS, QualType RHS) { | 
| Fariborz Jahanian | c87c879 | 2011-07-12 23:20:13 +0000 | [diff] [blame] | 7471 | return typesAreCompatible(LHS, RHS); | 
| Fariborz Jahanian | c0f6af2 | 2011-07-12 22:05:16 +0000 | [diff] [blame] | 7472 | } | 
|  | 7473 |  | 
| Fariborz Jahanian | b8b0ea3 | 2010-03-17 00:20:01 +0000 | [diff] [blame] | 7474 | bool ASTContext::typesAreBlockPointerCompatible(QualType LHS, QualType RHS) { | 
|  | 7475 | return !mergeTypes(LHS, RHS, true).isNull(); | 
|  | 7476 | } | 
|  | 7477 |  | 
| Peter Collingbourne | a99fdcf | 2010-10-24 18:30:18 +0000 | [diff] [blame] | 7478 | /// mergeTransparentUnionType - if T is a transparent union type and a member | 
|  | 7479 | /// of T is compatible with SubType, return the merged type, else return | 
|  | 7480 | /// QualType() | 
|  | 7481 | QualType ASTContext::mergeTransparentUnionType(QualType T, QualType SubType, | 
|  | 7482 | bool OfBlockPointer, | 
|  | 7483 | bool Unqualified) { | 
|  | 7484 | if (const RecordType *UT = T->getAsUnionType()) { | 
|  | 7485 | RecordDecl *UD = UT->getDecl(); | 
|  | 7486 | if (UD->hasAttr<TransparentUnionAttr>()) { | 
| Aaron Ballman | e8a8bae | 2014-03-08 20:12:42 +0000 | [diff] [blame] | 7487 | for (const auto *I : UD->fields()) { | 
|  | 7488 | QualType ET = I->getType().getUnqualifiedType(); | 
| Peter Collingbourne | a99fdcf | 2010-10-24 18:30:18 +0000 | [diff] [blame] | 7489 | QualType MT = mergeTypes(ET, SubType, OfBlockPointer, Unqualified); | 
|  | 7490 | if (!MT.isNull()) | 
|  | 7491 | return MT; | 
|  | 7492 | } | 
|  | 7493 | } | 
|  | 7494 | } | 
|  | 7495 |  | 
|  | 7496 | return QualType(); | 
|  | 7497 | } | 
|  | 7498 |  | 
| Alp Toker | 9cacbab | 2014-01-20 20:26:09 +0000 | [diff] [blame] | 7499 | /// mergeFunctionParameterTypes - merge two types which appear as function | 
|  | 7500 | /// parameter types | 
|  | 7501 | QualType ASTContext::mergeFunctionParameterTypes(QualType lhs, QualType rhs, | 
|  | 7502 | bool OfBlockPointer, | 
|  | 7503 | bool Unqualified) { | 
| Peter Collingbourne | a99fdcf | 2010-10-24 18:30:18 +0000 | [diff] [blame] | 7504 | // GNU extension: two types are compatible if they appear as a function | 
|  | 7505 | // argument, one of the types is a transparent union type and the other | 
|  | 7506 | // type is compatible with a union member | 
|  | 7507 | QualType lmerge = mergeTransparentUnionType(lhs, rhs, OfBlockPointer, | 
|  | 7508 | Unqualified); | 
|  | 7509 | if (!lmerge.isNull()) | 
|  | 7510 | return lmerge; | 
|  | 7511 |  | 
|  | 7512 | QualType rmerge = mergeTransparentUnionType(rhs, lhs, OfBlockPointer, | 
|  | 7513 | Unqualified); | 
|  | 7514 | if (!rmerge.isNull()) | 
|  | 7515 | return rmerge; | 
|  | 7516 |  | 
|  | 7517 | return mergeTypes(lhs, rhs, OfBlockPointer, Unqualified); | 
|  | 7518 | } | 
|  | 7519 |  | 
| Fariborz Jahanian | b8b0ea3 | 2010-03-17 00:20:01 +0000 | [diff] [blame] | 7520 | QualType ASTContext::mergeFunctionTypes(QualType lhs, QualType rhs, | 
| Douglas Gregor | 17ea3f5 | 2010-07-29 15:18:02 +0000 | [diff] [blame] | 7521 | bool OfBlockPointer, | 
|  | 7522 | bool Unqualified) { | 
| John McCall | 9dd450b | 2009-09-21 23:43:11 +0000 | [diff] [blame] | 7523 | const FunctionType *lbase = lhs->getAs<FunctionType>(); | 
|  | 7524 | const FunctionType *rbase = rhs->getAs<FunctionType>(); | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 7525 | const FunctionProtoType *lproto = dyn_cast<FunctionProtoType>(lbase); | 
|  | 7526 | const FunctionProtoType *rproto = dyn_cast<FunctionProtoType>(rbase); | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7527 | bool allLTypes = true; | 
|  | 7528 | bool allRTypes = true; | 
|  | 7529 |  | 
|  | 7530 | // Check return type | 
| Fariborz Jahanian | b8b0ea3 | 2010-03-17 00:20:01 +0000 | [diff] [blame] | 7531 | QualType retType; | 
| Fariborz Jahanian | 1782597 | 2011-02-11 18:46:17 +0000 | [diff] [blame] | 7532 | if (OfBlockPointer) { | 
| Alp Toker | 314cc81 | 2014-01-25 16:55:45 +0000 | [diff] [blame] | 7533 | QualType RHS = rbase->getReturnType(); | 
|  | 7534 | QualType LHS = lbase->getReturnType(); | 
| Fariborz Jahanian | 1782597 | 2011-02-11 18:46:17 +0000 | [diff] [blame] | 7535 | bool UnqualifiedResult = Unqualified; | 
|  | 7536 | if (!UnqualifiedResult) | 
|  | 7537 | UnqualifiedResult = (!RHS.hasQualifiers() && LHS.hasQualifiers()); | 
| Fariborz Jahanian | 90186f8 | 2011-03-14 16:07:00 +0000 | [diff] [blame] | 7538 | retType = mergeTypes(LHS, RHS, true, UnqualifiedResult, true); | 
| Fariborz Jahanian | 1782597 | 2011-02-11 18:46:17 +0000 | [diff] [blame] | 7539 | } | 
| Fariborz Jahanian | b8b0ea3 | 2010-03-17 00:20:01 +0000 | [diff] [blame] | 7540 | else | 
| Alp Toker | 314cc81 | 2014-01-25 16:55:45 +0000 | [diff] [blame] | 7541 | retType = mergeTypes(lbase->getReturnType(), rbase->getReturnType(), false, | 
| John McCall | 8c6b56f | 2010-12-15 01:06:38 +0000 | [diff] [blame] | 7542 | Unqualified); | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7543 | if (retType.isNull()) return QualType(); | 
| Douglas Gregor | 17ea3f5 | 2010-07-29 15:18:02 +0000 | [diff] [blame] | 7544 |  | 
|  | 7545 | if (Unqualified) | 
|  | 7546 | retType = retType.getUnqualifiedType(); | 
|  | 7547 |  | 
| Alp Toker | 314cc81 | 2014-01-25 16:55:45 +0000 | [diff] [blame] | 7548 | CanQualType LRetType = getCanonicalType(lbase->getReturnType()); | 
|  | 7549 | CanQualType RRetType = getCanonicalType(rbase->getReturnType()); | 
| Douglas Gregor | 17ea3f5 | 2010-07-29 15:18:02 +0000 | [diff] [blame] | 7550 | if (Unqualified) { | 
|  | 7551 | LRetType = LRetType.getUnqualifiedType(); | 
|  | 7552 | RRetType = RRetType.getUnqualifiedType(); | 
|  | 7553 | } | 
|  | 7554 |  | 
|  | 7555 | if (getCanonicalType(retType) != LRetType) | 
| Chris Lattner | 465fa32 | 2008-10-05 17:34:18 +0000 | [diff] [blame] | 7556 | allLTypes = false; | 
| Douglas Gregor | 17ea3f5 | 2010-07-29 15:18:02 +0000 | [diff] [blame] | 7557 | if (getCanonicalType(retType) != RRetType) | 
| Chris Lattner | 465fa32 | 2008-10-05 17:34:18 +0000 | [diff] [blame] | 7558 | allRTypes = false; | 
| John McCall | 8c6b56f | 2010-12-15 01:06:38 +0000 | [diff] [blame] | 7559 |  | 
| Daniel Dunbar | edd5bae | 2010-04-28 16:20:58 +0000 | [diff] [blame] | 7560 | // FIXME: double check this | 
|  | 7561 | // FIXME: should we error if lbase->getRegParmAttr() != 0 && | 
|  | 7562 | //                           rbase->getRegParmAttr() != 0 && | 
|  | 7563 | //                           lbase->getRegParmAttr() != rbase->getRegParmAttr()? | 
| Rafael Espindola | c50c27c | 2010-03-30 20:24:48 +0000 | [diff] [blame] | 7564 | FunctionType::ExtInfo lbaseInfo = lbase->getExtInfo(); | 
|  | 7565 | FunctionType::ExtInfo rbaseInfo = rbase->getExtInfo(); | 
| John McCall | 8c6b56f | 2010-12-15 01:06:38 +0000 | [diff] [blame] | 7566 |  | 
| Douglas Gregor | 8c94086 | 2010-01-18 17:14:39 +0000 | [diff] [blame] | 7567 | // Compatible functions must have compatible calling conventions | 
| Reid Kleckner | 78af070 | 2013-08-27 23:08:25 +0000 | [diff] [blame] | 7568 | if (lbaseInfo.getCC() != rbaseInfo.getCC()) | 
| Douglas Gregor | 8c94086 | 2010-01-18 17:14:39 +0000 | [diff] [blame] | 7569 | return QualType(); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 7570 |  | 
| John McCall | 8c6b56f | 2010-12-15 01:06:38 +0000 | [diff] [blame] | 7571 | // Regparm is part of the calling convention. | 
| Eli Friedman | c5b20b5 | 2011-04-09 08:18:08 +0000 | [diff] [blame] | 7572 | if (lbaseInfo.getHasRegParm() != rbaseInfo.getHasRegParm()) | 
|  | 7573 | return QualType(); | 
| John McCall | 8c6b56f | 2010-12-15 01:06:38 +0000 | [diff] [blame] | 7574 | if (lbaseInfo.getRegParm() != rbaseInfo.getRegParm()) | 
|  | 7575 | return QualType(); | 
|  | 7576 |  | 
| John McCall | 31168b0 | 2011-06-15 23:02:42 +0000 | [diff] [blame] | 7577 | if (lbaseInfo.getProducesResult() != rbaseInfo.getProducesResult()) | 
|  | 7578 | return QualType(); | 
|  | 7579 |  | 
| John McCall | 8c6b56f | 2010-12-15 01:06:38 +0000 | [diff] [blame] | 7580 | // FIXME: some uses, e.g. conditional exprs, really want this to be 'both'. | 
|  | 7581 | bool NoReturn = lbaseInfo.getNoReturn() || rbaseInfo.getNoReturn(); | 
| John McCall | 8c6b56f | 2010-12-15 01:06:38 +0000 | [diff] [blame] | 7582 |  | 
| Rafael Espindola | 8778c28 | 2012-11-29 16:09:03 +0000 | [diff] [blame] | 7583 | if (lbaseInfo.getNoReturn() != NoReturn) | 
|  | 7584 | allLTypes = false; | 
|  | 7585 | if (rbaseInfo.getNoReturn() != NoReturn) | 
|  | 7586 | allRTypes = false; | 
|  | 7587 |  | 
| John McCall | 31168b0 | 2011-06-15 23:02:42 +0000 | [diff] [blame] | 7588 | FunctionType::ExtInfo einfo = lbaseInfo.withNoReturn(NoReturn); | 
| John McCall | db40c7f | 2010-12-14 08:05:40 +0000 | [diff] [blame] | 7589 |  | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7590 | if (lproto && rproto) { // two C99 style function prototypes | 
| Sebastian Redl | 5068f77ac | 2009-05-27 22:11:52 +0000 | [diff] [blame] | 7591 | assert(!lproto->hasExceptionSpec() && !rproto->hasExceptionSpec() && | 
|  | 7592 | "C++ shouldn't be here"); | 
| Alp Toker | 601b22c | 2014-01-21 23:35:24 +0000 | [diff] [blame] | 7593 | // Compatible functions must have the same number of parameters | 
|  | 7594 | if (lproto->getNumParams() != rproto->getNumParams()) | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7595 | return QualType(); | 
|  | 7596 |  | 
|  | 7597 | // Variadic and non-variadic functions aren't compatible | 
|  | 7598 | if (lproto->isVariadic() != rproto->isVariadic()) | 
|  | 7599 | return QualType(); | 
|  | 7600 |  | 
| Argyrios Kyrtzidis | 22a3735 | 2008-10-26 16:43:14 +0000 | [diff] [blame] | 7601 | if (lproto->getTypeQuals() != rproto->getTypeQuals()) | 
|  | 7602 | return QualType(); | 
|  | 7603 |  | 
| John McCall | 18afab7 | 2016-03-01 00:49:02 +0000 | [diff] [blame] | 7604 | if (!doFunctionTypesMatchOnExtParameterInfos(rproto, lproto)) | 
| Fariborz Jahanian | 9767697 | 2011-09-28 21:52:05 +0000 | [diff] [blame] | 7605 | return QualType(); | 
| Alp Toker | 601b22c | 2014-01-21 23:35:24 +0000 | [diff] [blame] | 7606 |  | 
|  | 7607 | // Check parameter type compatibility | 
| Chris Lattner | 0e62c1c | 2011-07-23 10:55:15 +0000 | [diff] [blame] | 7608 | SmallVector<QualType, 10> types; | 
| Alp Toker | 601b22c | 2014-01-21 23:35:24 +0000 | [diff] [blame] | 7609 | for (unsigned i = 0, n = lproto->getNumParams(); i < n; i++) { | 
|  | 7610 | QualType lParamType = lproto->getParamType(i).getUnqualifiedType(); | 
|  | 7611 | QualType rParamType = rproto->getParamType(i).getUnqualifiedType(); | 
|  | 7612 | QualType paramType = mergeFunctionParameterTypes( | 
|  | 7613 | lParamType, rParamType, OfBlockPointer, Unqualified); | 
|  | 7614 | if (paramType.isNull()) | 
|  | 7615 | return QualType(); | 
|  | 7616 |  | 
| Douglas Gregor | 17ea3f5 | 2010-07-29 15:18:02 +0000 | [diff] [blame] | 7617 | if (Unqualified) | 
| Alp Toker | 601b22c | 2014-01-21 23:35:24 +0000 | [diff] [blame] | 7618 | paramType = paramType.getUnqualifiedType(); | 
|  | 7619 |  | 
|  | 7620 | types.push_back(paramType); | 
| Douglas Gregor | 17ea3f5 | 2010-07-29 15:18:02 +0000 | [diff] [blame] | 7621 | if (Unqualified) { | 
| Alp Toker | 601b22c | 2014-01-21 23:35:24 +0000 | [diff] [blame] | 7622 | lParamType = lParamType.getUnqualifiedType(); | 
|  | 7623 | rParamType = rParamType.getUnqualifiedType(); | 
| Douglas Gregor | 17ea3f5 | 2010-07-29 15:18:02 +0000 | [diff] [blame] | 7624 | } | 
| Alp Toker | 601b22c | 2014-01-21 23:35:24 +0000 | [diff] [blame] | 7625 |  | 
|  | 7626 | if (getCanonicalType(paramType) != getCanonicalType(lParamType)) | 
| Chris Lattner | 465fa32 | 2008-10-05 17:34:18 +0000 | [diff] [blame] | 7627 | allLTypes = false; | 
| Alp Toker | 601b22c | 2014-01-21 23:35:24 +0000 | [diff] [blame] | 7628 | if (getCanonicalType(paramType) != getCanonicalType(rParamType)) | 
| Chris Lattner | 465fa32 | 2008-10-05 17:34:18 +0000 | [diff] [blame] | 7629 | allRTypes = false; | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7630 | } | 
| Fariborz Jahanian | 9767697 | 2011-09-28 21:52:05 +0000 | [diff] [blame] | 7631 |  | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7632 | if (allLTypes) return lhs; | 
|  | 7633 | if (allRTypes) return rhs; | 
| John McCall | db40c7f | 2010-12-14 08:05:40 +0000 | [diff] [blame] | 7634 |  | 
|  | 7635 | FunctionProtoType::ExtProtoInfo EPI = lproto->getExtProtoInfo(); | 
|  | 7636 | EPI.ExtInfo = einfo; | 
| Jordan Rose | 5c38272 | 2013-03-08 21:51:21 +0000 | [diff] [blame] | 7637 | return getFunctionType(retType, types, EPI); | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7638 | } | 
|  | 7639 |  | 
|  | 7640 | if (lproto) allRTypes = false; | 
|  | 7641 | if (rproto) allLTypes = false; | 
|  | 7642 |  | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 7643 | const FunctionProtoType *proto = lproto ? lproto : rproto; | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7644 | if (proto) { | 
| Sebastian Redl | 5068f77ac | 2009-05-27 22:11:52 +0000 | [diff] [blame] | 7645 | assert(!proto->hasExceptionSpec() && "C++ shouldn't be here"); | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7646 | if (proto->isVariadic()) return QualType(); | 
|  | 7647 | // Check that the types are compatible with the types that | 
|  | 7648 | // would result from default argument promotions (C99 6.7.5.3p15). | 
|  | 7649 | // The only types actually affected are promotable integer | 
|  | 7650 | // types and floats, which would be passed as a different | 
|  | 7651 | // type depending on whether the prototype is visible. | 
| Alp Toker | 601b22c | 2014-01-21 23:35:24 +0000 | [diff] [blame] | 7652 | for (unsigned i = 0, n = proto->getNumParams(); i < n; ++i) { | 
|  | 7653 | QualType paramTy = proto->getParamType(i); | 
| Alp Toker | 9cacbab | 2014-01-20 20:26:09 +0000 | [diff] [blame] | 7654 |  | 
| Eli Friedman | 448ce40 | 2012-08-30 00:44:15 +0000 | [diff] [blame] | 7655 | // Look at the converted type of enum types, since that is the type used | 
| Douglas Gregor | 2973d40 | 2010-02-03 19:27:29 +0000 | [diff] [blame] | 7656 | // to pass enum values. | 
| Alp Toker | 601b22c | 2014-01-21 23:35:24 +0000 | [diff] [blame] | 7657 | if (const EnumType *Enum = paramTy->getAs<EnumType>()) { | 
|  | 7658 | paramTy = Enum->getDecl()->getIntegerType(); | 
|  | 7659 | if (paramTy.isNull()) | 
| Eli Friedman | 448ce40 | 2012-08-30 00:44:15 +0000 | [diff] [blame] | 7660 | return QualType(); | 
|  | 7661 | } | 
| Alp Toker | 601b22c | 2014-01-21 23:35:24 +0000 | [diff] [blame] | 7662 |  | 
|  | 7663 | if (paramTy->isPromotableIntegerType() || | 
|  | 7664 | getCanonicalType(paramTy).getUnqualifiedType() == FloatTy) | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7665 | return QualType(); | 
|  | 7666 | } | 
|  | 7667 |  | 
|  | 7668 | if (allLTypes) return lhs; | 
|  | 7669 | if (allRTypes) return rhs; | 
| John McCall | db40c7f | 2010-12-14 08:05:40 +0000 | [diff] [blame] | 7670 |  | 
|  | 7671 | FunctionProtoType::ExtProtoInfo EPI = proto->getExtProtoInfo(); | 
|  | 7672 | EPI.ExtInfo = einfo; | 
| Alp Toker | 9cacbab | 2014-01-20 20:26:09 +0000 | [diff] [blame] | 7673 | return getFunctionType(retType, proto->getParamTypes(), EPI); | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7674 | } | 
|  | 7675 |  | 
|  | 7676 | if (allLTypes) return lhs; | 
|  | 7677 | if (allRTypes) return rhs; | 
| John McCall | 8c6b56f | 2010-12-15 01:06:38 +0000 | [diff] [blame] | 7678 | return getFunctionNoProtoType(retType, einfo); | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7679 | } | 
|  | 7680 |  | 
| John McCall | 433c2e6 | 2013-03-21 00:10:07 +0000 | [diff] [blame] | 7681 | /// Given that we have an enum type and a non-enum type, try to merge them. | 
|  | 7682 | static QualType mergeEnumWithInteger(ASTContext &Context, const EnumType *ET, | 
|  | 7683 | QualType other, bool isBlockReturnType) { | 
|  | 7684 | // C99 6.7.2.2p4: Each enumerated type shall be compatible with char, | 
|  | 7685 | // a signed integer type, or an unsigned integer type. | 
|  | 7686 | // Compatibility is based on the underlying type, not the promotion | 
|  | 7687 | // type. | 
|  | 7688 | QualType underlyingType = ET->getDecl()->getIntegerType(); | 
|  | 7689 | if (underlyingType.isNull()) return QualType(); | 
|  | 7690 | if (Context.hasSameType(underlyingType, other)) | 
|  | 7691 | return other; | 
|  | 7692 |  | 
|  | 7693 | // In block return types, we're more permissive and accept any | 
|  | 7694 | // integral type of the same size. | 
|  | 7695 | if (isBlockReturnType && other->isIntegerType() && | 
|  | 7696 | Context.getTypeSize(underlyingType) == Context.getTypeSize(other)) | 
|  | 7697 | return other; | 
|  | 7698 |  | 
|  | 7699 | return QualType(); | 
|  | 7700 | } | 
|  | 7701 |  | 
| Fariborz Jahanian | b8b0ea3 | 2010-03-17 00:20:01 +0000 | [diff] [blame] | 7702 | QualType ASTContext::mergeTypes(QualType LHS, QualType RHS, | 
| Douglas Gregor | 17ea3f5 | 2010-07-29 15:18:02 +0000 | [diff] [blame] | 7703 | bool OfBlockPointer, | 
| Fariborz Jahanian | 90186f8 | 2011-03-14 16:07:00 +0000 | [diff] [blame] | 7704 | bool Unqualified, bool BlockReturnType) { | 
| Bill Wendling | db4e349 | 2007-12-03 07:33:35 +0000 | [diff] [blame] | 7705 | // C++ [expr]: If an expression initially has the type "reference to T", the | 
|  | 7706 | // type is adjusted to "T" prior to any further analysis, the expression | 
|  | 7707 | // designates the object or function denoted by the reference, and the | 
| Sebastian Redl | 0f8b23f | 2009-03-16 23:22:08 +0000 | [diff] [blame] | 7708 | // expression is an lvalue unless the reference is an rvalue reference and | 
|  | 7709 | // the expression is a function call (possibly inside parentheses). | 
| Douglas Gregor | 21e771e | 2010-02-03 21:02:30 +0000 | [diff] [blame] | 7710 | assert(!LHS->getAs<ReferenceType>() && "LHS is a reference type?"); | 
|  | 7711 | assert(!RHS->getAs<ReferenceType>() && "RHS is a reference type?"); | 
| Douglas Gregor | 17ea3f5 | 2010-07-29 15:18:02 +0000 | [diff] [blame] | 7712 |  | 
|  | 7713 | if (Unqualified) { | 
|  | 7714 | LHS = LHS.getUnqualifiedType(); | 
|  | 7715 | RHS = RHS.getUnqualifiedType(); | 
|  | 7716 | } | 
| Douglas Gregor | 21e771e | 2010-02-03 21:02:30 +0000 | [diff] [blame] | 7717 |  | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7718 | QualType LHSCan = getCanonicalType(LHS), | 
|  | 7719 | RHSCan = getCanonicalType(RHS); | 
|  | 7720 |  | 
|  | 7721 | // If two types are identical, they are compatible. | 
|  | 7722 | if (LHSCan == RHSCan) | 
|  | 7723 | return LHS; | 
|  | 7724 |  | 
| John McCall | 8ccfcb5 | 2009-09-24 19:53:00 +0000 | [diff] [blame] | 7725 | // If the qualifiers are different, the types aren't compatible... mostly. | 
| Douglas Gregor | 1b8fe5b7 | 2009-11-16 21:35:15 +0000 | [diff] [blame] | 7726 | Qualifiers LQuals = LHSCan.getLocalQualifiers(); | 
|  | 7727 | Qualifiers RQuals = RHSCan.getLocalQualifiers(); | 
| John McCall | 8ccfcb5 | 2009-09-24 19:53:00 +0000 | [diff] [blame] | 7728 | if (LQuals != RQuals) { | 
| Yaxun Liu | a1a87ad | 2016-04-12 19:43:36 +0000 | [diff] [blame] | 7729 | if (getLangOpts().OpenCL) { | 
| Yaxun Liu | ab93394 | 2016-04-28 17:34:57 +0000 | [diff] [blame] | 7730 | if (LHSCan.getUnqualifiedType() != RHSCan.getUnqualifiedType() || | 
| Yaxun Liu | a1a87ad | 2016-04-12 19:43:36 +0000 | [diff] [blame] | 7731 | LQuals.getCVRQualifiers() != RQuals.getCVRQualifiers()) | 
|  | 7732 | return QualType(); | 
|  | 7733 | if (LQuals.isAddressSpaceSupersetOf(RQuals)) | 
|  | 7734 | return LHS; | 
|  | 7735 | if (RQuals.isAddressSpaceSupersetOf(LQuals)) | 
|  | 7736 | return RHS; | 
|  | 7737 | } | 
| John McCall | 8ccfcb5 | 2009-09-24 19:53:00 +0000 | [diff] [blame] | 7738 | // If any of these qualifiers are different, we have a type | 
|  | 7739 | // mismatch. | 
|  | 7740 | if (LQuals.getCVRQualifiers() != RQuals.getCVRQualifiers() || | 
| John McCall | 31168b0 | 2011-06-15 23:02:42 +0000 | [diff] [blame] | 7741 | LQuals.getAddressSpace() != RQuals.getAddressSpace() || | 
|  | 7742 | LQuals.getObjCLifetime() != RQuals.getObjCLifetime()) | 
| John McCall | 8ccfcb5 | 2009-09-24 19:53:00 +0000 | [diff] [blame] | 7743 | return QualType(); | 
|  | 7744 |  | 
|  | 7745 | // Exactly one GC qualifier difference is allowed: __strong is | 
|  | 7746 | // okay if the other type has no GC qualifier but is an Objective | 
|  | 7747 | // C object pointer (i.e. implicitly strong by default).  We fix | 
|  | 7748 | // this by pretending that the unqualified type was actually | 
|  | 7749 | // qualified __strong. | 
|  | 7750 | Qualifiers::GC GC_L = LQuals.getObjCGCAttr(); | 
|  | 7751 | Qualifiers::GC GC_R = RQuals.getObjCGCAttr(); | 
|  | 7752 | assert((GC_L != GC_R) && "unequal qualifier sets had only equal elements"); | 
|  | 7753 |  | 
|  | 7754 | if (GC_L == Qualifiers::Weak || GC_R == Qualifiers::Weak) | 
|  | 7755 | return QualType(); | 
|  | 7756 |  | 
|  | 7757 | if (GC_L == Qualifiers::Strong && RHSCan->isObjCObjectPointerType()) { | 
|  | 7758 | return mergeTypes(LHS, getObjCGCQualType(RHS, Qualifiers::Strong)); | 
|  | 7759 | } | 
|  | 7760 | if (GC_R == Qualifiers::Strong && LHSCan->isObjCObjectPointerType()) { | 
|  | 7761 | return mergeTypes(getObjCGCQualType(LHS, Qualifiers::Strong), RHS); | 
|  | 7762 | } | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7763 | return QualType(); | 
| John McCall | 8ccfcb5 | 2009-09-24 19:53:00 +0000 | [diff] [blame] | 7764 | } | 
|  | 7765 |  | 
|  | 7766 | // Okay, qualifiers are equal. | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7767 |  | 
| Eli Friedman | dcca633 | 2009-06-01 01:22:52 +0000 | [diff] [blame] | 7768 | Type::TypeClass LHSClass = LHSCan->getTypeClass(); | 
|  | 7769 | Type::TypeClass RHSClass = RHSCan->getTypeClass(); | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7770 |  | 
| Chris Lattner | fd65291 | 2008-01-14 05:45:46 +0000 | [diff] [blame] | 7771 | // We want to consider the two function types to be the same for these | 
|  | 7772 | // comparisons, just force one to the other. | 
|  | 7773 | if (LHSClass == Type::FunctionProto) LHSClass = Type::FunctionNoProto; | 
|  | 7774 | if (RHSClass == Type::FunctionProto) RHSClass = Type::FunctionNoProto; | 
| Eli Friedman | 16f9096 | 2008-02-12 08:23:06 +0000 | [diff] [blame] | 7775 |  | 
|  | 7776 | // Same as above for arrays | 
| Chris Lattner | 9555466 | 2008-04-07 05:43:21 +0000 | [diff] [blame] | 7777 | if (LHSClass == Type::VariableArray || LHSClass == Type::IncompleteArray) | 
|  | 7778 | LHSClass = Type::ConstantArray; | 
|  | 7779 | if (RHSClass == Type::VariableArray || RHSClass == Type::IncompleteArray) | 
|  | 7780 | RHSClass = Type::ConstantArray; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 7781 |  | 
| John McCall | 8b07ec2 | 2010-05-15 11:32:37 +0000 | [diff] [blame] | 7782 | // ObjCInterfaces are just specialized ObjCObjects. | 
|  | 7783 | if (LHSClass == Type::ObjCInterface) LHSClass = Type::ObjCObject; | 
|  | 7784 | if (RHSClass == Type::ObjCInterface) RHSClass = Type::ObjCObject; | 
|  | 7785 |  | 
| Nate Begeman | ce4d7fc | 2008-04-18 23:10:10 +0000 | [diff] [blame] | 7786 | // Canonicalize ExtVector -> Vector. | 
|  | 7787 | if (LHSClass == Type::ExtVector) LHSClass = Type::Vector; | 
|  | 7788 | if (RHSClass == Type::ExtVector) RHSClass = Type::Vector; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 7789 |  | 
| Chris Lattner | 9555466 | 2008-04-07 05:43:21 +0000 | [diff] [blame] | 7790 | // If the canonical type classes don't match. | 
| Chris Lattner | fd65291 | 2008-01-14 05:45:46 +0000 | [diff] [blame] | 7791 | if (LHSClass != RHSClass) { | 
| John McCall | 433c2e6 | 2013-03-21 00:10:07 +0000 | [diff] [blame] | 7792 | // Note that we only have special rules for turning block enum | 
|  | 7793 | // returns into block int returns, not vice-versa. | 
| John McCall | 9dd450b | 2009-09-21 23:43:11 +0000 | [diff] [blame] | 7794 | if (const EnumType* ETy = LHS->getAs<EnumType>()) { | 
| John McCall | 433c2e6 | 2013-03-21 00:10:07 +0000 | [diff] [blame] | 7795 | return mergeEnumWithInteger(*this, ETy, RHS, false); | 
| Eli Friedman | a7bf7ed | 2008-02-12 08:46:17 +0000 | [diff] [blame] | 7796 | } | 
| John McCall | 9dd450b | 2009-09-21 23:43:11 +0000 | [diff] [blame] | 7797 | if (const EnumType* ETy = RHS->getAs<EnumType>()) { | 
| John McCall | 433c2e6 | 2013-03-21 00:10:07 +0000 | [diff] [blame] | 7798 | return mergeEnumWithInteger(*this, ETy, LHS, BlockReturnType); | 
| Eli Friedman | a7bf7ed | 2008-02-12 08:46:17 +0000 | [diff] [blame] | 7799 | } | 
| Fariborz Jahanian | 26d8371 | 2012-01-26 00:45:38 +0000 | [diff] [blame] | 7800 | // allow block pointer type to match an 'id' type. | 
| Fariborz Jahanian | 194904e | 2012-01-26 17:08:50 +0000 | [diff] [blame] | 7801 | if (OfBlockPointer && !BlockReturnType) { | 
|  | 7802 | if (LHS->isObjCIdType() && RHS->isBlockPointerType()) | 
|  | 7803 | return LHS; | 
|  | 7804 | if (RHS->isObjCIdType() && LHS->isBlockPointerType()) | 
|  | 7805 | return RHS; | 
|  | 7806 | } | 
| Fariborz Jahanian | 26d8371 | 2012-01-26 00:45:38 +0000 | [diff] [blame] | 7807 |  | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7808 | return QualType(); | 
| Steve Naroff | 32e44c0 | 2007-10-15 20:41:53 +0000 | [diff] [blame] | 7809 | } | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7810 |  | 
| Steve Naroff | c6edcbd | 2008-01-09 22:43:08 +0000 | [diff] [blame] | 7811 | // The canonical type classes match. | 
| Chris Lattner | fd65291 | 2008-01-14 05:45:46 +0000 | [diff] [blame] | 7812 | switch (LHSClass) { | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 7813 | #define TYPE(Class, Base) | 
|  | 7814 | #define ABSTRACT_TYPE(Class, Base) | 
| John McCall | bd8d9bd | 2010-03-01 23:49:17 +0000 | [diff] [blame] | 7815 | #define NON_CANONICAL_UNLESS_DEPENDENT_TYPE(Class, Base) case Type::Class: | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 7816 | #define NON_CANONICAL_TYPE(Class, Base) case Type::Class: | 
|  | 7817 | #define DEPENDENT_TYPE(Class, Base) case Type::Class: | 
|  | 7818 | #include "clang/AST/TypeNodes.def" | 
| David Blaikie | 83d382b | 2011-09-23 05:06:16 +0000 | [diff] [blame] | 7819 | llvm_unreachable("Non-canonical and dependent types shouldn't get here"); | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 7820 |  | 
| Richard Smith | 27d807c | 2013-04-30 13:56:41 +0000 | [diff] [blame] | 7821 | case Type::Auto: | 
| Sebastian Redl | 0f8b23f | 2009-03-16 23:22:08 +0000 | [diff] [blame] | 7822 | case Type::LValueReference: | 
|  | 7823 | case Type::RValueReference: | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 7824 | case Type::MemberPointer: | 
| David Blaikie | 83d382b | 2011-09-23 05:06:16 +0000 | [diff] [blame] | 7825 | llvm_unreachable("C++ should never be in mergeTypes"); | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 7826 |  | 
| John McCall | 8b07ec2 | 2010-05-15 11:32:37 +0000 | [diff] [blame] | 7827 | case Type::ObjCInterface: | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 7828 | case Type::IncompleteArray: | 
|  | 7829 | case Type::VariableArray: | 
|  | 7830 | case Type::FunctionProto: | 
|  | 7831 | case Type::ExtVector: | 
| David Blaikie | 83d382b | 2011-09-23 05:06:16 +0000 | [diff] [blame] | 7832 | llvm_unreachable("Types are eliminated above"); | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 7833 |  | 
| Chris Lattner | fd65291 | 2008-01-14 05:45:46 +0000 | [diff] [blame] | 7834 | case Type::Pointer: | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7835 | { | 
|  | 7836 | // Merge two pointer types, while trying to preserve typedef info | 
| Ted Kremenek | c23c7e6 | 2009-07-29 21:53:49 +0000 | [diff] [blame] | 7837 | QualType LHSPointee = LHS->getAs<PointerType>()->getPointeeType(); | 
|  | 7838 | QualType RHSPointee = RHS->getAs<PointerType>()->getPointeeType(); | 
| Douglas Gregor | 17ea3f5 | 2010-07-29 15:18:02 +0000 | [diff] [blame] | 7839 | if (Unqualified) { | 
|  | 7840 | LHSPointee = LHSPointee.getUnqualifiedType(); | 
|  | 7841 | RHSPointee = RHSPointee.getUnqualifiedType(); | 
|  | 7842 | } | 
|  | 7843 | QualType ResultType = mergeTypes(LHSPointee, RHSPointee, false, | 
|  | 7844 | Unqualified); | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7845 | if (ResultType.isNull()) return QualType(); | 
| Eli Friedman | 091a9ac | 2009-06-02 05:28:56 +0000 | [diff] [blame] | 7846 | if (getCanonicalType(LHSPointee) == getCanonicalType(ResultType)) | 
| Chris Lattner | 465fa32 | 2008-10-05 17:34:18 +0000 | [diff] [blame] | 7847 | return LHS; | 
| Eli Friedman | 091a9ac | 2009-06-02 05:28:56 +0000 | [diff] [blame] | 7848 | if (getCanonicalType(RHSPointee) == getCanonicalType(ResultType)) | 
| Chris Lattner | 465fa32 | 2008-10-05 17:34:18 +0000 | [diff] [blame] | 7849 | return RHS; | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7850 | return getPointerType(ResultType); | 
|  | 7851 | } | 
| Steve Naroff | 68e167d | 2008-12-10 17:49:55 +0000 | [diff] [blame] | 7852 | case Type::BlockPointer: | 
|  | 7853 | { | 
|  | 7854 | // Merge two block pointer types, while trying to preserve typedef info | 
| Ted Kremenek | c23c7e6 | 2009-07-29 21:53:49 +0000 | [diff] [blame] | 7855 | QualType LHSPointee = LHS->getAs<BlockPointerType>()->getPointeeType(); | 
|  | 7856 | QualType RHSPointee = RHS->getAs<BlockPointerType>()->getPointeeType(); | 
| Douglas Gregor | 17ea3f5 | 2010-07-29 15:18:02 +0000 | [diff] [blame] | 7857 | if (Unqualified) { | 
|  | 7858 | LHSPointee = LHSPointee.getUnqualifiedType(); | 
|  | 7859 | RHSPointee = RHSPointee.getUnqualifiedType(); | 
|  | 7860 | } | 
|  | 7861 | QualType ResultType = mergeTypes(LHSPointee, RHSPointee, OfBlockPointer, | 
|  | 7862 | Unqualified); | 
| Steve Naroff | 68e167d | 2008-12-10 17:49:55 +0000 | [diff] [blame] | 7863 | if (ResultType.isNull()) return QualType(); | 
|  | 7864 | if (getCanonicalType(LHSPointee) == getCanonicalType(ResultType)) | 
|  | 7865 | return LHS; | 
|  | 7866 | if (getCanonicalType(RHSPointee) == getCanonicalType(ResultType)) | 
|  | 7867 | return RHS; | 
|  | 7868 | return getBlockPointerType(ResultType); | 
|  | 7869 | } | 
| Eli Friedman | 0dfb889 | 2011-10-06 23:00:33 +0000 | [diff] [blame] | 7870 | case Type::Atomic: | 
|  | 7871 | { | 
|  | 7872 | // Merge two pointer types, while trying to preserve typedef info | 
|  | 7873 | QualType LHSValue = LHS->getAs<AtomicType>()->getValueType(); | 
|  | 7874 | QualType RHSValue = RHS->getAs<AtomicType>()->getValueType(); | 
|  | 7875 | if (Unqualified) { | 
|  | 7876 | LHSValue = LHSValue.getUnqualifiedType(); | 
|  | 7877 | RHSValue = RHSValue.getUnqualifiedType(); | 
|  | 7878 | } | 
|  | 7879 | QualType ResultType = mergeTypes(LHSValue, RHSValue, false, | 
|  | 7880 | Unqualified); | 
|  | 7881 | if (ResultType.isNull()) return QualType(); | 
|  | 7882 | if (getCanonicalType(LHSValue) == getCanonicalType(ResultType)) | 
|  | 7883 | return LHS; | 
|  | 7884 | if (getCanonicalType(RHSValue) == getCanonicalType(ResultType)) | 
|  | 7885 | return RHS; | 
|  | 7886 | return getAtomicType(ResultType); | 
|  | 7887 | } | 
| Chris Lattner | fd65291 | 2008-01-14 05:45:46 +0000 | [diff] [blame] | 7888 | case Type::ConstantArray: | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7889 | { | 
|  | 7890 | const ConstantArrayType* LCAT = getAsConstantArrayType(LHS); | 
|  | 7891 | const ConstantArrayType* RCAT = getAsConstantArrayType(RHS); | 
|  | 7892 | if (LCAT && RCAT && RCAT->getSize() != LCAT->getSize()) | 
|  | 7893 | return QualType(); | 
|  | 7894 |  | 
|  | 7895 | QualType LHSElem = getAsArrayType(LHS)->getElementType(); | 
|  | 7896 | QualType RHSElem = getAsArrayType(RHS)->getElementType(); | 
| Douglas Gregor | 17ea3f5 | 2010-07-29 15:18:02 +0000 | [diff] [blame] | 7897 | if (Unqualified) { | 
|  | 7898 | LHSElem = LHSElem.getUnqualifiedType(); | 
|  | 7899 | RHSElem = RHSElem.getUnqualifiedType(); | 
|  | 7900 | } | 
|  | 7901 |  | 
|  | 7902 | QualType ResultType = mergeTypes(LHSElem, RHSElem, false, Unqualified); | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7903 | if (ResultType.isNull()) return QualType(); | 
| Chris Lattner | 465fa32 | 2008-10-05 17:34:18 +0000 | [diff] [blame] | 7904 | if (LCAT && getCanonicalType(LHSElem) == getCanonicalType(ResultType)) | 
|  | 7905 | return LHS; | 
|  | 7906 | if (RCAT && getCanonicalType(RHSElem) == getCanonicalType(ResultType)) | 
|  | 7907 | return RHS; | 
| Eli Friedman | 3e62c21 | 2008-08-22 01:48:21 +0000 | [diff] [blame] | 7908 | if (LCAT) return getConstantArrayType(ResultType, LCAT->getSize(), | 
|  | 7909 | ArrayType::ArraySizeModifier(), 0); | 
|  | 7910 | if (RCAT) return getConstantArrayType(ResultType, RCAT->getSize(), | 
|  | 7911 | ArrayType::ArraySizeModifier(), 0); | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7912 | const VariableArrayType* LVAT = getAsVariableArrayType(LHS); | 
|  | 7913 | const VariableArrayType* RVAT = getAsVariableArrayType(RHS); | 
| Chris Lattner | 465fa32 | 2008-10-05 17:34:18 +0000 | [diff] [blame] | 7914 | if (LVAT && getCanonicalType(LHSElem) == getCanonicalType(ResultType)) | 
|  | 7915 | return LHS; | 
|  | 7916 | if (RVAT && getCanonicalType(RHSElem) == getCanonicalType(ResultType)) | 
|  | 7917 | return RHS; | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7918 | if (LVAT) { | 
|  | 7919 | // FIXME: This isn't correct! But tricky to implement because | 
|  | 7920 | // the array's size has to be the size of LHS, but the type | 
|  | 7921 | // has to be different. | 
|  | 7922 | return LHS; | 
|  | 7923 | } | 
|  | 7924 | if (RVAT) { | 
|  | 7925 | // FIXME: This isn't correct! But tricky to implement because | 
|  | 7926 | // the array's size has to be the size of RHS, but the type | 
|  | 7927 | // has to be different. | 
|  | 7928 | return RHS; | 
|  | 7929 | } | 
| Eli Friedman | 3e62c21 | 2008-08-22 01:48:21 +0000 | [diff] [blame] | 7930 | if (getCanonicalType(LHSElem) == getCanonicalType(ResultType)) return LHS; | 
|  | 7931 | if (getCanonicalType(RHSElem) == getCanonicalType(ResultType)) return RHS; | 
| Douglas Gregor | 0431825 | 2009-07-06 15:59:29 +0000 | [diff] [blame] | 7932 | return getIncompleteArrayType(ResultType, | 
|  | 7933 | ArrayType::ArraySizeModifier(), 0); | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7934 | } | 
| Chris Lattner | fd65291 | 2008-01-14 05:45:46 +0000 | [diff] [blame] | 7935 | case Type::FunctionNoProto: | 
| Douglas Gregor | 17ea3f5 | 2010-07-29 15:18:02 +0000 | [diff] [blame] | 7936 | return mergeFunctionTypes(LHS, RHS, OfBlockPointer, Unqualified); | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 7937 | case Type::Record: | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 7938 | case Type::Enum: | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7939 | return QualType(); | 
| Chris Lattner | fd65291 | 2008-01-14 05:45:46 +0000 | [diff] [blame] | 7940 | case Type::Builtin: | 
| Chris Lattner | 7bbd3d7 | 2008-04-07 05:55:38 +0000 | [diff] [blame] | 7941 | // Only exactly equal builtin types are compatible, which is tested above. | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7942 | return QualType(); | 
| Daniel Dunbar | 804c044 | 2009-01-28 21:22:12 +0000 | [diff] [blame] | 7943 | case Type::Complex: | 
|  | 7944 | // Distinct complex types are incompatible. | 
|  | 7945 | return QualType(); | 
| Chris Lattner | 7bbd3d7 | 2008-04-07 05:55:38 +0000 | [diff] [blame] | 7946 | case Type::Vector: | 
| Eli Friedman | cad9638 | 2009-02-27 23:04:43 +0000 | [diff] [blame] | 7947 | // FIXME: The merged type should be an ExtVector! | 
| John McCall | 44c064b | 2010-03-12 23:14:13 +0000 | [diff] [blame] | 7948 | if (areCompatVectorTypes(LHSCan->getAs<VectorType>(), | 
|  | 7949 | RHSCan->getAs<VectorType>())) | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7950 | return LHS; | 
| Chris Lattner | 465fa32 | 2008-10-05 17:34:18 +0000 | [diff] [blame] | 7951 | return QualType(); | 
| John McCall | 8b07ec2 | 2010-05-15 11:32:37 +0000 | [diff] [blame] | 7952 | case Type::ObjCObject: { | 
|  | 7953 | // Check if the types are assignment compatible. | 
| Eli Friedman | cad9638 | 2009-02-27 23:04:43 +0000 | [diff] [blame] | 7954 | // FIXME: This should be type compatibility, e.g. whether | 
|  | 7955 | // "LHS x; RHS x;" at global scope is legal. | 
| John McCall | 8b07ec2 | 2010-05-15 11:32:37 +0000 | [diff] [blame] | 7956 | const ObjCObjectType* LHSIface = LHS->getAs<ObjCObjectType>(); | 
|  | 7957 | const ObjCObjectType* RHSIface = RHS->getAs<ObjCObjectType>(); | 
|  | 7958 | if (canAssignObjCInterfaces(LHSIface, RHSIface)) | 
| Steve Naroff | 7a7814c | 2009-02-21 16:18:07 +0000 | [diff] [blame] | 7959 | return LHS; | 
|  | 7960 |  | 
| Eli Friedman | 47f7711 | 2008-08-22 00:56:42 +0000 | [diff] [blame] | 7961 | return QualType(); | 
| Cedric Venet | 4fc88b7 | 2009-02-21 17:14:49 +0000 | [diff] [blame] | 7962 | } | 
| Steve Naroff | 7cae42b | 2009-07-10 23:34:53 +0000 | [diff] [blame] | 7963 | case Type::ObjCObjectPointer: { | 
| Fariborz Jahanian | b8b0ea3 | 2010-03-17 00:20:01 +0000 | [diff] [blame] | 7964 | if (OfBlockPointer) { | 
|  | 7965 | if (canAssignObjCInterfacesInBlockPointer( | 
|  | 7966 | LHS->getAs<ObjCObjectPointerType>(), | 
| Fariborz Jahanian | 90186f8 | 2011-03-14 16:07:00 +0000 | [diff] [blame] | 7967 | RHS->getAs<ObjCObjectPointerType>(), | 
|  | 7968 | BlockReturnType)) | 
| David Blaikie | 8a40f70 | 2012-01-17 06:56:22 +0000 | [diff] [blame] | 7969 | return LHS; | 
| Fariborz Jahanian | b8b0ea3 | 2010-03-17 00:20:01 +0000 | [diff] [blame] | 7970 | return QualType(); | 
|  | 7971 | } | 
| John McCall | 9dd450b | 2009-09-21 23:43:11 +0000 | [diff] [blame] | 7972 | if (canAssignObjCInterfaces(LHS->getAs<ObjCObjectPointerType>(), | 
|  | 7973 | RHS->getAs<ObjCObjectPointerType>())) | 
| Steve Naroff | 7cae42b | 2009-07-10 23:34:53 +0000 | [diff] [blame] | 7974 | return LHS; | 
|  | 7975 |  | 
| Steve Naroff | c68cfcf | 2008-12-10 22:14:21 +0000 | [diff] [blame] | 7976 | return QualType(); | 
| David Blaikie | 8a40f70 | 2012-01-17 06:56:22 +0000 | [diff] [blame] | 7977 | } | 
| Xiuli Pan | 9c14e28 | 2016-01-09 12:53:17 +0000 | [diff] [blame] | 7978 | case Type::Pipe: | 
|  | 7979 | { | 
|  | 7980 | // Merge two pointer types, while trying to preserve typedef info | 
|  | 7981 | QualType LHSValue = LHS->getAs<PipeType>()->getElementType(); | 
|  | 7982 | QualType RHSValue = RHS->getAs<PipeType>()->getElementType(); | 
|  | 7983 | if (Unqualified) { | 
|  | 7984 | LHSValue = LHSValue.getUnqualifiedType(); | 
|  | 7985 | RHSValue = RHSValue.getUnqualifiedType(); | 
|  | 7986 | } | 
|  | 7987 | QualType ResultType = mergeTypes(LHSValue, RHSValue, false, | 
|  | 7988 | Unqualified); | 
|  | 7989 | if (ResultType.isNull()) return QualType(); | 
|  | 7990 | if (getCanonicalType(LHSValue) == getCanonicalType(ResultType)) | 
|  | 7991 | return LHS; | 
|  | 7992 | if (getCanonicalType(RHSValue) == getCanonicalType(ResultType)) | 
|  | 7993 | return RHS; | 
|  | 7994 | return getPipeType(ResultType); | 
|  | 7995 | } | 
| Steve Naroff | 32e44c0 | 2007-10-15 20:41:53 +0000 | [diff] [blame] | 7996 | } | 
| Douglas Gregor | deaad8c | 2009-02-26 23:50:07 +0000 | [diff] [blame] | 7997 |  | 
| David Blaikie | 8a40f70 | 2012-01-17 06:56:22 +0000 | [diff] [blame] | 7998 | llvm_unreachable("Invalid Type::Class!"); | 
| Steve Naroff | 32e44c0 | 2007-10-15 20:41:53 +0000 | [diff] [blame] | 7999 | } | 
| Ted Kremenek | fc581a9 | 2007-10-31 17:10:13 +0000 | [diff] [blame] | 8000 |  | 
| John McCall | 18afab7 | 2016-03-01 00:49:02 +0000 | [diff] [blame] | 8001 | bool ASTContext::doFunctionTypesMatchOnExtParameterInfos( | 
|  | 8002 | const FunctionProtoType *firstFnType, | 
|  | 8003 | const FunctionProtoType *secondFnType) { | 
|  | 8004 | // Fast path: if the first type doesn't have ext parameter infos, | 
|  | 8005 | // we match if and only if they second type also doesn't have them. | 
|  | 8006 | if (!firstFnType->hasExtParameterInfos()) | 
|  | 8007 | return !secondFnType->hasExtParameterInfos(); | 
|  | 8008 |  | 
|  | 8009 | // Otherwise, we can only match if the second type has them. | 
|  | 8010 | if (!secondFnType->hasExtParameterInfos()) | 
| Fariborz Jahanian | 9767697 | 2011-09-28 21:52:05 +0000 | [diff] [blame] | 8011 | return false; | 
| John McCall | 18afab7 | 2016-03-01 00:49:02 +0000 | [diff] [blame] | 8012 |  | 
|  | 8013 | auto firstEPI = firstFnType->getExtParameterInfos(); | 
|  | 8014 | auto secondEPI = secondFnType->getExtParameterInfos(); | 
|  | 8015 | assert(firstEPI.size() == secondEPI.size()); | 
|  | 8016 |  | 
|  | 8017 | for (size_t i = 0, n = firstEPI.size(); i != n; ++i) { | 
|  | 8018 | if (firstEPI[i] != secondEPI[i]) | 
|  | 8019 | return false; | 
|  | 8020 | } | 
| Fariborz Jahanian | 9767697 | 2011-09-28 21:52:05 +0000 | [diff] [blame] | 8021 | return true; | 
|  | 8022 | } | 
|  | 8023 |  | 
| Chandler Carruth | 21c9060 | 2015-12-30 03:24:14 +0000 | [diff] [blame] | 8024 | void ASTContext::ResetObjCLayout(const ObjCContainerDecl *CD) { | 
|  | 8025 | ObjCLayouts[CD] = nullptr; | 
|  | 8026 | } | 
|  | 8027 |  | 
| Fariborz Jahanian | f633ebd | 2010-05-19 21:37:30 +0000 | [diff] [blame] | 8028 | /// mergeObjCGCQualifiers - This routine merges ObjC's GC attribute of 'LHS' and | 
|  | 8029 | /// 'RHS' attributes and returns the merged version; including for function | 
|  | 8030 | /// return types. | 
|  | 8031 | QualType ASTContext::mergeObjCGCQualifiers(QualType LHS, QualType RHS) { | 
|  | 8032 | QualType LHSCan = getCanonicalType(LHS), | 
|  | 8033 | RHSCan = getCanonicalType(RHS); | 
|  | 8034 | // If two types are identical, they are compatible. | 
|  | 8035 | if (LHSCan == RHSCan) | 
|  | 8036 | return LHS; | 
|  | 8037 | if (RHSCan->isFunctionType()) { | 
|  | 8038 | if (!LHSCan->isFunctionType()) | 
|  | 8039 | return QualType(); | 
| Alp Toker | 314cc81 | 2014-01-25 16:55:45 +0000 | [diff] [blame] | 8040 | QualType OldReturnType = | 
|  | 8041 | cast<FunctionType>(RHSCan.getTypePtr())->getReturnType(); | 
| Fariborz Jahanian | f633ebd | 2010-05-19 21:37:30 +0000 | [diff] [blame] | 8042 | QualType NewReturnType = | 
| Alp Toker | 314cc81 | 2014-01-25 16:55:45 +0000 | [diff] [blame] | 8043 | cast<FunctionType>(LHSCan.getTypePtr())->getReturnType(); | 
| Fariborz Jahanian | f633ebd | 2010-05-19 21:37:30 +0000 | [diff] [blame] | 8044 | QualType ResReturnType = | 
|  | 8045 | mergeObjCGCQualifiers(NewReturnType, OldReturnType); | 
|  | 8046 | if (ResReturnType.isNull()) | 
|  | 8047 | return QualType(); | 
|  | 8048 | if (ResReturnType == NewReturnType || ResReturnType == OldReturnType) { | 
|  | 8049 | // id foo(); ... __strong id foo(); or: __strong id foo(); ... id foo(); | 
|  | 8050 | // In either case, use OldReturnType to build the new function type. | 
|  | 8051 | const FunctionType *F = LHS->getAs<FunctionType>(); | 
|  | 8052 | if (const FunctionProtoType *FPT = cast<FunctionProtoType>(F)) { | 
| John McCall | db40c7f | 2010-12-14 08:05:40 +0000 | [diff] [blame] | 8053 | FunctionProtoType::ExtProtoInfo EPI = FPT->getExtProtoInfo(); | 
|  | 8054 | EPI.ExtInfo = getFunctionExtInfo(LHS); | 
| Reid Kleckner | 896b32f | 2013-06-10 20:51:09 +0000 | [diff] [blame] | 8055 | QualType ResultType = | 
| Alp Toker | 9cacbab | 2014-01-20 20:26:09 +0000 | [diff] [blame] | 8056 | getFunctionType(OldReturnType, FPT->getParamTypes(), EPI); | 
| Fariborz Jahanian | f633ebd | 2010-05-19 21:37:30 +0000 | [diff] [blame] | 8057 | return ResultType; | 
|  | 8058 | } | 
|  | 8059 | } | 
|  | 8060 | return QualType(); | 
|  | 8061 | } | 
|  | 8062 |  | 
|  | 8063 | // If the qualifiers are different, the types can still be merged. | 
|  | 8064 | Qualifiers LQuals = LHSCan.getLocalQualifiers(); | 
|  | 8065 | Qualifiers RQuals = RHSCan.getLocalQualifiers(); | 
|  | 8066 | if (LQuals != RQuals) { | 
|  | 8067 | // If any of these qualifiers are different, we have a type mismatch. | 
|  | 8068 | if (LQuals.getCVRQualifiers() != RQuals.getCVRQualifiers() || | 
|  | 8069 | LQuals.getAddressSpace() != RQuals.getAddressSpace()) | 
|  | 8070 | return QualType(); | 
|  | 8071 |  | 
|  | 8072 | // Exactly one GC qualifier difference is allowed: __strong is | 
|  | 8073 | // okay if the other type has no GC qualifier but is an Objective | 
|  | 8074 | // C object pointer (i.e. implicitly strong by default).  We fix | 
|  | 8075 | // this by pretending that the unqualified type was actually | 
|  | 8076 | // qualified __strong. | 
|  | 8077 | Qualifiers::GC GC_L = LQuals.getObjCGCAttr(); | 
|  | 8078 | Qualifiers::GC GC_R = RQuals.getObjCGCAttr(); | 
|  | 8079 | assert((GC_L != GC_R) && "unequal qualifier sets had only equal elements"); | 
|  | 8080 |  | 
|  | 8081 | if (GC_L == Qualifiers::Weak || GC_R == Qualifiers::Weak) | 
|  | 8082 | return QualType(); | 
|  | 8083 |  | 
|  | 8084 | if (GC_L == Qualifiers::Strong) | 
|  | 8085 | return LHS; | 
|  | 8086 | if (GC_R == Qualifiers::Strong) | 
|  | 8087 | return RHS; | 
|  | 8088 | return QualType(); | 
|  | 8089 | } | 
|  | 8090 |  | 
|  | 8091 | if (LHSCan->isObjCObjectPointerType() && RHSCan->isObjCObjectPointerType()) { | 
|  | 8092 | QualType LHSBaseQT = LHS->getAs<ObjCObjectPointerType>()->getPointeeType(); | 
|  | 8093 | QualType RHSBaseQT = RHS->getAs<ObjCObjectPointerType>()->getPointeeType(); | 
|  | 8094 | QualType ResQT = mergeObjCGCQualifiers(LHSBaseQT, RHSBaseQT); | 
|  | 8095 | if (ResQT == LHSBaseQT) | 
|  | 8096 | return LHS; | 
|  | 8097 | if (ResQT == RHSBaseQT) | 
|  | 8098 | return RHS; | 
|  | 8099 | } | 
|  | 8100 | return QualType(); | 
|  | 8101 | } | 
|  | 8102 |  | 
| Chris Lattner | 4ba0cef | 2008-04-07 07:01:58 +0000 | [diff] [blame] | 8103 | //===----------------------------------------------------------------------===// | 
| Eli Friedman | 4f89ccb | 2008-06-28 06:23:08 +0000 | [diff] [blame] | 8104 | //                         Integer Predicates | 
|  | 8105 | //===----------------------------------------------------------------------===// | 
| Chris Lattner | 3c91971 | 2009-01-16 07:15:35 +0000 | [diff] [blame] | 8106 |  | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 8107 | unsigned ASTContext::getIntWidth(QualType T) const { | 
| Richard Smith | e952106 | 2013-10-15 04:56:17 +0000 | [diff] [blame] | 8108 | if (const EnumType *ET = T->getAs<EnumType>()) | 
| Eli Friedman | ee275c8 | 2009-12-10 22:29:29 +0000 | [diff] [blame] | 8109 | T = ET->getDecl()->getIntegerType(); | 
| Douglas Gregor | 0bf3140 | 2010-10-08 23:50:27 +0000 | [diff] [blame] | 8110 | if (T->isBooleanType()) | 
|  | 8111 | return 1; | 
| Eli Friedman | 1efaaea | 2009-02-13 02:31:07 +0000 | [diff] [blame] | 8112 | // For builtin types, just use the standard type sizing method | 
| Eli Friedman | 4f89ccb | 2008-06-28 06:23:08 +0000 | [diff] [blame] | 8113 | return (unsigned)getTypeSize(T); | 
|  | 8114 | } | 
|  | 8115 |  | 
| Abramo Bagnara | 1364049 | 2012-09-09 10:21:24 +0000 | [diff] [blame] | 8116 | QualType ASTContext::getCorrespondingUnsignedType(QualType T) const { | 
| Douglas Gregor | 5cc2c8b | 2010-07-23 15:58:24 +0000 | [diff] [blame] | 8117 | assert(T->hasSignedIntegerRepresentation() && "Unexpected type"); | 
| Chris Lattner | ec3a156 | 2009-10-17 20:33:28 +0000 | [diff] [blame] | 8118 |  | 
|  | 8119 | // Turn <4 x signed int> -> <4 x unsigned int> | 
|  | 8120 | if (const VectorType *VTy = T->getAs<VectorType>()) | 
|  | 8121 | return getVectorType(getCorrespondingUnsignedType(VTy->getElementType()), | 
| Bob Wilson | aeb5644 | 2010-11-10 21:56:12 +0000 | [diff] [blame] | 8122 | VTy->getNumElements(), VTy->getVectorKind()); | 
| Chris Lattner | ec3a156 | 2009-10-17 20:33:28 +0000 | [diff] [blame] | 8123 |  | 
|  | 8124 | // For enums, we return the unsigned version of the base type. | 
|  | 8125 | if (const EnumType *ETy = T->getAs<EnumType>()) | 
| Eli Friedman | 4f89ccb | 2008-06-28 06:23:08 +0000 | [diff] [blame] | 8126 | T = ETy->getDecl()->getIntegerType(); | 
| Chris Lattner | ec3a156 | 2009-10-17 20:33:28 +0000 | [diff] [blame] | 8127 |  | 
|  | 8128 | const BuiltinType *BTy = T->getAs<BuiltinType>(); | 
|  | 8129 | assert(BTy && "Unexpected signed integer type"); | 
| Eli Friedman | 4f89ccb | 2008-06-28 06:23:08 +0000 | [diff] [blame] | 8130 | switch (BTy->getKind()) { | 
|  | 8131 | case BuiltinType::Char_S: | 
|  | 8132 | case BuiltinType::SChar: | 
|  | 8133 | return UnsignedCharTy; | 
|  | 8134 | case BuiltinType::Short: | 
|  | 8135 | return UnsignedShortTy; | 
|  | 8136 | case BuiltinType::Int: | 
|  | 8137 | return UnsignedIntTy; | 
|  | 8138 | case BuiltinType::Long: | 
|  | 8139 | return UnsignedLongTy; | 
|  | 8140 | case BuiltinType::LongLong: | 
|  | 8141 | return UnsignedLongLongTy; | 
| Chris Lattner | f122cef | 2009-04-30 02:43:43 +0000 | [diff] [blame] | 8142 | case BuiltinType::Int128: | 
|  | 8143 | return UnsignedInt128Ty; | 
| Eli Friedman | 4f89ccb | 2008-06-28 06:23:08 +0000 | [diff] [blame] | 8144 | default: | 
| David Blaikie | 83d382b | 2011-09-23 05:06:16 +0000 | [diff] [blame] | 8145 | llvm_unreachable("Unexpected signed integer type"); | 
| Eli Friedman | 4f89ccb | 2008-06-28 06:23:08 +0000 | [diff] [blame] | 8146 | } | 
|  | 8147 | } | 
|  | 8148 |  | 
| Angel Garcia Gomez | 637d1e6 | 2015-10-20 13:23:58 +0000 | [diff] [blame] | 8149 | ASTMutationListener::~ASTMutationListener() { } | 
| Argyrios Kyrtzidis | 65ad569 | 2010-10-24 17:26:36 +0000 | [diff] [blame] | 8150 |  | 
| Richard Smith | 1fa5d64 | 2013-05-11 05:45:24 +0000 | [diff] [blame] | 8151 | void ASTMutationListener::DeducedReturnType(const FunctionDecl *FD, | 
|  | 8152 | QualType ReturnType) {} | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8153 |  | 
|  | 8154 | //===----------------------------------------------------------------------===// | 
|  | 8155 | //                          Builtin Type Computation | 
|  | 8156 | //===----------------------------------------------------------------------===// | 
|  | 8157 |  | 
|  | 8158 | /// DecodeTypeFromStr - This decodes one type descriptor from Str, advancing the | 
| Chris Lattner | dc226c2 | 2010-10-01 22:42:38 +0000 | [diff] [blame] | 8159 | /// pointer over the consumed characters.  This returns the resultant type.  If | 
|  | 8160 | /// AllowTypeModifiers is false then modifier like * are not parsed, just basic | 
|  | 8161 | /// types.  This allows "v2i*" to be parsed as a pointer to a v2i instead of | 
|  | 8162 | /// a vector of "i*". | 
| Chris Lattner | bd6e693 | 2010-10-01 22:53:11 +0000 | [diff] [blame] | 8163 | /// | 
|  | 8164 | /// RequiresICE is filled in on return to indicate whether the value is required | 
|  | 8165 | /// to be an Integer Constant Expression. | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 8166 | static QualType DecodeTypeFromStr(const char *&Str, const ASTContext &Context, | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8167 | ASTContext::GetBuiltinTypeError &Error, | 
| Chris Lattner | bd6e693 | 2010-10-01 22:53:11 +0000 | [diff] [blame] | 8168 | bool &RequiresICE, | 
| Chris Lattner | dc226c2 | 2010-10-01 22:42:38 +0000 | [diff] [blame] | 8169 | bool AllowTypeModifiers) { | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8170 | // Modifiers. | 
|  | 8171 | int HowLong = 0; | 
|  | 8172 | bool Signed = false, Unsigned = false; | 
| Chris Lattner | bd6e693 | 2010-10-01 22:53:11 +0000 | [diff] [blame] | 8173 | RequiresICE = false; | 
| Chris Lattner | 8473339 | 2010-10-01 07:13:18 +0000 | [diff] [blame] | 8174 |  | 
| Chris Lattner | dc226c2 | 2010-10-01 22:42:38 +0000 | [diff] [blame] | 8175 | // Read the prefixed modifiers first. | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8176 | bool Done = false; | 
|  | 8177 | while (!Done) { | 
|  | 8178 | switch (*Str++) { | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 8179 | default: Done = true; --Str; break; | 
| Chris Lattner | 8473339 | 2010-10-01 07:13:18 +0000 | [diff] [blame] | 8180 | case 'I': | 
| Chris Lattner | bd6e693 | 2010-10-01 22:53:11 +0000 | [diff] [blame] | 8181 | RequiresICE = true; | 
| Chris Lattner | 8473339 | 2010-10-01 07:13:18 +0000 | [diff] [blame] | 8182 | break; | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8183 | case 'S': | 
|  | 8184 | assert(!Unsigned && "Can't use both 'S' and 'U' modifiers!"); | 
|  | 8185 | assert(!Signed && "Can't use 'S' modifier multiple times!"); | 
|  | 8186 | Signed = true; | 
|  | 8187 | break; | 
|  | 8188 | case 'U': | 
|  | 8189 | assert(!Signed && "Can't use both 'S' and 'U' modifiers!"); | 
| Sean Silva | 2a99514 | 2015-01-16 21:44:26 +0000 | [diff] [blame] | 8190 | assert(!Unsigned && "Can't use 'U' modifier multiple times!"); | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8191 | Unsigned = true; | 
|  | 8192 | break; | 
|  | 8193 | case 'L': | 
|  | 8194 | assert(HowLong <= 2 && "Can't have LLLL modifier"); | 
|  | 8195 | ++HowLong; | 
|  | 8196 | break; | 
| Kevin Qin | ad64f6d | 2014-02-24 02:45:03 +0000 | [diff] [blame] | 8197 | case 'W': | 
|  | 8198 | // This modifier represents int64 type. | 
|  | 8199 | assert(HowLong == 0 && "Can't use both 'L' and 'W' modifiers!"); | 
|  | 8200 | switch (Context.getTargetInfo().getInt64Type()) { | 
|  | 8201 | default: | 
|  | 8202 | llvm_unreachable("Unexpected integer type"); | 
|  | 8203 | case TargetInfo::SignedLong: | 
|  | 8204 | HowLong = 1; | 
|  | 8205 | break; | 
|  | 8206 | case TargetInfo::SignedLongLong: | 
|  | 8207 | HowLong = 2; | 
|  | 8208 | break; | 
|  | 8209 | } | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8210 | } | 
|  | 8211 | } | 
|  | 8212 |  | 
|  | 8213 | QualType Type; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 8214 |  | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8215 | // Read the base type. | 
|  | 8216 | switch (*Str++) { | 
| David Blaikie | 83d382b | 2011-09-23 05:06:16 +0000 | [diff] [blame] | 8217 | default: llvm_unreachable("Unknown builtin type letter!"); | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8218 | case 'v': | 
|  | 8219 | assert(HowLong == 0 && !Signed && !Unsigned && | 
|  | 8220 | "Bad modifiers used with 'v'!"); | 
|  | 8221 | Type = Context.VoidTy; | 
|  | 8222 | break; | 
| Jack Carter | 24bef98 | 2013-08-15 15:16:57 +0000 | [diff] [blame] | 8223 | case 'h': | 
|  | 8224 | assert(HowLong == 0 && !Signed && !Unsigned && | 
| Sean Silva | 2a99514 | 2015-01-16 21:44:26 +0000 | [diff] [blame] | 8225 | "Bad modifiers used with 'h'!"); | 
| Jack Carter | 24bef98 | 2013-08-15 15:16:57 +0000 | [diff] [blame] | 8226 | Type = Context.HalfTy; | 
|  | 8227 | break; | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8228 | case 'f': | 
|  | 8229 | assert(HowLong == 0 && !Signed && !Unsigned && | 
|  | 8230 | "Bad modifiers used with 'f'!"); | 
|  | 8231 | Type = Context.FloatTy; | 
|  | 8232 | break; | 
|  | 8233 | case 'd': | 
|  | 8234 | assert(HowLong < 2 && !Signed && !Unsigned && | 
|  | 8235 | "Bad modifiers used with 'd'!"); | 
|  | 8236 | if (HowLong) | 
|  | 8237 | Type = Context.LongDoubleTy; | 
|  | 8238 | else | 
|  | 8239 | Type = Context.DoubleTy; | 
|  | 8240 | break; | 
|  | 8241 | case 's': | 
|  | 8242 | assert(HowLong == 0 && "Bad modifiers used with 's'!"); | 
|  | 8243 | if (Unsigned) | 
|  | 8244 | Type = Context.UnsignedShortTy; | 
|  | 8245 | else | 
|  | 8246 | Type = Context.ShortTy; | 
|  | 8247 | break; | 
|  | 8248 | case 'i': | 
|  | 8249 | if (HowLong == 3) | 
|  | 8250 | Type = Unsigned ? Context.UnsignedInt128Ty : Context.Int128Ty; | 
|  | 8251 | else if (HowLong == 2) | 
|  | 8252 | Type = Unsigned ? Context.UnsignedLongLongTy : Context.LongLongTy; | 
|  | 8253 | else if (HowLong == 1) | 
|  | 8254 | Type = Unsigned ? Context.UnsignedLongTy : Context.LongTy; | 
|  | 8255 | else | 
|  | 8256 | Type = Unsigned ? Context.UnsignedIntTy : Context.IntTy; | 
|  | 8257 | break; | 
|  | 8258 | case 'c': | 
|  | 8259 | assert(HowLong == 0 && "Bad modifiers used with 'c'!"); | 
|  | 8260 | if (Signed) | 
|  | 8261 | Type = Context.SignedCharTy; | 
|  | 8262 | else if (Unsigned) | 
|  | 8263 | Type = Context.UnsignedCharTy; | 
|  | 8264 | else | 
|  | 8265 | Type = Context.CharTy; | 
|  | 8266 | break; | 
|  | 8267 | case 'b': // boolean | 
|  | 8268 | assert(HowLong == 0 && !Signed && !Unsigned && "Bad modifiers for 'b'!"); | 
|  | 8269 | Type = Context.BoolTy; | 
|  | 8270 | break; | 
|  | 8271 | case 'z':  // size_t. | 
|  | 8272 | assert(HowLong == 0 && !Signed && !Unsigned && "Bad modifiers for 'z'!"); | 
|  | 8273 | Type = Context.getSizeType(); | 
|  | 8274 | break; | 
|  | 8275 | case 'F': | 
|  | 8276 | Type = Context.getCFConstantStringType(); | 
|  | 8277 | break; | 
| Fariborz Jahanian | d11da7e | 2010-11-09 21:38:20 +0000 | [diff] [blame] | 8278 | case 'G': | 
|  | 8279 | Type = Context.getObjCIdType(); | 
|  | 8280 | break; | 
|  | 8281 | case 'H': | 
|  | 8282 | Type = Context.getObjCSelType(); | 
|  | 8283 | break; | 
| Fariborz Jahanian | cb6c867 | 2013-01-04 18:45:40 +0000 | [diff] [blame] | 8284 | case 'M': | 
|  | 8285 | Type = Context.getObjCSuperType(); | 
|  | 8286 | break; | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8287 | case 'a': | 
|  | 8288 | Type = Context.getBuiltinVaListType(); | 
|  | 8289 | assert(!Type.isNull() && "builtin va list type not initialized!"); | 
|  | 8290 | break; | 
|  | 8291 | case 'A': | 
|  | 8292 | // This is a "reference" to a va_list; however, what exactly | 
|  | 8293 | // this means depends on how va_list is defined. There are two | 
|  | 8294 | // different kinds of va_list: ones passed by value, and ones | 
|  | 8295 | // passed by reference.  An example of a by-value va_list is | 
|  | 8296 | // x86, where va_list is a char*. An example of by-ref va_list | 
|  | 8297 | // is x86-64, where va_list is a __va_list_tag[1]. For x86, | 
|  | 8298 | // we want this argument to be a char*&; for x86-64, we want | 
|  | 8299 | // it to be a __va_list_tag*. | 
|  | 8300 | Type = Context.getBuiltinVaListType(); | 
|  | 8301 | assert(!Type.isNull() && "builtin va list type not initialized!"); | 
| Chris Lattner | bd6e693 | 2010-10-01 22:53:11 +0000 | [diff] [blame] | 8302 | if (Type->isArrayType()) | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8303 | Type = Context.getArrayDecayedType(Type); | 
| Chris Lattner | bd6e693 | 2010-10-01 22:53:11 +0000 | [diff] [blame] | 8304 | else | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8305 | Type = Context.getLValueReferenceType(Type); | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8306 | break; | 
|  | 8307 | case 'V': { | 
|  | 8308 | char *End; | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8309 | unsigned NumElements = strtoul(Str, &End, 10); | 
|  | 8310 | assert(End != Str && "Missing vector size"); | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8311 | Str = End; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 8312 |  | 
| Chris Lattner | bd6e693 | 2010-10-01 22:53:11 +0000 | [diff] [blame] | 8313 | QualType ElementType = DecodeTypeFromStr(Str, Context, Error, | 
|  | 8314 | RequiresICE, false); | 
|  | 8315 | assert(!RequiresICE && "Can't require vector ICE"); | 
| Chris Lattner | dc226c2 | 2010-10-01 22:42:38 +0000 | [diff] [blame] | 8316 |  | 
|  | 8317 | // TODO: No way to make AltiVec vectors in builtins yet. | 
| Chris Lattner | 37141f4 | 2010-06-23 06:00:24 +0000 | [diff] [blame] | 8318 | Type = Context.getVectorType(ElementType, NumElements, | 
| Bob Wilson | aeb5644 | 2010-11-10 21:56:12 +0000 | [diff] [blame] | 8319 | VectorType::GenericVector); | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8320 | break; | 
|  | 8321 | } | 
| Douglas Gregor | fed6699 | 2012-06-07 18:08:25 +0000 | [diff] [blame] | 8322 | case 'E': { | 
|  | 8323 | char *End; | 
|  | 8324 |  | 
|  | 8325 | unsigned NumElements = strtoul(Str, &End, 10); | 
|  | 8326 | assert(End != Str && "Missing vector size"); | 
|  | 8327 |  | 
|  | 8328 | Str = End; | 
|  | 8329 |  | 
|  | 8330 | QualType ElementType = DecodeTypeFromStr(Str, Context, Error, RequiresICE, | 
|  | 8331 | false); | 
|  | 8332 | Type = Context.getExtVectorType(ElementType, NumElements); | 
|  | 8333 | break; | 
|  | 8334 | } | 
| Douglas Gregor | 40ef7c5 | 2009-09-28 21:45:01 +0000 | [diff] [blame] | 8335 | case 'X': { | 
| Chris Lattner | bd6e693 | 2010-10-01 22:53:11 +0000 | [diff] [blame] | 8336 | QualType ElementType = DecodeTypeFromStr(Str, Context, Error, RequiresICE, | 
|  | 8337 | false); | 
|  | 8338 | assert(!RequiresICE && "Can't require complex ICE"); | 
| Douglas Gregor | 40ef7c5 | 2009-09-28 21:45:01 +0000 | [diff] [blame] | 8339 | Type = Context.getComplexType(ElementType); | 
|  | 8340 | break; | 
| Fariborz Jahanian | 73952fc | 2011-08-23 23:33:09 +0000 | [diff] [blame] | 8341 | } | 
|  | 8342 | case 'Y' : { | 
|  | 8343 | Type = Context.getPointerDiffType(); | 
|  | 8344 | break; | 
|  | 8345 | } | 
| Chris Lattner | a58b3af | 2009-07-28 22:49:34 +0000 | [diff] [blame] | 8346 | case 'P': | 
| Douglas Gregor | 27821ce | 2009-07-07 16:35:42 +0000 | [diff] [blame] | 8347 | Type = Context.getFILEType(); | 
|  | 8348 | if (Type.isNull()) { | 
| Mike Stump | 93246cc | 2009-07-28 23:57:15 +0000 | [diff] [blame] | 8349 | Error = ASTContext::GE_Missing_stdio; | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8350 | return QualType(); | 
|  | 8351 | } | 
| Mike Stump | 2adb4da | 2009-07-28 23:47:15 +0000 | [diff] [blame] | 8352 | break; | 
| Chris Lattner | a58b3af | 2009-07-28 22:49:34 +0000 | [diff] [blame] | 8353 | case 'J': | 
| Mike Stump | 93246cc | 2009-07-28 23:57:15 +0000 | [diff] [blame] | 8354 | if (Signed) | 
| Mike Stump | a4de80b | 2009-07-28 02:25:19 +0000 | [diff] [blame] | 8355 | Type = Context.getsigjmp_bufType(); | 
| Mike Stump | 93246cc | 2009-07-28 23:57:15 +0000 | [diff] [blame] | 8356 | else | 
|  | 8357 | Type = Context.getjmp_bufType(); | 
|  | 8358 |  | 
| Mike Stump | 2adb4da | 2009-07-28 23:47:15 +0000 | [diff] [blame] | 8359 | if (Type.isNull()) { | 
| Mike Stump | 93246cc | 2009-07-28 23:57:15 +0000 | [diff] [blame] | 8360 | Error = ASTContext::GE_Missing_setjmp; | 
| Mike Stump | 2adb4da | 2009-07-28 23:47:15 +0000 | [diff] [blame] | 8361 | return QualType(); | 
|  | 8362 | } | 
|  | 8363 | break; | 
| Rafael Espindola | 6cfa82b | 2011-11-13 21:51:09 +0000 | [diff] [blame] | 8364 | case 'K': | 
|  | 8365 | assert(HowLong == 0 && !Signed && !Unsigned && "Bad modifiers for 'K'!"); | 
|  | 8366 | Type = Context.getucontext_tType(); | 
|  | 8367 |  | 
|  | 8368 | if (Type.isNull()) { | 
|  | 8369 | Error = ASTContext::GE_Missing_ucontext; | 
|  | 8370 | return QualType(); | 
|  | 8371 | } | 
|  | 8372 | break; | 
| Eli Friedman | 4e91899e | 2012-11-27 02:58:24 +0000 | [diff] [blame] | 8373 | case 'p': | 
|  | 8374 | Type = Context.getProcessIDType(); | 
|  | 8375 | break; | 
| Mike Stump | a4de80b | 2009-07-28 02:25:19 +0000 | [diff] [blame] | 8376 | } | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 8377 |  | 
| Chris Lattner | dc226c2 | 2010-10-01 22:42:38 +0000 | [diff] [blame] | 8378 | // If there are modifiers and if we're allowed to parse them, go for it. | 
|  | 8379 | Done = !AllowTypeModifiers; | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8380 | while (!Done) { | 
| John McCall | b8b9466 | 2010-03-12 04:21:28 +0000 | [diff] [blame] | 8381 | switch (char c = *Str++) { | 
| Chris Lattner | dc226c2 | 2010-10-01 22:42:38 +0000 | [diff] [blame] | 8382 | default: Done = true; --Str; break; | 
|  | 8383 | case '*': | 
|  | 8384 | case '&': { | 
|  | 8385 | // Both pointers and references can have their pointee types | 
|  | 8386 | // qualified with an address space. | 
|  | 8387 | char *End; | 
|  | 8388 | unsigned AddrSpace = strtoul(Str, &End, 10); | 
|  | 8389 | if (End != Str && AddrSpace != 0) { | 
|  | 8390 | Type = Context.getAddrSpaceQualType(Type, AddrSpace); | 
|  | 8391 | Str = End; | 
|  | 8392 | } | 
|  | 8393 | if (c == '*') | 
|  | 8394 | Type = Context.getPointerType(Type); | 
|  | 8395 | else | 
|  | 8396 | Type = Context.getLValueReferenceType(Type); | 
|  | 8397 | break; | 
|  | 8398 | } | 
|  | 8399 | // FIXME: There's no way to have a built-in with an rvalue ref arg. | 
|  | 8400 | case 'C': | 
|  | 8401 | Type = Type.withConst(); | 
|  | 8402 | break; | 
|  | 8403 | case 'D': | 
|  | 8404 | Type = Context.getVolatileType(Type); | 
|  | 8405 | break; | 
| Ted Kremenek | f2a2f5f | 2012-01-20 21:40:12 +0000 | [diff] [blame] | 8406 | case 'R': | 
|  | 8407 | Type = Type.withRestrict(); | 
|  | 8408 | break; | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8409 | } | 
|  | 8410 | } | 
| Chris Lattner | 8473339 | 2010-10-01 07:13:18 +0000 | [diff] [blame] | 8411 |  | 
| Chris Lattner | bd6e693 | 2010-10-01 22:53:11 +0000 | [diff] [blame] | 8412 | assert((!RequiresICE || Type->isIntegralOrEnumerationType()) && | 
| Chris Lattner | 8473339 | 2010-10-01 07:13:18 +0000 | [diff] [blame] | 8413 | "Integer constant 'I' type must be an integer"); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 8414 |  | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8415 | return Type; | 
|  | 8416 | } | 
|  | 8417 |  | 
|  | 8418 | /// GetBuiltinType - Return the type for the specified builtin. | 
| Chris Lattner | dc226c2 | 2010-10-01 22:42:38 +0000 | [diff] [blame] | 8419 | QualType ASTContext::GetBuiltinType(unsigned Id, | 
| Chris Lattner | bd6e693 | 2010-10-01 22:53:11 +0000 | [diff] [blame] | 8420 | GetBuiltinTypeError &Error, | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 8421 | unsigned *IntegerConstantArgs) const { | 
| Eric Christopher | 02d5d86 | 2015-08-06 01:01:12 +0000 | [diff] [blame] | 8422 | const char *TypeStr = BuiltinInfo.getTypeString(Id); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 8423 |  | 
| Chris Lattner | 0e62c1c | 2011-07-23 10:55:15 +0000 | [diff] [blame] | 8424 | SmallVector<QualType, 8> ArgTypes; | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 8425 |  | 
| Chris Lattner | bd6e693 | 2010-10-01 22:53:11 +0000 | [diff] [blame] | 8426 | bool RequiresICE = false; | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8427 | Error = GE_None; | 
| Chris Lattner | bd6e693 | 2010-10-01 22:53:11 +0000 | [diff] [blame] | 8428 | QualType ResType = DecodeTypeFromStr(TypeStr, *this, Error, | 
|  | 8429 | RequiresICE, true); | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8430 | if (Error != GE_None) | 
|  | 8431 | return QualType(); | 
| Chris Lattner | bd6e693 | 2010-10-01 22:53:11 +0000 | [diff] [blame] | 8432 |  | 
|  | 8433 | assert(!RequiresICE && "Result of intrinsic cannot be required to be an ICE"); | 
|  | 8434 |  | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8435 | while (TypeStr[0] && TypeStr[0] != '.') { | 
| Chris Lattner | bd6e693 | 2010-10-01 22:53:11 +0000 | [diff] [blame] | 8436 | QualType Ty = DecodeTypeFromStr(TypeStr, *this, Error, RequiresICE, true); | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8437 | if (Error != GE_None) | 
|  | 8438 | return QualType(); | 
|  | 8439 |  | 
| Chris Lattner | bd6e693 | 2010-10-01 22:53:11 +0000 | [diff] [blame] | 8440 | // If this argument is required to be an IntegerConstantExpression and the | 
|  | 8441 | // caller cares, fill in the bitmask we return. | 
|  | 8442 | if (RequiresICE && IntegerConstantArgs) | 
|  | 8443 | *IntegerConstantArgs |= 1 << ArgTypes.size(); | 
|  | 8444 |  | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8445 | // Do array -> pointer decay.  The builtin should use the decayed type. | 
|  | 8446 | if (Ty->isArrayType()) | 
|  | 8447 | Ty = getArrayDecayedType(Ty); | 
| Mike Stump | 11289f4 | 2009-09-09 15:08:12 +0000 | [diff] [blame] | 8448 |  | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8449 | ArgTypes.push_back(Ty); | 
|  | 8450 | } | 
|  | 8451 |  | 
| David Majnemer | ba3e5ec | 2015-03-13 18:26:17 +0000 | [diff] [blame] | 8452 | if (Id == Builtin::BI__GetExceptionInfo) | 
|  | 8453 | return QualType(); | 
|  | 8454 |  | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8455 | assert((TypeStr[0] != '.' || TypeStr[1] == 0) && | 
|  | 8456 | "'.' should only occur at end of builtin type list!"); | 
|  | 8457 |  | 
| Reid Kleckner | 78af070 | 2013-08-27 23:08:25 +0000 | [diff] [blame] | 8458 | FunctionType::ExtInfo EI(CC_C); | 
| John McCall | 991eb4b | 2010-12-21 00:44:39 +0000 | [diff] [blame] | 8459 | if (BuiltinInfo.isNoReturn(Id)) EI = EI.withNoReturn(true); | 
|  | 8460 |  | 
|  | 8461 | bool Variadic = (TypeStr[0] == '.'); | 
|  | 8462 |  | 
|  | 8463 | // We really shouldn't be making a no-proto type here, especially in C++. | 
|  | 8464 | if (ArgTypes.empty() && Variadic) | 
|  | 8465 | return getFunctionNoProtoType(ResType, EI); | 
| Douglas Gregor | 36c569f | 2010-02-21 22:15:06 +0000 | [diff] [blame] | 8466 |  | 
| John McCall | db40c7f | 2010-12-14 08:05:40 +0000 | [diff] [blame] | 8467 | FunctionProtoType::ExtProtoInfo EPI; | 
| John McCall | 991eb4b | 2010-12-21 00:44:39 +0000 | [diff] [blame] | 8468 | EPI.ExtInfo = EI; | 
|  | 8469 | EPI.Variadic = Variadic; | 
| John McCall | db40c7f | 2010-12-14 08:05:40 +0000 | [diff] [blame] | 8470 |  | 
| Jordan Rose | 5c38272 | 2013-03-08 21:51:21 +0000 | [diff] [blame] | 8471 | return getFunctionType(ResType, ArgTypes, EPI); | 
| Chris Lattner | ecd79c6 | 2009-06-14 00:45:47 +0000 | [diff] [blame] | 8472 | } | 
| Eli Friedman | 5ae98ee | 2009-08-19 07:44:53 +0000 | [diff] [blame] | 8473 |  | 
| Hans Wennborg | b0f2f14 | 2014-05-15 22:07:49 +0000 | [diff] [blame] | 8474 | static GVALinkage basicGVALinkageForFunction(const ASTContext &Context, | 
|  | 8475 | const FunctionDecl *FD) { | 
| Rafael Espindola | 3ae0005 | 2013-05-13 00:12:11 +0000 | [diff] [blame] | 8476 | if (!FD->isExternallyVisible()) | 
| Argyrios Kyrtzidis | c81af03 | 2010-07-29 18:15:58 +0000 | [diff] [blame] | 8477 | return GVA_Internal; | 
| Argyrios Kyrtzidis | c81af03 | 2010-07-29 18:15:58 +0000 | [diff] [blame] | 8478 |  | 
| Rafael Espindola | 3ae0005 | 2013-05-13 00:12:11 +0000 | [diff] [blame] | 8479 | GVALinkage External = GVA_StrongExternal; | 
|  | 8480 | switch (FD->getTemplateSpecializationKind()) { | 
|  | 8481 | case TSK_Undeclared: | 
|  | 8482 | case TSK_ExplicitSpecialization: | 
|  | 8483 | External = GVA_StrongExternal; | 
|  | 8484 | break; | 
| Argyrios Kyrtzidis | c81af03 | 2010-07-29 18:15:58 +0000 | [diff] [blame] | 8485 |  | 
| Rafael Espindola | 3ae0005 | 2013-05-13 00:12:11 +0000 | [diff] [blame] | 8486 | case TSK_ExplicitInstantiationDefinition: | 
| David Majnemer | 54e3ba5 | 2014-04-02 23:17:29 +0000 | [diff] [blame] | 8487 | return GVA_StrongODR; | 
| Rafael Espindola | 3ae0005 | 2013-05-13 00:12:11 +0000 | [diff] [blame] | 8488 |  | 
| David Majnemer | c3d0733 | 2014-05-15 06:25:57 +0000 | [diff] [blame] | 8489 | // C++11 [temp.explicit]p10: | 
|  | 8490 | //   [ Note: The intent is that an inline function that is the subject of | 
|  | 8491 | //   an explicit instantiation declaration will still be implicitly | 
|  | 8492 | //   instantiated when used so that the body can be considered for | 
|  | 8493 | //   inlining, but that no out-of-line copy of the inline function would be | 
|  | 8494 | //   generated in the translation unit. -- end note ] | 
| Rafael Espindola | 3ae0005 | 2013-05-13 00:12:11 +0000 | [diff] [blame] | 8495 | case TSK_ExplicitInstantiationDeclaration: | 
| David Majnemer | 27d69db | 2014-04-28 22:17:59 +0000 | [diff] [blame] | 8496 | return GVA_AvailableExternally; | 
|  | 8497 |  | 
| Rafael Espindola | 3ae0005 | 2013-05-13 00:12:11 +0000 | [diff] [blame] | 8498 | case TSK_ImplicitInstantiation: | 
| David Majnemer | 27d69db | 2014-04-28 22:17:59 +0000 | [diff] [blame] | 8499 | External = GVA_DiscardableODR; | 
| Rafael Espindola | 3ae0005 | 2013-05-13 00:12:11 +0000 | [diff] [blame] | 8500 | break; | 
| Argyrios Kyrtzidis | c81af03 | 2010-07-29 18:15:58 +0000 | [diff] [blame] | 8501 | } | 
|  | 8502 |  | 
|  | 8503 | if (!FD->isInlined()) | 
|  | 8504 | return External; | 
| David Majnemer | 62f0ffd | 2013-08-01 17:26:42 +0000 | [diff] [blame] | 8505 |  | 
| David Majnemer | 3f02150 | 2015-10-08 04:53:31 +0000 | [diff] [blame] | 8506 | if ((!Context.getLangOpts().CPlusPlus && | 
|  | 8507 | !Context.getTargetInfo().getCXXABI().isMicrosoft() && | 
| Hans Wennborg | b0f2f14 | 2014-05-15 22:07:49 +0000 | [diff] [blame] | 8508 | !FD->hasAttr<DLLExportAttr>()) || | 
| David Majnemer | 62f0ffd | 2013-08-01 17:26:42 +0000 | [diff] [blame] | 8509 | FD->hasAttr<GNUInlineAttr>()) { | 
| Hans Wennborg | b0f2f14 | 2014-05-15 22:07:49 +0000 | [diff] [blame] | 8510 | // FIXME: This doesn't match gcc's behavior for dllexport inline functions. | 
|  | 8511 |  | 
| Argyrios Kyrtzidis | c81af03 | 2010-07-29 18:15:58 +0000 | [diff] [blame] | 8512 | // GNU or C99 inline semantics. Determine whether this symbol should be | 
|  | 8513 | // externally visible. | 
|  | 8514 | if (FD->isInlineDefinitionExternallyVisible()) | 
|  | 8515 | return External; | 
|  | 8516 |  | 
|  | 8517 | // C99 inline semantics, where the symbol is not externally visible. | 
| David Majnemer | 27d69db | 2014-04-28 22:17:59 +0000 | [diff] [blame] | 8518 | return GVA_AvailableExternally; | 
| Argyrios Kyrtzidis | c81af03 | 2010-07-29 18:15:58 +0000 | [diff] [blame] | 8519 | } | 
|  | 8520 |  | 
| David Majnemer | 54e3ba5 | 2014-04-02 23:17:29 +0000 | [diff] [blame] | 8521 | // Functions specified with extern and inline in -fms-compatibility mode | 
|  | 8522 | // forcibly get emitted.  While the body of the function cannot be later | 
|  | 8523 | // replaced, the function definition cannot be discarded. | 
| David Majnemer | 7376870 | 2015-03-20 00:02:27 +0000 | [diff] [blame] | 8524 | if (FD->isMSExternInline()) | 
| David Majnemer | 54e3ba5 | 2014-04-02 23:17:29 +0000 | [diff] [blame] | 8525 | return GVA_StrongODR; | 
|  | 8526 |  | 
| David Majnemer | 27d69db | 2014-04-28 22:17:59 +0000 | [diff] [blame] | 8527 | return GVA_DiscardableODR; | 
| Argyrios Kyrtzidis | c81af03 | 2010-07-29 18:15:58 +0000 | [diff] [blame] | 8528 | } | 
|  | 8529 |  | 
| Artem Belevich | ca2b951 | 2016-05-02 20:30:03 +0000 | [diff] [blame] | 8530 | static GVALinkage adjustGVALinkageForAttributes(const ASTContext &Context, | 
|  | 8531 | GVALinkage L, const Decl *D) { | 
| Hans Wennborg | b0f2f14 | 2014-05-15 22:07:49 +0000 | [diff] [blame] | 8532 | // See http://msdn.microsoft.com/en-us/library/xa0d9ste.aspx | 
|  | 8533 | // dllexport/dllimport on inline functions. | 
|  | 8534 | if (D->hasAttr<DLLImportAttr>()) { | 
|  | 8535 | if (L == GVA_DiscardableODR || L == GVA_StrongODR) | 
|  | 8536 | return GVA_AvailableExternally; | 
| Artem Belevich | ca2b951 | 2016-05-02 20:30:03 +0000 | [diff] [blame] | 8537 | } else if (D->hasAttr<DLLExportAttr>()) { | 
| Hans Wennborg | b0f2f14 | 2014-05-15 22:07:49 +0000 | [diff] [blame] | 8538 | if (L == GVA_DiscardableODR) | 
|  | 8539 | return GVA_StrongODR; | 
| Artem Belevich | ca2b951 | 2016-05-02 20:30:03 +0000 | [diff] [blame] | 8540 | } else if (Context.getLangOpts().CUDA && Context.getLangOpts().CUDAIsDevice && | 
|  | 8541 | D->hasAttr<CUDAGlobalAttr>()) { | 
|  | 8542 | // Device-side functions with __global__ attribute must always be | 
|  | 8543 | // visible externally so they can be launched from host. | 
|  | 8544 | if (L == GVA_DiscardableODR || L == GVA_Internal) | 
|  | 8545 | return GVA_StrongODR; | 
| Hans Wennborg | b0f2f14 | 2014-05-15 22:07:49 +0000 | [diff] [blame] | 8546 | } | 
|  | 8547 | return L; | 
|  | 8548 | } | 
|  | 8549 |  | 
|  | 8550 | GVALinkage ASTContext::GetGVALinkageForFunction(const FunctionDecl *FD) const { | 
| Artem Belevich | ca2b951 | 2016-05-02 20:30:03 +0000 | [diff] [blame] | 8551 | return adjustGVALinkageForAttributes( | 
|  | 8552 | *this, basicGVALinkageForFunction(*this, FD), FD); | 
| Hans Wennborg | b0f2f14 | 2014-05-15 22:07:49 +0000 | [diff] [blame] | 8553 | } | 
|  | 8554 |  | 
|  | 8555 | static GVALinkage basicGVALinkageForVariable(const ASTContext &Context, | 
|  | 8556 | const VarDecl *VD) { | 
| Rafael Espindola | 3ae0005 | 2013-05-13 00:12:11 +0000 | [diff] [blame] | 8557 | if (!VD->isExternallyVisible()) | 
|  | 8558 | return GVA_Internal; | 
|  | 8559 |  | 
| David Majnemer | 27d69db | 2014-04-28 22:17:59 +0000 | [diff] [blame] | 8560 | if (VD->isStaticLocal()) { | 
|  | 8561 | GVALinkage StaticLocalLinkage = GVA_DiscardableODR; | 
|  | 8562 | const DeclContext *LexicalContext = VD->getParentFunctionOrMethod(); | 
|  | 8563 | while (LexicalContext && !isa<FunctionDecl>(LexicalContext)) | 
|  | 8564 | LexicalContext = LexicalContext->getLexicalParent(); | 
|  | 8565 |  | 
| Richard Smith | 9e2341d | 2015-03-23 03:25:59 +0000 | [diff] [blame] | 8566 | // Let the static local variable inherit its linkage from the nearest | 
| David Majnemer | 27d69db | 2014-04-28 22:17:59 +0000 | [diff] [blame] | 8567 | // enclosing function. | 
|  | 8568 | if (LexicalContext) | 
|  | 8569 | StaticLocalLinkage = | 
| Hans Wennborg | b0f2f14 | 2014-05-15 22:07:49 +0000 | [diff] [blame] | 8570 | Context.GetGVALinkageForFunction(cast<FunctionDecl>(LexicalContext)); | 
| David Majnemer | 27d69db | 2014-04-28 22:17:59 +0000 | [diff] [blame] | 8571 |  | 
|  | 8572 | // GVA_StrongODR function linkage is stronger than what we need, | 
|  | 8573 | // downgrade to GVA_DiscardableODR. | 
|  | 8574 | // This allows us to discard the variable if we never end up needing it. | 
|  | 8575 | return StaticLocalLinkage == GVA_StrongODR ? GVA_DiscardableODR | 
|  | 8576 | : StaticLocalLinkage; | 
|  | 8577 | } | 
|  | 8578 |  | 
| Hans Wennborg | 56fc62b | 2014-07-17 20:25:23 +0000 | [diff] [blame] | 8579 | // MSVC treats in-class initialized static data members as definitions. | 
|  | 8580 | // By giving them non-strong linkage, out-of-line definitions won't | 
|  | 8581 | // cause link errors. | 
|  | 8582 | if (Context.isMSStaticDataMemberInlineDefinition(VD)) | 
|  | 8583 | return GVA_DiscardableODR; | 
|  | 8584 |  | 
| Richard Smith | d9b9009 | 2016-07-02 01:32:16 +0000 | [diff] [blame] | 8585 | // Most non-template variables have strong linkage; inline variables are | 
|  | 8586 | // linkonce_odr or (occasionally, for compatibility) weak_odr. | 
|  | 8587 | GVALinkage StrongLinkage; | 
|  | 8588 | switch (Context.getInlineVariableDefinitionKind(VD)) { | 
|  | 8589 | case ASTContext::InlineVariableDefinitionKind::None: | 
|  | 8590 | StrongLinkage = GVA_StrongExternal; | 
|  | 8591 | break; | 
|  | 8592 | case ASTContext::InlineVariableDefinitionKind::Weak: | 
|  | 8593 | case ASTContext::InlineVariableDefinitionKind::WeakUnknown: | 
| Richard Smith | 62f19e7 | 2016-06-25 00:15:56 +0000 | [diff] [blame] | 8594 | StrongLinkage = GVA_DiscardableODR; | 
| Richard Smith | d9b9009 | 2016-07-02 01:32:16 +0000 | [diff] [blame] | 8595 | break; | 
|  | 8596 | case ASTContext::InlineVariableDefinitionKind::Strong: | 
|  | 8597 | StrongLinkage = GVA_StrongODR; | 
|  | 8598 | break; | 
|  | 8599 | } | 
| Richard Smith | 62f19e7 | 2016-06-25 00:15:56 +0000 | [diff] [blame] | 8600 |  | 
| Richard Smith | 8809a0c | 2013-09-27 20:14:12 +0000 | [diff] [blame] | 8601 | switch (VD->getTemplateSpecializationKind()) { | 
| Rafael Espindola | 3ae0005 | 2013-05-13 00:12:11 +0000 | [diff] [blame] | 8602 | case TSK_Undeclared: | 
| Richard Smith | 62f19e7 | 2016-06-25 00:15:56 +0000 | [diff] [blame] | 8603 | return StrongLinkage; | 
| Argyrios Kyrtzidis | c81af03 | 2010-07-29 18:15:58 +0000 | [diff] [blame] | 8604 |  | 
| David Majnemer | 6d1780c | 2015-07-17 23:36:49 +0000 | [diff] [blame] | 8605 | case TSK_ExplicitSpecialization: | 
| David Majnemer | 3f02150 | 2015-10-08 04:53:31 +0000 | [diff] [blame] | 8606 | return Context.getTargetInfo().getCXXABI().isMicrosoft() && | 
|  | 8607 | VD->isStaticDataMember() | 
| David Majnemer | 6d1780c | 2015-07-17 23:36:49 +0000 | [diff] [blame] | 8608 | ? GVA_StrongODR | 
| Richard Smith | 62f19e7 | 2016-06-25 00:15:56 +0000 | [diff] [blame] | 8609 | : StrongLinkage; | 
| David Majnemer | 6d1780c | 2015-07-17 23:36:49 +0000 | [diff] [blame] | 8610 |  | 
| Rafael Espindola | 3ae0005 | 2013-05-13 00:12:11 +0000 | [diff] [blame] | 8611 | case TSK_ExplicitInstantiationDefinition: | 
| David Majnemer | 54e3ba5 | 2014-04-02 23:17:29 +0000 | [diff] [blame] | 8612 | return GVA_StrongODR; | 
| Argyrios Kyrtzidis | c81af03 | 2010-07-29 18:15:58 +0000 | [diff] [blame] | 8613 |  | 
| David Majnemer | 27d69db | 2014-04-28 22:17:59 +0000 | [diff] [blame] | 8614 | case TSK_ExplicitInstantiationDeclaration: | 
|  | 8615 | return GVA_AvailableExternally; | 
|  | 8616 |  | 
| Rafael Espindola | 3ae0005 | 2013-05-13 00:12:11 +0000 | [diff] [blame] | 8617 | case TSK_ImplicitInstantiation: | 
| David Majnemer | 27d69db | 2014-04-28 22:17:59 +0000 | [diff] [blame] | 8618 | return GVA_DiscardableODR; | 
| Argyrios Kyrtzidis | c81af03 | 2010-07-29 18:15:58 +0000 | [diff] [blame] | 8619 | } | 
| Rafael Espindola | 27699c8 | 2013-05-13 14:05:53 +0000 | [diff] [blame] | 8620 |  | 
|  | 8621 | llvm_unreachable("Invalid Linkage!"); | 
| Argyrios Kyrtzidis | c81af03 | 2010-07-29 18:15:58 +0000 | [diff] [blame] | 8622 | } | 
|  | 8623 |  | 
| Hans Wennborg | b0f2f14 | 2014-05-15 22:07:49 +0000 | [diff] [blame] | 8624 | GVALinkage ASTContext::GetGVALinkageForVariable(const VarDecl *VD) { | 
| Artem Belevich | ca2b951 | 2016-05-02 20:30:03 +0000 | [diff] [blame] | 8625 | return adjustGVALinkageForAttributes( | 
|  | 8626 | *this, basicGVALinkageForVariable(*this, VD), VD); | 
| Hans Wennborg | b0f2f14 | 2014-05-15 22:07:49 +0000 | [diff] [blame] | 8627 | } | 
|  | 8628 |  | 
| Argyrios Kyrtzidis | c904933 | 2010-07-29 20:08:05 +0000 | [diff] [blame] | 8629 | bool ASTContext::DeclMustBeEmitted(const Decl *D) { | 
| Argyrios Kyrtzidis | c81af03 | 2010-07-29 18:15:58 +0000 | [diff] [blame] | 8630 | if (const VarDecl *VD = dyn_cast<VarDecl>(D)) { | 
|  | 8631 | if (!VD->isFileVarDecl()) | 
|  | 8632 | return false; | 
| Renato Golin | 9258aa5 | 2014-05-21 10:40:27 +0000 | [diff] [blame] | 8633 | // Global named register variables (GNU extension) are never emitted. | 
|  | 8634 | if (VD->getStorageClass() == SC_Register) | 
|  | 8635 | return false; | 
| Richard Smith | 7747ce2 | 2015-08-19 20:49:38 +0000 | [diff] [blame] | 8636 | if (VD->getDescribedVarTemplate() || | 
|  | 8637 | isa<VarTemplatePartialSpecializationDecl>(VD)) | 
|  | 8638 | return false; | 
| Richard Smith | 5205a8c | 2013-04-01 20:22:16 +0000 | [diff] [blame] | 8639 | } else if (const FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) { | 
|  | 8640 | // We never need to emit an uninstantiated function template. | 
|  | 8641 | if (FD->getTemplatedKind() == FunctionDecl::TK_FunctionTemplate) | 
|  | 8642 | return false; | 
| Nico Weber | 6622029 | 2016-03-02 17:28:48 +0000 | [diff] [blame] | 8643 | } else if (isa<PragmaCommentDecl>(D)) | 
|  | 8644 | return true; | 
| Dmitry Polukhin | 0b0da29 | 2016-04-06 11:38:59 +0000 | [diff] [blame] | 8645 | else if (isa<OMPThreadPrivateDecl>(D) || | 
|  | 8646 | D->hasAttr<OMPDeclareTargetDeclAttr>()) | 
|  | 8647 | return true; | 
| Nico Weber | cbbaeb1 | 2016-03-02 19:28:54 +0000 | [diff] [blame] | 8648 | else if (isa<PragmaDetectMismatchDecl>(D)) | 
|  | 8649 | return true; | 
| Nico Weber | 6622029 | 2016-03-02 17:28:48 +0000 | [diff] [blame] | 8650 | else if (isa<OMPThreadPrivateDecl>(D)) | 
| Alexey Bataev | c5b1d32 | 2016-03-04 09:22:22 +0000 | [diff] [blame] | 8651 | return !D->getDeclContext()->isDependentContext(); | 
|  | 8652 | else if (isa<OMPDeclareReductionDecl>(D)) | 
|  | 8653 | return !D->getDeclContext()->isDependentContext(); | 
| Richard Smith | dc1f042 | 2016-07-20 19:10:16 +0000 | [diff] [blame] | 8654 | else if (isa<ImportDecl>(D)) | 
|  | 8655 | return true; | 
| Alexey Bataev | 9772000 | 2014-11-11 04:05:39 +0000 | [diff] [blame] | 8656 | else | 
| Richard Smith | 5205a8c | 2013-04-01 20:22:16 +0000 | [diff] [blame] | 8657 | return false; | 
|  | 8658 |  | 
|  | 8659 | // If this is a member of a class template, we do not need to emit it. | 
|  | 8660 | if (D->getDeclContext()->isDependentContext()) | 
| Argyrios Kyrtzidis | c81af03 | 2010-07-29 18:15:58 +0000 | [diff] [blame] | 8661 | return false; | 
|  | 8662 |  | 
| Argyrios Kyrtzidis | 6e03a74 | 2010-07-29 20:07:52 +0000 | [diff] [blame] | 8663 | // Weak references don't produce any output by themselves. | 
|  | 8664 | if (D->hasAttr<WeakRefAttr>()) | 
|  | 8665 | return false; | 
|  | 8666 |  | 
| Argyrios Kyrtzidis | c81af03 | 2010-07-29 18:15:58 +0000 | [diff] [blame] | 8667 | // Aliases and used decls are required. | 
|  | 8668 | if (D->hasAttr<AliasAttr>() || D->hasAttr<UsedAttr>()) | 
|  | 8669 | return true; | 
|  | 8670 |  | 
|  | 8671 | if (const FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) { | 
|  | 8672 | // Forward declarations aren't required. | 
| Alexis Hunt | 4a8ea10 | 2011-05-06 20:44:56 +0000 | [diff] [blame] | 8673 | if (!FD->doesThisDeclarationHaveABody()) | 
| Nick Lewycky | 26da4dd | 2011-07-18 05:26:13 +0000 | [diff] [blame] | 8674 | return FD->doesDeclarationForceExternallyVisibleDefinition(); | 
| Argyrios Kyrtzidis | c81af03 | 2010-07-29 18:15:58 +0000 | [diff] [blame] | 8675 |  | 
|  | 8676 | // Constructors and destructors are required. | 
|  | 8677 | if (FD->hasAttr<ConstructorAttr>() || FD->hasAttr<DestructorAttr>()) | 
|  | 8678 | return true; | 
|  | 8679 |  | 
| John McCall | 6bd2a89 | 2013-01-25 22:31:03 +0000 | [diff] [blame] | 8680 | // The key function for a class is required.  This rule only comes | 
|  | 8681 | // into play when inline functions can be key functions, though. | 
|  | 8682 | if (getTargetInfo().getCXXABI().canKeyFunctionBeInline()) { | 
|  | 8683 | if (const CXXMethodDecl *MD = dyn_cast<CXXMethodDecl>(FD)) { | 
|  | 8684 | const CXXRecordDecl *RD = MD->getParent(); | 
|  | 8685 | if (MD->isOutOfLine() && RD->isDynamicClass()) { | 
|  | 8686 | const CXXMethodDecl *KeyFunc = getCurrentKeyFunction(RD); | 
|  | 8687 | if (KeyFunc && KeyFunc->getCanonicalDecl() == MD->getCanonicalDecl()) | 
|  | 8688 | return true; | 
|  | 8689 | } | 
| Argyrios Kyrtzidis | c81af03 | 2010-07-29 18:15:58 +0000 | [diff] [blame] | 8690 | } | 
|  | 8691 | } | 
|  | 8692 |  | 
|  | 8693 | GVALinkage Linkage = GetGVALinkageForFunction(FD); | 
|  | 8694 |  | 
|  | 8695 | // static, static inline, always_inline, and extern inline functions can | 
|  | 8696 | // always be deferred.  Normal inline functions can be deferred in C99/C++. | 
|  | 8697 | // Implicit template instantiations can also be deferred in C++. | 
| David Majnemer | 27d69db | 2014-04-28 22:17:59 +0000 | [diff] [blame] | 8698 | if (Linkage == GVA_Internal || Linkage == GVA_AvailableExternally || | 
|  | 8699 | Linkage == GVA_DiscardableODR) | 
| Argyrios Kyrtzidis | c81af03 | 2010-07-29 18:15:58 +0000 | [diff] [blame] | 8700 | return false; | 
|  | 8701 | return true; | 
|  | 8702 | } | 
| Douglas Gregor | 87d8124 | 2011-09-10 00:22:34 +0000 | [diff] [blame] | 8703 |  | 
| Argyrios Kyrtzidis | c81af03 | 2010-07-29 18:15:58 +0000 | [diff] [blame] | 8704 | const VarDecl *VD = cast<VarDecl>(D); | 
|  | 8705 | assert(VD->isFileVarDecl() && "Expected file scoped var"); | 
|  | 8706 |  | 
| Hans Wennborg | 56fc62b | 2014-07-17 20:25:23 +0000 | [diff] [blame] | 8707 | if (VD->isThisDeclarationADefinition() == VarDecl::DeclarationOnly && | 
|  | 8708 | !isMSStaticDataMemberInlineDefinition(VD)) | 
| Argyrios Kyrtzidis | 6e03a74 | 2010-07-29 20:07:52 +0000 | [diff] [blame] | 8709 | return false; | 
|  | 8710 |  | 
| Richard Smith | a0e5e54 | 2012-11-12 21:38:00 +0000 | [diff] [blame] | 8711 | // Variables that can be needed in other TUs are required. | 
| Argyrios Kyrtzidis | c81af03 | 2010-07-29 18:15:58 +0000 | [diff] [blame] | 8712 | GVALinkage L = GetGVALinkageForVariable(VD); | 
| David Majnemer | c3d0733 | 2014-05-15 06:25:57 +0000 | [diff] [blame] | 8713 | if (L != GVA_Internal && L != GVA_AvailableExternally && | 
|  | 8714 | L != GVA_DiscardableODR) | 
| Richard Smith | a0e5e54 | 2012-11-12 21:38:00 +0000 | [diff] [blame] | 8715 | return true; | 
| Argyrios Kyrtzidis | c81af03 | 2010-07-29 18:15:58 +0000 | [diff] [blame] | 8716 |  | 
| Richard Smith | a0e5e54 | 2012-11-12 21:38:00 +0000 | [diff] [blame] | 8717 | // Variables that have destruction with side-effects are required. | 
|  | 8718 | if (VD->getType().isDestructedType()) | 
|  | 8719 | return true; | 
|  | 8720 |  | 
|  | 8721 | // Variables that have initialization with side-effects are required. | 
| Richard Smith | 7747ce2 | 2015-08-19 20:49:38 +0000 | [diff] [blame] | 8722 | if (VD->getInit() && VD->getInit()->HasSideEffects(*this) && | 
|  | 8723 | !VD->evaluateValue()) | 
| Richard Smith | a0e5e54 | 2012-11-12 21:38:00 +0000 | [diff] [blame] | 8724 | return true; | 
|  | 8725 |  | 
| Richard Smith | da38363 | 2016-08-15 01:33:41 +0000 | [diff] [blame] | 8726 | // Likewise, variables with tuple-like bindings are required if their | 
|  | 8727 | // bindings have side-effects. | 
|  | 8728 | if (auto *DD = dyn_cast<DecompositionDecl>(VD)) | 
|  | 8729 | for (auto *BD : DD->bindings()) | 
|  | 8730 | if (auto *BindingVD = BD->getHoldingVar()) | 
|  | 8731 | if (DeclMustBeEmitted(BindingVD)) | 
|  | 8732 | return true; | 
|  | 8733 |  | 
| Richard Smith | a0e5e54 | 2012-11-12 21:38:00 +0000 | [diff] [blame] | 8734 | return false; | 
| Argyrios Kyrtzidis | c81af03 | 2010-07-29 18:15:58 +0000 | [diff] [blame] | 8735 | } | 
| Charles Davis | 53c59df | 2010-08-16 03:33:14 +0000 | [diff] [blame] | 8736 |  | 
| Reid Kleckner | 78af070 | 2013-08-27 23:08:25 +0000 | [diff] [blame] | 8737 | CallingConv ASTContext::getDefaultCallingConvention(bool IsVariadic, | 
|  | 8738 | bool IsCXXMethod) const { | 
| Charles Davis | 99202b3 | 2010-11-09 18:04:24 +0000 | [diff] [blame] | 8739 | // Pass through to the C++ ABI object | 
| Reid Kleckner | 78af070 | 2013-08-27 23:08:25 +0000 | [diff] [blame] | 8740 | if (IsCXXMethod) | 
|  | 8741 | return ABI->getDefaultMethodCallConv(IsVariadic); | 
| Timur Iskhodzhanov | c5098ad | 2012-07-12 09:50:54 +0000 | [diff] [blame] | 8742 |  | 
| Alexey Bataev | a754718 | 2016-05-18 09:06:38 +0000 | [diff] [blame] | 8743 | switch (LangOpts.getDefaultCallingConv()) { | 
|  | 8744 | case LangOptions::DCC_None: | 
|  | 8745 | break; | 
|  | 8746 | case LangOptions::DCC_CDecl: | 
|  | 8747 | return CC_C; | 
|  | 8748 | case LangOptions::DCC_FastCall: | 
|  | 8749 | if (getTargetInfo().hasFeature("sse2")) | 
|  | 8750 | return CC_X86FastCall; | 
|  | 8751 | break; | 
|  | 8752 | case LangOptions::DCC_StdCall: | 
|  | 8753 | if (!IsVariadic) | 
|  | 8754 | return CC_X86StdCall; | 
|  | 8755 | break; | 
|  | 8756 | case LangOptions::DCC_VectorCall: | 
|  | 8757 | // __vectorcall cannot be applied to variadic functions. | 
|  | 8758 | if (!IsVariadic) | 
|  | 8759 | return CC_X86VectorCall; | 
|  | 8760 | break; | 
|  | 8761 | } | 
| Alexander Kornienko | 21de0ae | 2015-01-20 11:20:41 +0000 | [diff] [blame] | 8762 | return Target->getDefaultCallingConv(TargetInfo::CCMT_Unknown); | 
| Charles Davis | 99202b3 | 2010-11-09 18:04:24 +0000 | [diff] [blame] | 8763 | } | 
|  | 8764 |  | 
| Jay Foad | 39c7980 | 2011-01-12 09:06:06 +0000 | [diff] [blame] | 8765 | bool ASTContext::isNearlyEmpty(const CXXRecordDecl *RD) const { | 
| Anders Carlsson | 60a6263 | 2010-11-25 01:51:53 +0000 | [diff] [blame] | 8766 | // Pass through to the C++ ABI object | 
|  | 8767 | return ABI->isNearlyEmpty(RD); | 
|  | 8768 | } | 
|  | 8769 |  | 
| Reid Kleckner | 96f8f93 | 2014-02-05 17:27:08 +0000 | [diff] [blame] | 8770 | VTableContextBase *ASTContext::getVTableContext() { | 
|  | 8771 | if (!VTContext.get()) { | 
|  | 8772 | if (Target->getCXXABI().isMicrosoft()) | 
|  | 8773 | VTContext.reset(new MicrosoftVTableContext(*this)); | 
|  | 8774 | else | 
|  | 8775 | VTContext.reset(new ItaniumVTableContext(*this)); | 
|  | 8776 | } | 
|  | 8777 | return VTContext.get(); | 
|  | 8778 | } | 
|  | 8779 |  | 
| Peter Collingbourne | 0ff0b37 | 2011-01-13 18:57:25 +0000 | [diff] [blame] | 8780 | MangleContext *ASTContext::createMangleContext() { | 
| John McCall | 359b885 | 2013-01-25 22:30:49 +0000 | [diff] [blame] | 8781 | switch (Target->getCXXABI().getKind()) { | 
| Tim Northover | 9bb857a | 2013-01-31 12:13:10 +0000 | [diff] [blame] | 8782 | case TargetCXXABI::GenericAArch64: | 
| John McCall | 359b885 | 2013-01-25 22:30:49 +0000 | [diff] [blame] | 8783 | case TargetCXXABI::GenericItanium: | 
|  | 8784 | case TargetCXXABI::GenericARM: | 
| Zoran Jovanovic | 26a1216 | 2015-02-18 15:21:35 +0000 | [diff] [blame] | 8785 | case TargetCXXABI::GenericMIPS: | 
| John McCall | 359b885 | 2013-01-25 22:30:49 +0000 | [diff] [blame] | 8786 | case TargetCXXABI::iOS: | 
| Tim Northover | a2ee433 | 2014-03-29 15:09:45 +0000 | [diff] [blame] | 8787 | case TargetCXXABI::iOS64: | 
| Dan Gohman | c285307 | 2015-09-03 22:51:53 +0000 | [diff] [blame] | 8788 | case TargetCXXABI::WebAssembly: | 
| Tim Northover | 756447a | 2015-10-30 16:30:36 +0000 | [diff] [blame] | 8789 | case TargetCXXABI::WatchOS: | 
| Timur Iskhodzhanov | 6745522 | 2013-10-03 06:26:13 +0000 | [diff] [blame] | 8790 | return ItaniumMangleContext::create(*this, getDiagnostics()); | 
| John McCall | 359b885 | 2013-01-25 22:30:49 +0000 | [diff] [blame] | 8791 | case TargetCXXABI::Microsoft: | 
| Timur Iskhodzhanov | 6745522 | 2013-10-03 06:26:13 +0000 | [diff] [blame] | 8792 | return MicrosoftMangleContext::create(*this, getDiagnostics()); | 
| Peter Collingbourne | 0ff0b37 | 2011-01-13 18:57:25 +0000 | [diff] [blame] | 8793 | } | 
| David Blaikie | 83d382b | 2011-09-23 05:06:16 +0000 | [diff] [blame] | 8794 | llvm_unreachable("Unsupported ABI"); | 
| Peter Collingbourne | 0ff0b37 | 2011-01-13 18:57:25 +0000 | [diff] [blame] | 8795 | } | 
|  | 8796 |  | 
| Angel Garcia Gomez | 637d1e6 | 2015-10-20 13:23:58 +0000 | [diff] [blame] | 8797 | CXXABI::~CXXABI() {} | 
| Ted Kremenek | f5df0ce | 2011-04-28 04:53:38 +0000 | [diff] [blame] | 8798 |  | 
|  | 8799 | size_t ASTContext::getSideTableAllocatedMemory() const { | 
| Larisse Voufo | 39a1e50 | 2013-08-06 01:03:05 +0000 | [diff] [blame] | 8800 | return ASTRecordLayouts.getMemorySize() + | 
|  | 8801 | llvm::capacity_in_bytes(ObjCLayouts) + | 
|  | 8802 | llvm::capacity_in_bytes(KeyFunctions) + | 
|  | 8803 | llvm::capacity_in_bytes(ObjCImpls) + | 
|  | 8804 | llvm::capacity_in_bytes(BlockVarCopyInits) + | 
|  | 8805 | llvm::capacity_in_bytes(DeclAttrs) + | 
|  | 8806 | llvm::capacity_in_bytes(TemplateOrInstantiation) + | 
|  | 8807 | llvm::capacity_in_bytes(InstantiatedFromUsingDecl) + | 
|  | 8808 | llvm::capacity_in_bytes(InstantiatedFromUsingShadowDecl) + | 
|  | 8809 | llvm::capacity_in_bytes(InstantiatedFromUnnamedFieldDecl) + | 
|  | 8810 | llvm::capacity_in_bytes(OverriddenMethods) + | 
|  | 8811 | llvm::capacity_in_bytes(Types) + | 
|  | 8812 | llvm::capacity_in_bytes(VariableArrayTypes) + | 
|  | 8813 | llvm::capacity_in_bytes(ClassScopeSpecializationPattern); | 
| Ted Kremenek | f5df0ce | 2011-04-28 04:53:38 +0000 | [diff] [blame] | 8814 | } | 
| Ted Kremenek | 540017e | 2011-10-06 05:00:56 +0000 | [diff] [blame] | 8815 |  | 
| Stepan Dyatkovskiy | 5a63792 | 2013-09-05 11:23:21 +0000 | [diff] [blame] | 8816 | /// getIntTypeForBitwidth - | 
|  | 8817 | /// sets integer QualTy according to specified details: | 
|  | 8818 | /// bitwidth, signed/unsigned. | 
|  | 8819 | /// Returns empty type if there is no appropriate target types. | 
|  | 8820 | QualType ASTContext::getIntTypeForBitwidth(unsigned DestWidth, | 
|  | 8821 | unsigned Signed) const { | 
|  | 8822 | TargetInfo::IntType Ty = getTargetInfo().getIntTypeByWidth(DestWidth, Signed); | 
|  | 8823 | CanQualType QualTy = getFromTargetType(Ty); | 
|  | 8824 | if (!QualTy && DestWidth == 128) | 
|  | 8825 | return Signed ? Int128Ty : UnsignedInt128Ty; | 
|  | 8826 | return QualTy; | 
|  | 8827 | } | 
|  | 8828 |  | 
|  | 8829 | /// getRealTypeForBitwidth - | 
|  | 8830 | /// sets floating point QualTy according to specified bitwidth. | 
|  | 8831 | /// Returns empty type if there is no appropriate target types. | 
|  | 8832 | QualType ASTContext::getRealTypeForBitwidth(unsigned DestWidth) const { | 
|  | 8833 | TargetInfo::RealType Ty = getTargetInfo().getRealTypeByWidth(DestWidth); | 
|  | 8834 | switch (Ty) { | 
|  | 8835 | case TargetInfo::Float: | 
|  | 8836 | return FloatTy; | 
|  | 8837 | case TargetInfo::Double: | 
|  | 8838 | return DoubleTy; | 
|  | 8839 | case TargetInfo::LongDouble: | 
|  | 8840 | return LongDoubleTy; | 
| Nemanja Ivanovic | bb1ea2d | 2016-05-09 08:52:33 +0000 | [diff] [blame] | 8841 | case TargetInfo::Float128: | 
|  | 8842 | return Float128Ty; | 
| Stepan Dyatkovskiy | 5a63792 | 2013-09-05 11:23:21 +0000 | [diff] [blame] | 8843 | case TargetInfo::NoFloat: | 
|  | 8844 | return QualType(); | 
|  | 8845 | } | 
|  | 8846 |  | 
|  | 8847 | llvm_unreachable("Unhandled TargetInfo::RealType value"); | 
|  | 8848 | } | 
|  | 8849 |  | 
| Eli Friedman | 3b7d46c | 2013-07-10 00:30:46 +0000 | [diff] [blame] | 8850 | void ASTContext::setManglingNumber(const NamedDecl *ND, unsigned Number) { | 
|  | 8851 | if (Number > 1) | 
|  | 8852 | MangleNumbers[ND] = Number; | 
| David Blaikie | 095deba | 2012-11-14 01:52:05 +0000 | [diff] [blame] | 8853 | } | 
|  | 8854 |  | 
| Eli Friedman | 3b7d46c | 2013-07-10 00:30:46 +0000 | [diff] [blame] | 8855 | unsigned ASTContext::getManglingNumber(const NamedDecl *ND) const { | 
| Richard Smith | e9b02d6 | 2016-03-21 22:33:02 +0000 | [diff] [blame] | 8856 | auto I = MangleNumbers.find(ND); | 
| Eli Friedman | 3b7d46c | 2013-07-10 00:30:46 +0000 | [diff] [blame] | 8857 | return I != MangleNumbers.end() ? I->second : 1; | 
| David Blaikie | 095deba | 2012-11-14 01:52:05 +0000 | [diff] [blame] | 8858 | } | 
|  | 8859 |  | 
| David Majnemer | 2206bf5 | 2014-03-05 08:57:59 +0000 | [diff] [blame] | 8860 | void ASTContext::setStaticLocalNumber(const VarDecl *VD, unsigned Number) { | 
|  | 8861 | if (Number > 1) | 
|  | 8862 | StaticLocalNumbers[VD] = Number; | 
|  | 8863 | } | 
|  | 8864 |  | 
|  | 8865 | unsigned ASTContext::getStaticLocalNumber(const VarDecl *VD) const { | 
| Richard Smith | e9b02d6 | 2016-03-21 22:33:02 +0000 | [diff] [blame] | 8866 | auto I = StaticLocalNumbers.find(VD); | 
| David Majnemer | 2206bf5 | 2014-03-05 08:57:59 +0000 | [diff] [blame] | 8867 | return I != StaticLocalNumbers.end() ? I->second : 1; | 
|  | 8868 | } | 
|  | 8869 |  | 
| Eli Friedman | 3b7d46c | 2013-07-10 00:30:46 +0000 | [diff] [blame] | 8870 | MangleNumberingContext & | 
|  | 8871 | ASTContext::getManglingNumberContext(const DeclContext *DC) { | 
| Reid Kleckner | d8110b6 | 2013-09-10 20:14:30 +0000 | [diff] [blame] | 8872 | assert(LangOpts.CPlusPlus);  // We don't need mangling numbers for plain C. | 
|  | 8873 | MangleNumberingContext *&MCtx = MangleNumberingContexts[DC]; | 
|  | 8874 | if (!MCtx) | 
|  | 8875 | MCtx = createMangleNumberingContext(); | 
|  | 8876 | return *MCtx; | 
|  | 8877 | } | 
|  | 8878 |  | 
|  | 8879 | MangleNumberingContext *ASTContext::createMangleNumberingContext() const { | 
|  | 8880 | return ABI->createMangleNumberingContext(); | 
| Douglas Gregor | 6379854 | 2012-02-20 19:44:39 +0000 | [diff] [blame] | 8881 | } | 
|  | 8882 |  | 
| David Majnemer | e7a818f | 2015-03-06 18:53:55 +0000 | [diff] [blame] | 8883 | const CXXConstructorDecl * | 
|  | 8884 | ASTContext::getCopyConstructorForExceptionObject(CXXRecordDecl *RD) { | 
|  | 8885 | return ABI->getCopyConstructorForExceptionObject( | 
|  | 8886 | cast<CXXRecordDecl>(RD->getFirstDecl())); | 
|  | 8887 | } | 
|  | 8888 |  | 
|  | 8889 | void ASTContext::addCopyConstructorForExceptionObject(CXXRecordDecl *RD, | 
|  | 8890 | CXXConstructorDecl *CD) { | 
|  | 8891 | return ABI->addCopyConstructorForExceptionObject( | 
|  | 8892 | cast<CXXRecordDecl>(RD->getFirstDecl()), | 
|  | 8893 | cast<CXXConstructorDecl>(CD->getFirstDecl())); | 
|  | 8894 | } | 
|  | 8895 |  | 
| David Majnemer | dfa6d20 | 2015-03-11 18:36:39 +0000 | [diff] [blame] | 8896 | void ASTContext::addDefaultArgExprForConstructor(const CXXConstructorDecl *CD, | 
|  | 8897 | unsigned ParmIdx, Expr *DAE) { | 
|  | 8898 | ABI->addDefaultArgExprForConstructor( | 
|  | 8899 | cast<CXXConstructorDecl>(CD->getFirstDecl()), ParmIdx, DAE); | 
|  | 8900 | } | 
|  | 8901 |  | 
|  | 8902 | Expr *ASTContext::getDefaultArgExprForConstructor(const CXXConstructorDecl *CD, | 
|  | 8903 | unsigned ParmIdx) { | 
|  | 8904 | return ABI->getDefaultArgExprForConstructor( | 
|  | 8905 | cast<CXXConstructorDecl>(CD->getFirstDecl()), ParmIdx); | 
|  | 8906 | } | 
|  | 8907 |  | 
| David Majnemer | 0035052 | 2015-08-31 18:48:39 +0000 | [diff] [blame] | 8908 | void ASTContext::addTypedefNameForUnnamedTagDecl(TagDecl *TD, | 
|  | 8909 | TypedefNameDecl *DD) { | 
|  | 8910 | return ABI->addTypedefNameForUnnamedTagDecl(TD, DD); | 
|  | 8911 | } | 
|  | 8912 |  | 
|  | 8913 | TypedefNameDecl * | 
|  | 8914 | ASTContext::getTypedefNameForUnnamedTagDecl(const TagDecl *TD) { | 
|  | 8915 | return ABI->getTypedefNameForUnnamedTagDecl(TD); | 
|  | 8916 | } | 
|  | 8917 |  | 
|  | 8918 | void ASTContext::addDeclaratorForUnnamedTagDecl(TagDecl *TD, | 
|  | 8919 | DeclaratorDecl *DD) { | 
|  | 8920 | return ABI->addDeclaratorForUnnamedTagDecl(TD, DD); | 
|  | 8921 | } | 
|  | 8922 |  | 
|  | 8923 | DeclaratorDecl *ASTContext::getDeclaratorForUnnamedTagDecl(const TagDecl *TD) { | 
|  | 8924 | return ABI->getDeclaratorForUnnamedTagDecl(TD); | 
|  | 8925 | } | 
|  | 8926 |  | 
| Ted Kremenek | 540017e | 2011-10-06 05:00:56 +0000 | [diff] [blame] | 8927 | void ASTContext::setParameterIndex(const ParmVarDecl *D, unsigned int index) { | 
|  | 8928 | ParamIndices[D] = index; | 
|  | 8929 | } | 
|  | 8930 |  | 
|  | 8931 | unsigned ASTContext::getParameterIndex(const ParmVarDecl *D) const { | 
|  | 8932 | ParameterIndexTable::const_iterator I = ParamIndices.find(D); | 
|  | 8933 | assert(I != ParamIndices.end() && | 
|  | 8934 | "ParmIndices lacks entry set by ParmVarDecl"); | 
|  | 8935 | return I->second; | 
|  | 8936 | } | 
| Fariborz Jahanian | 615de76 | 2013-05-28 17:37:39 +0000 | [diff] [blame] | 8937 |  | 
| Richard Smith | e6c0144 | 2013-06-05 00:46:14 +0000 | [diff] [blame] | 8938 | APValue * | 
|  | 8939 | ASTContext::getMaterializedTemporaryValue(const MaterializeTemporaryExpr *E, | 
|  | 8940 | bool MayCreate) { | 
|  | 8941 | assert(E && E->getStorageDuration() == SD_Static && | 
|  | 8942 | "don't need to cache the computed value for this temporary"); | 
| David Majnemer | 2dcef9e | 2015-08-13 23:50:15 +0000 | [diff] [blame] | 8943 | if (MayCreate) { | 
|  | 8944 | APValue *&MTVI = MaterializedTemporaryValues[E]; | 
|  | 8945 | if (!MTVI) | 
|  | 8946 | MTVI = new (*this) APValue; | 
|  | 8947 | return MTVI; | 
|  | 8948 | } | 
| Richard Smith | e6c0144 | 2013-06-05 00:46:14 +0000 | [diff] [blame] | 8949 |  | 
| David Majnemer | 2dcef9e | 2015-08-13 23:50:15 +0000 | [diff] [blame] | 8950 | return MaterializedTemporaryValues.lookup(E); | 
| Richard Smith | e6c0144 | 2013-06-05 00:46:14 +0000 | [diff] [blame] | 8951 | } | 
|  | 8952 |  | 
| Fariborz Jahanian | 615de76 | 2013-05-28 17:37:39 +0000 | [diff] [blame] | 8953 | bool ASTContext::AtomicUsesUnsupportedLibcall(const AtomicExpr *E) const { | 
|  | 8954 | const llvm::Triple &T = getTargetInfo().getTriple(); | 
|  | 8955 | if (!T.isOSDarwin()) | 
|  | 8956 | return false; | 
|  | 8957 |  | 
| Bob Wilson | 2c82c3d | 2013-11-02 23:27:49 +0000 | [diff] [blame] | 8958 | if (!(T.isiOS() && T.isOSVersionLT(7)) && | 
|  | 8959 | !(T.isMacOSX() && T.isOSVersionLT(10, 9))) | 
|  | 8960 | return false; | 
|  | 8961 |  | 
| Fariborz Jahanian | 615de76 | 2013-05-28 17:37:39 +0000 | [diff] [blame] | 8962 | QualType AtomicTy = E->getPtr()->getType()->getPointeeType(); | 
|  | 8963 | CharUnits sizeChars = getTypeSizeInChars(AtomicTy); | 
|  | 8964 | uint64_t Size = sizeChars.getQuantity(); | 
|  | 8965 | CharUnits alignChars = getTypeAlignInChars(AtomicTy); | 
|  | 8966 | unsigned Align = alignChars.getQuantity(); | 
|  | 8967 | unsigned MaxInlineWidthInBits = getTargetInfo().getMaxAtomicInlineWidth(); | 
|  | 8968 | return (Size != Align || toBits(sizeChars) > MaxInlineWidthInBits); | 
|  | 8969 | } | 
| Reid Kleckner | 2ab0ac5 | 2013-06-17 12:56:08 +0000 | [diff] [blame] | 8970 |  | 
|  | 8971 | namespace { | 
|  | 8972 |  | 
| Benjamin Kramer | 94355ae | 2015-10-23 09:04:55 +0000 | [diff] [blame] | 8973 | ast_type_traits::DynTypedNode getSingleDynTypedNodeFromParentMap( | 
|  | 8974 | ASTContext::ParentMapPointers::mapped_type U) { | 
| Benjamin Kramer | fbfa7a1 | 2015-10-22 11:26:35 +0000 | [diff] [blame] | 8975 | if (const auto *D = U.dyn_cast<const Decl *>()) | 
| Benjamin Kramer | 8e4a176 | 2015-10-22 11:21:40 +0000 | [diff] [blame] | 8976 | return ast_type_traits::DynTypedNode::create(*D); | 
| Benjamin Kramer | fbfa7a1 | 2015-10-22 11:26:35 +0000 | [diff] [blame] | 8977 | if (const auto *S = U.dyn_cast<const Stmt *>()) | 
| Benjamin Kramer | 8e4a176 | 2015-10-22 11:21:40 +0000 | [diff] [blame] | 8978 | return ast_type_traits::DynTypedNode::create(*S); | 
| Benjamin Kramer | fbfa7a1 | 2015-10-22 11:26:35 +0000 | [diff] [blame] | 8979 | return *U.get<ast_type_traits::DynTypedNode *>(); | 
| Benjamin Kramer | 8e4a176 | 2015-10-22 11:21:40 +0000 | [diff] [blame] | 8980 | } | 
|  | 8981 |  | 
| Benjamin Kramer | 94355ae | 2015-10-23 09:04:55 +0000 | [diff] [blame] | 8982 | /// Template specializations to abstract away from pointers and TypeLocs. | 
|  | 8983 | /// @{ | 
|  | 8984 | template <typename T> | 
|  | 8985 | ast_type_traits::DynTypedNode createDynTypedNode(const T &Node) { | 
|  | 8986 | return ast_type_traits::DynTypedNode::create(*Node); | 
|  | 8987 | } | 
|  | 8988 | template <> | 
|  | 8989 | ast_type_traits::DynTypedNode createDynTypedNode(const TypeLoc &Node) { | 
|  | 8990 | return ast_type_traits::DynTypedNode::create(Node); | 
|  | 8991 | } | 
|  | 8992 | template <> | 
|  | 8993 | ast_type_traits::DynTypedNode | 
|  | 8994 | createDynTypedNode(const NestedNameSpecifierLoc &Node) { | 
|  | 8995 | return ast_type_traits::DynTypedNode::create(Node); | 
|  | 8996 | } | 
|  | 8997 | /// @} | 
|  | 8998 |  | 
| Reid Kleckner | 2ab0ac5 | 2013-06-17 12:56:08 +0000 | [diff] [blame] | 8999 | /// \brief A \c RecursiveASTVisitor that builds a map from nodes to their | 
|  | 9000 | /// parents as defined by the \c RecursiveASTVisitor. | 
|  | 9001 | /// | 
|  | 9002 | /// Note that the relationship described here is purely in terms of AST | 
|  | 9003 | /// traversal - there are other relationships (for example declaration context) | 
|  | 9004 | /// in the AST that are better modeled by special matchers. | 
|  | 9005 | /// | 
| Benjamin Kramer | e8c51fd | 2015-10-21 10:07:26 +0000 | [diff] [blame] | 9006 | /// FIXME: Currently only builds up the map using \c Stmt and \c Decl nodes. | 
| Reid Kleckner | 2ab0ac5 | 2013-06-17 12:56:08 +0000 | [diff] [blame] | 9007 | class ParentMapASTVisitor : public RecursiveASTVisitor<ParentMapASTVisitor> { | 
| Reid Kleckner | 2ab0ac5 | 2013-06-17 12:56:08 +0000 | [diff] [blame] | 9008 | public: | 
|  | 9009 | /// \brief Builds and returns the translation unit's parent map. | 
|  | 9010 | /// | 
|  | 9011 | ///  The caller takes ownership of the returned \c ParentMap. | 
| Benjamin Kramer | 94355ae | 2015-10-23 09:04:55 +0000 | [diff] [blame] | 9012 | static std::pair<ASTContext::ParentMapPointers *, | 
|  | 9013 | ASTContext::ParentMapOtherNodes *> | 
|  | 9014 | buildMap(TranslationUnitDecl &TU) { | 
|  | 9015 | ParentMapASTVisitor Visitor(new ASTContext::ParentMapPointers, | 
|  | 9016 | new ASTContext::ParentMapOtherNodes); | 
| Reid Kleckner | 2ab0ac5 | 2013-06-17 12:56:08 +0000 | [diff] [blame] | 9017 | Visitor.TraverseDecl(&TU); | 
| Benjamin Kramer | 94355ae | 2015-10-23 09:04:55 +0000 | [diff] [blame] | 9018 | return std::make_pair(Visitor.Parents, Visitor.OtherParents); | 
| Reid Kleckner | 2ab0ac5 | 2013-06-17 12:56:08 +0000 | [diff] [blame] | 9019 | } | 
|  | 9020 |  | 
|  | 9021 | private: | 
|  | 9022 | typedef RecursiveASTVisitor<ParentMapASTVisitor> VisitorBase; | 
|  | 9023 |  | 
| Benjamin Kramer | 94355ae | 2015-10-23 09:04:55 +0000 | [diff] [blame] | 9024 | ParentMapASTVisitor(ASTContext::ParentMapPointers *Parents, | 
|  | 9025 | ASTContext::ParentMapOtherNodes *OtherParents) | 
|  | 9026 | : Parents(Parents), OtherParents(OtherParents) {} | 
| Reid Kleckner | 2ab0ac5 | 2013-06-17 12:56:08 +0000 | [diff] [blame] | 9027 |  | 
|  | 9028 | bool shouldVisitTemplateInstantiations() const { | 
|  | 9029 | return true; | 
|  | 9030 | } | 
|  | 9031 | bool shouldVisitImplicitCode() const { | 
|  | 9032 | return true; | 
|  | 9033 | } | 
| Reid Kleckner | 2ab0ac5 | 2013-06-17 12:56:08 +0000 | [diff] [blame] | 9034 |  | 
| Richard Smith | 8583872 | 2015-11-24 03:09:01 +0000 | [diff] [blame] | 9035 | template <typename T, typename MapNodeTy, typename BaseTraverseFn, | 
|  | 9036 | typename MapTy> | 
| Benjamin Kramer | 94355ae | 2015-10-23 09:04:55 +0000 | [diff] [blame] | 9037 | bool TraverseNode(T Node, MapNodeTy MapNode, | 
| Richard Smith | 8583872 | 2015-11-24 03:09:01 +0000 | [diff] [blame] | 9038 | BaseTraverseFn BaseTraverse, MapTy *Parents) { | 
| Craig Topper | 36250ad | 2014-05-12 05:36:57 +0000 | [diff] [blame] | 9039 | if (!Node) | 
| Reid Kleckner | 2ab0ac5 | 2013-06-17 12:56:08 +0000 | [diff] [blame] | 9040 | return true; | 
| Manuel Klimek | 95403e6 | 2014-05-21 13:28:59 +0000 | [diff] [blame] | 9041 | if (ParentStack.size() > 0) { | 
| Benjamin Kramer | e8c51fd | 2015-10-21 10:07:26 +0000 | [diff] [blame] | 9042 | // FIXME: Currently we add the same parent multiple times, but only | 
|  | 9043 | // when no memoization data is available for the type. | 
|  | 9044 | // For example when we visit all subexpressions of template | 
|  | 9045 | // instantiations; this is suboptimal, but benign: the only way to | 
|  | 9046 | // visit those is with hasAncestor / hasParent, and those do not create | 
|  | 9047 | // new matches. | 
|  | 9048 | // The plan is to enable DynTypedNode to be storable in a map or hash | 
|  | 9049 | // map. The main problem there is to implement hash functions / | 
|  | 9050 | // comparison operators for all types that DynTypedNode supports that | 
|  | 9051 | // do not have pointer identity. | 
| Benjamin Kramer | 94355ae | 2015-10-23 09:04:55 +0000 | [diff] [blame] | 9052 | auto &NodeOrVector = (*Parents)[MapNode]; | 
| Manuel Klimek | 95403e6 | 2014-05-21 13:28:59 +0000 | [diff] [blame] | 9053 | if (NodeOrVector.isNull()) { | 
| Benjamin Kramer | 8e4a176 | 2015-10-22 11:21:40 +0000 | [diff] [blame] | 9054 | if (const auto *D = ParentStack.back().get<Decl>()) | 
|  | 9055 | NodeOrVector = D; | 
|  | 9056 | else if (const auto *S = ParentStack.back().get<Stmt>()) | 
|  | 9057 | NodeOrVector = S; | 
|  | 9058 | else | 
|  | 9059 | NodeOrVector = | 
|  | 9060 | new ast_type_traits::DynTypedNode(ParentStack.back()); | 
| Manuel Klimek | 95403e6 | 2014-05-21 13:28:59 +0000 | [diff] [blame] | 9061 | } else { | 
| Benjamin Kramer | 8e4a176 | 2015-10-22 11:21:40 +0000 | [diff] [blame] | 9062 | if (!NodeOrVector.template is<ASTContext::ParentVector *>()) { | 
|  | 9063 | auto *Vector = new ASTContext::ParentVector( | 
|  | 9064 | 1, getSingleDynTypedNodeFromParentMap(NodeOrVector)); | 
| Benjamin Kramer | 8e4a176 | 2015-10-22 11:21:40 +0000 | [diff] [blame] | 9065 | if (auto *Node = | 
|  | 9066 | NodeOrVector | 
|  | 9067 | .template dyn_cast<ast_type_traits::DynTypedNode *>()) | 
|  | 9068 | delete Node; | 
| Benjamin Kramer | 422b3ff | 2015-10-23 13:24:18 +0000 | [diff] [blame] | 9069 | NodeOrVector = Vector; | 
| Samuel Benzaquen | 3ca0a7b | 2014-06-13 13:31:40 +0000 | [diff] [blame] | 9070 | } | 
| Samuel Benzaquen | 3ca0a7b | 2014-06-13 13:31:40 +0000 | [diff] [blame] | 9071 |  | 
|  | 9072 | auto *Vector = | 
|  | 9073 | NodeOrVector.template get<ASTContext::ParentVector *>(); | 
|  | 9074 | // Skip duplicates for types that have memoization data. | 
|  | 9075 | // We must check that the type has memoization data before calling | 
|  | 9076 | // std::find() because DynTypedNode::operator== can't compare all | 
|  | 9077 | // types. | 
|  | 9078 | bool Found = ParentStack.back().getMemoizationData() && | 
|  | 9079 | std::find(Vector->begin(), Vector->end(), | 
|  | 9080 | ParentStack.back()) != Vector->end(); | 
|  | 9081 | if (!Found) | 
|  | 9082 | Vector->push_back(ParentStack.back()); | 
| Manuel Klimek | 95403e6 | 2014-05-21 13:28:59 +0000 | [diff] [blame] | 9083 | } | 
|  | 9084 | } | 
| Benjamin Kramer | 94355ae | 2015-10-23 09:04:55 +0000 | [diff] [blame] | 9085 | ParentStack.push_back(createDynTypedNode(Node)); | 
| Richard Smith | 8583872 | 2015-11-24 03:09:01 +0000 | [diff] [blame] | 9086 | bool Result = BaseTraverse(); | 
| Reid Kleckner | 2ab0ac5 | 2013-06-17 12:56:08 +0000 | [diff] [blame] | 9087 | ParentStack.pop_back(); | 
|  | 9088 | return Result; | 
|  | 9089 | } | 
|  | 9090 |  | 
|  | 9091 | bool TraverseDecl(Decl *DeclNode) { | 
| Richard Smith | 8583872 | 2015-11-24 03:09:01 +0000 | [diff] [blame] | 9092 | return TraverseNode(DeclNode, DeclNode, | 
|  | 9093 | [&] { return VisitorBase::TraverseDecl(DeclNode); }, | 
| Benjamin Kramer | 94355ae | 2015-10-23 09:04:55 +0000 | [diff] [blame] | 9094 | Parents); | 
| Reid Kleckner | 2ab0ac5 | 2013-06-17 12:56:08 +0000 | [diff] [blame] | 9095 | } | 
|  | 9096 |  | 
|  | 9097 | bool TraverseStmt(Stmt *StmtNode) { | 
| Richard Smith | 8583872 | 2015-11-24 03:09:01 +0000 | [diff] [blame] | 9098 | return TraverseNode(StmtNode, StmtNode, | 
|  | 9099 | [&] { return VisitorBase::TraverseStmt(StmtNode); }, | 
| Benjamin Kramer | 94355ae | 2015-10-23 09:04:55 +0000 | [diff] [blame] | 9100 | Parents); | 
| Reid Kleckner | 2ab0ac5 | 2013-06-17 12:56:08 +0000 | [diff] [blame] | 9101 | } | 
|  | 9102 |  | 
| Benjamin Kramer | 94355ae | 2015-10-23 09:04:55 +0000 | [diff] [blame] | 9103 | bool TraverseTypeLoc(TypeLoc TypeLocNode) { | 
| Richard Smith | 8583872 | 2015-11-24 03:09:01 +0000 | [diff] [blame] | 9104 | return TraverseNode( | 
|  | 9105 | TypeLocNode, ast_type_traits::DynTypedNode::create(TypeLocNode), | 
|  | 9106 | [&] { return VisitorBase::TraverseTypeLoc(TypeLocNode); }, | 
|  | 9107 | OtherParents); | 
| Benjamin Kramer | 94355ae | 2015-10-23 09:04:55 +0000 | [diff] [blame] | 9108 | } | 
|  | 9109 |  | 
|  | 9110 | bool TraverseNestedNameSpecifierLoc(NestedNameSpecifierLoc NNSLocNode) { | 
|  | 9111 | return TraverseNode( | 
|  | 9112 | NNSLocNode, ast_type_traits::DynTypedNode::create(NNSLocNode), | 
| Richard Smith | 8583872 | 2015-11-24 03:09:01 +0000 | [diff] [blame] | 9113 | [&] { | 
|  | 9114 | return VisitorBase::TraverseNestedNameSpecifierLoc(NNSLocNode); | 
|  | 9115 | }, | 
|  | 9116 | OtherParents); | 
| Benjamin Kramer | 94355ae | 2015-10-23 09:04:55 +0000 | [diff] [blame] | 9117 | } | 
|  | 9118 |  | 
|  | 9119 | ASTContext::ParentMapPointers *Parents; | 
|  | 9120 | ASTContext::ParentMapOtherNodes *OtherParents; | 
| Reid Kleckner | 2ab0ac5 | 2013-06-17 12:56:08 +0000 | [diff] [blame] | 9121 | llvm::SmallVector<ast_type_traits::DynTypedNode, 16> ParentStack; | 
|  | 9122 |  | 
|  | 9123 | friend class RecursiveASTVisitor<ParentMapASTVisitor>; | 
|  | 9124 | }; | 
|  | 9125 |  | 
| Eugene Zelenko | d4304d2 | 2015-11-04 21:37:17 +0000 | [diff] [blame] | 9126 | } // anonymous namespace | 
| Reid Kleckner | 2ab0ac5 | 2013-06-17 12:56:08 +0000 | [diff] [blame] | 9127 |  | 
| Benjamin Kramer | 94355ae | 2015-10-23 09:04:55 +0000 | [diff] [blame] | 9128 | template <typename NodeTy, typename MapTy> | 
|  | 9129 | static ASTContext::DynTypedNodeList getDynNodeFromMap(const NodeTy &Node, | 
|  | 9130 | const MapTy &Map) { | 
|  | 9131 | auto I = Map.find(Node); | 
|  | 9132 | if (I == Map.end()) { | 
| Benjamin Kramer | 8e4a176 | 2015-10-22 11:21:40 +0000 | [diff] [blame] | 9133 | return llvm::ArrayRef<ast_type_traits::DynTypedNode>(); | 
| Reid Kleckner | 2ab0ac5 | 2013-06-17 12:56:08 +0000 | [diff] [blame] | 9134 | } | 
| Benjamin Kramer | 94355ae | 2015-10-23 09:04:55 +0000 | [diff] [blame] | 9135 | if (auto *V = I->second.template dyn_cast<ASTContext::ParentVector *>()) { | 
| Benjamin Kramer | 8e4a176 | 2015-10-22 11:21:40 +0000 | [diff] [blame] | 9136 | return llvm::makeArrayRef(*V); | 
| Manuel Klimek | 95403e6 | 2014-05-21 13:28:59 +0000 | [diff] [blame] | 9137 | } | 
| Benjamin Kramer | 8e4a176 | 2015-10-22 11:21:40 +0000 | [diff] [blame] | 9138 | return getSingleDynTypedNodeFromParentMap(I->second); | 
| Reid Kleckner | 2ab0ac5 | 2013-06-17 12:56:08 +0000 | [diff] [blame] | 9139 | } | 
| Fariborz Jahanian | d36150d | 2013-07-15 21:22:08 +0000 | [diff] [blame] | 9140 |  | 
| Benjamin Kramer | 94355ae | 2015-10-23 09:04:55 +0000 | [diff] [blame] | 9141 | ASTContext::DynTypedNodeList | 
|  | 9142 | ASTContext::getParents(const ast_type_traits::DynTypedNode &Node) { | 
|  | 9143 | if (!PointerParents) { | 
|  | 9144 | // We always need to run over the whole translation unit, as | 
|  | 9145 | // hasAncestor can escape any subtree. | 
|  | 9146 | auto Maps = ParentMapASTVisitor::buildMap(*getTranslationUnitDecl()); | 
|  | 9147 | PointerParents.reset(Maps.first); | 
|  | 9148 | OtherParents.reset(Maps.second); | 
|  | 9149 | } | 
|  | 9150 | if (Node.getNodeKind().hasPointerIdentity()) | 
|  | 9151 | return getDynNodeFromMap(Node.getMemoizationData(), *PointerParents); | 
|  | 9152 | return getDynNodeFromMap(Node, *OtherParents); | 
|  | 9153 | } | 
|  | 9154 |  | 
| Fariborz Jahanian | d36150d | 2013-07-15 21:22:08 +0000 | [diff] [blame] | 9155 | bool | 
|  | 9156 | ASTContext::ObjCMethodsAreEqual(const ObjCMethodDecl *MethodDecl, | 
|  | 9157 | const ObjCMethodDecl *MethodImpl) { | 
|  | 9158 | // No point trying to match an unavailable/deprecated mothod. | 
|  | 9159 | if (MethodDecl->hasAttr<UnavailableAttr>() | 
|  | 9160 | || MethodDecl->hasAttr<DeprecatedAttr>()) | 
|  | 9161 | return false; | 
|  | 9162 | if (MethodDecl->getObjCDeclQualifier() != | 
|  | 9163 | MethodImpl->getObjCDeclQualifier()) | 
|  | 9164 | return false; | 
| Alp Toker | 314cc81 | 2014-01-25 16:55:45 +0000 | [diff] [blame] | 9165 | if (!hasSameType(MethodDecl->getReturnType(), MethodImpl->getReturnType())) | 
| Fariborz Jahanian | d36150d | 2013-07-15 21:22:08 +0000 | [diff] [blame] | 9166 | return false; | 
|  | 9167 |  | 
|  | 9168 | if (MethodDecl->param_size() != MethodImpl->param_size()) | 
|  | 9169 | return false; | 
|  | 9170 |  | 
|  | 9171 | for (ObjCMethodDecl::param_const_iterator IM = MethodImpl->param_begin(), | 
|  | 9172 | IF = MethodDecl->param_begin(), EM = MethodImpl->param_end(), | 
|  | 9173 | EF = MethodDecl->param_end(); | 
|  | 9174 | IM != EM && IF != EF; ++IM, ++IF) { | 
|  | 9175 | const ParmVarDecl *DeclVar = (*IF); | 
|  | 9176 | const ParmVarDecl *ImplVar = (*IM); | 
|  | 9177 | if (ImplVar->getObjCDeclQualifier() != DeclVar->getObjCDeclQualifier()) | 
|  | 9178 | return false; | 
|  | 9179 | if (!hasSameType(DeclVar->getType(), ImplVar->getType())) | 
|  | 9180 | return false; | 
|  | 9181 | } | 
|  | 9182 | return (MethodDecl->isVariadic() == MethodImpl->isVariadic()); | 
|  | 9183 |  | 
|  | 9184 | } | 
| Richard Smith | 053f6c6 | 2014-05-16 23:01:30 +0000 | [diff] [blame] | 9185 |  | 
|  | 9186 | // Explicitly instantiate this in case a Redeclarable<T> is used from a TU that | 
|  | 9187 | // doesn't include ASTContext.h | 
|  | 9188 | template | 
|  | 9189 | clang::LazyGenerationalUpdatePtr< | 
|  | 9190 | const Decl *, Decl *, &ExternalASTSource::CompleteRedeclChain>::ValueType | 
|  | 9191 | clang::LazyGenerationalUpdatePtr< | 
|  | 9192 | const Decl *, Decl *, &ExternalASTSource::CompleteRedeclChain>::makeValue( | 
|  | 9193 | const clang::ASTContext &Ctx, Decl *Value); |