File: | clang/lib/AST/DeclTemplate.cpp |
Warning: | line 83, column 9 Called C++ object pointer is null |
Press '?' to see keyboard shortcuts
Keyboard shortcuts:
1 | //===- DeclTemplate.cpp - Template Declaration AST Node Implementation ----===// | |||
2 | // | |||
3 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. | |||
4 | // See https://llvm.org/LICENSE.txt for license information. | |||
5 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception | |||
6 | // | |||
7 | //===----------------------------------------------------------------------===// | |||
8 | // | |||
9 | // This file implements the C++ related Decl classes for templates. | |||
10 | // | |||
11 | //===----------------------------------------------------------------------===// | |||
12 | ||||
13 | #include "clang/AST/DeclTemplate.h" | |||
14 | #include "clang/AST/ASTContext.h" | |||
15 | #include "clang/AST/ASTMutationListener.h" | |||
16 | #include "clang/AST/DeclCXX.h" | |||
17 | #include "clang/AST/DeclarationName.h" | |||
18 | #include "clang/AST/Expr.h" | |||
19 | #include "clang/AST/ExternalASTSource.h" | |||
20 | #include "clang/AST/TemplateBase.h" | |||
21 | #include "clang/AST/TemplateName.h" | |||
22 | #include "clang/AST/Type.h" | |||
23 | #include "clang/AST/TypeLoc.h" | |||
24 | #include "clang/Basic/Builtins.h" | |||
25 | #include "clang/Basic/LLVM.h" | |||
26 | #include "clang/Basic/SourceLocation.h" | |||
27 | #include "llvm/ADT/ArrayRef.h" | |||
28 | #include "llvm/ADT/FoldingSet.h" | |||
29 | #include "llvm/ADT/None.h" | |||
30 | #include "llvm/ADT/PointerUnion.h" | |||
31 | #include "llvm/ADT/SmallVector.h" | |||
32 | #include "llvm/Support/Casting.h" | |||
33 | #include "llvm/Support/ErrorHandling.h" | |||
34 | #include <algorithm> | |||
35 | #include <cassert> | |||
36 | #include <cstdint> | |||
37 | #include <memory> | |||
38 | #include <utility> | |||
39 | ||||
40 | using namespace clang; | |||
41 | ||||
42 | //===----------------------------------------------------------------------===// | |||
43 | // TemplateParameterList Implementation | |||
44 | //===----------------------------------------------------------------------===// | |||
45 | ||||
46 | ||||
47 | TemplateParameterList::TemplateParameterList(const ASTContext& C, | |||
48 | SourceLocation TemplateLoc, | |||
49 | SourceLocation LAngleLoc, | |||
50 | ArrayRef<NamedDecl *> Params, | |||
51 | SourceLocation RAngleLoc, | |||
52 | Expr *RequiresClause) | |||
53 | : TemplateLoc(TemplateLoc), LAngleLoc(LAngleLoc), RAngleLoc(RAngleLoc), | |||
54 | NumParams(Params.size()), ContainsUnexpandedParameterPack(false), | |||
55 | HasRequiresClause(RequiresClause != nullptr), | |||
56 | HasConstrainedParameters(false) { | |||
57 | for (unsigned Idx = 0; Idx < NumParams; ++Idx) { | |||
58 | NamedDecl *P = Params[Idx]; | |||
59 | begin()[Idx] = P; | |||
60 | ||||
61 | bool IsPack = P->isTemplateParameterPack(); | |||
62 | if (const auto *NTTP
| |||
63 | if (!IsPack && NTTP->getType()->containsUnexpandedParameterPack()) | |||
64 | ContainsUnexpandedParameterPack = true; | |||
65 | if (NTTP->hasPlaceholderTypeConstraint()) | |||
66 | HasConstrainedParameters = true; | |||
67 | } else if (const auto *TTP
| |||
68 | if (!IsPack && | |||
69 | TTP->getTemplateParameters()->containsUnexpandedParameterPack()) | |||
70 | ContainsUnexpandedParameterPack = true; | |||
71 | } else if (const TypeConstraint *TC = | |||
72 | cast<TemplateTypeParmDecl>(P)->getTypeConstraint()) { | |||
73 | if (TC->getImmediatelyDeclaredConstraint() | |||
74 | ->containsUnexpandedParameterPack()) | |||
75 | ContainsUnexpandedParameterPack = true; | |||
76 | HasConstrainedParameters = true; | |||
77 | } | |||
78 | // FIXME: If a default argument contains an unexpanded parameter pack, the | |||
79 | // template parameter list does too. | |||
80 | } | |||
81 | ||||
82 | if (HasRequiresClause) { | |||
83 | if (RequiresClause->containsUnexpandedParameterPack()) | |||
| ||||
84 | ContainsUnexpandedParameterPack = true; | |||
85 | *getTrailingObjects<Expr *>() = RequiresClause; | |||
86 | } | |||
87 | } | |||
88 | ||||
89 | TemplateParameterList * | |||
90 | TemplateParameterList::Create(const ASTContext &C, SourceLocation TemplateLoc, | |||
91 | SourceLocation LAngleLoc, | |||
92 | ArrayRef<NamedDecl *> Params, | |||
93 | SourceLocation RAngleLoc, Expr *RequiresClause) { | |||
94 | void *Mem = C.Allocate(totalSizeToAlloc<NamedDecl *, Expr *>( | |||
95 | Params.size(), RequiresClause
| |||
96 | alignof(TemplateParameterList)); | |||
97 | return new (Mem) TemplateParameterList(C, TemplateLoc, LAngleLoc, Params, | |||
98 | RAngleLoc, RequiresClause); | |||
99 | } | |||
100 | ||||
101 | unsigned TemplateParameterList::getMinRequiredArguments() const { | |||
102 | unsigned NumRequiredArgs = 0; | |||
103 | for (const NamedDecl *P : asArray()) { | |||
104 | if (P->isTemplateParameterPack()) { | |||
105 | if (Optional<unsigned> Expansions = getExpandedPackSize(P)) { | |||
106 | NumRequiredArgs += *Expansions; | |||
107 | continue; | |||
108 | } | |||
109 | break; | |||
110 | } | |||
111 | ||||
112 | if (const auto *TTP = dyn_cast<TemplateTypeParmDecl>(P)) { | |||
113 | if (TTP->hasDefaultArgument()) | |||
114 | break; | |||
115 | } else if (const auto *NTTP = dyn_cast<NonTypeTemplateParmDecl>(P)) { | |||
116 | if (NTTP->hasDefaultArgument()) | |||
117 | break; | |||
118 | } else if (cast<TemplateTemplateParmDecl>(P)->hasDefaultArgument()) | |||
119 | break; | |||
120 | ||||
121 | ++NumRequiredArgs; | |||
122 | } | |||
123 | ||||
124 | return NumRequiredArgs; | |||
125 | } | |||
126 | ||||
127 | unsigned TemplateParameterList::getDepth() const { | |||
128 | if (size() == 0) | |||
129 | return 0; | |||
130 | ||||
131 | const NamedDecl *FirstParm = getParam(0); | |||
132 | if (const auto *TTP = dyn_cast<TemplateTypeParmDecl>(FirstParm)) | |||
133 | return TTP->getDepth(); | |||
134 | else if (const auto *NTTP = dyn_cast<NonTypeTemplateParmDecl>(FirstParm)) | |||
135 | return NTTP->getDepth(); | |||
136 | else | |||
137 | return cast<TemplateTemplateParmDecl>(FirstParm)->getDepth(); | |||
138 | } | |||
139 | ||||
140 | static void AdoptTemplateParameterList(TemplateParameterList *Params, | |||
141 | DeclContext *Owner) { | |||
142 | for (NamedDecl *P : *Params) { | |||
143 | P->setDeclContext(Owner); | |||
144 | ||||
145 | if (const auto *TTP = dyn_cast<TemplateTemplateParmDecl>(P)) | |||
146 | AdoptTemplateParameterList(TTP->getTemplateParameters(), Owner); | |||
147 | } | |||
148 | } | |||
149 | ||||
150 | void TemplateParameterList:: | |||
151 | getAssociatedConstraints(llvm::SmallVectorImpl<const Expr *> &AC) const { | |||
152 | if (HasConstrainedParameters) | |||
153 | for (const NamedDecl *Param : *this) { | |||
154 | if (const auto *TTP = dyn_cast<TemplateTypeParmDecl>(Param)) { | |||
155 | if (const auto *TC = TTP->getTypeConstraint()) | |||
156 | AC.push_back(TC->getImmediatelyDeclaredConstraint()); | |||
157 | } else if (const auto *NTTP = dyn_cast<NonTypeTemplateParmDecl>(Param)) { | |||
158 | if (const Expr *E = NTTP->getPlaceholderTypeConstraint()) | |||
159 | AC.push_back(E); | |||
160 | } | |||
161 | } | |||
162 | if (HasRequiresClause) | |||
163 | AC.push_back(getRequiresClause()); | |||
164 | } | |||
165 | ||||
166 | bool TemplateParameterList::hasAssociatedConstraints() const { | |||
167 | return HasRequiresClause || HasConstrainedParameters; | |||
168 | } | |||
169 | ||||
170 | namespace clang { | |||
171 | ||||
172 | void *allocateDefaultArgStorageChain(const ASTContext &C) { | |||
173 | return new (C) char[sizeof(void*) * 2]; | |||
174 | } | |||
175 | ||||
176 | } // namespace clang | |||
177 | ||||
178 | //===----------------------------------------------------------------------===// | |||
179 | // TemplateDecl Implementation | |||
180 | //===----------------------------------------------------------------------===// | |||
181 | ||||
182 | TemplateDecl::TemplateDecl(Kind DK, DeclContext *DC, SourceLocation L, | |||
183 | DeclarationName Name, TemplateParameterList *Params, | |||
184 | NamedDecl *Decl) | |||
185 | : NamedDecl(DK, DC, L, Name), TemplatedDecl(Decl), TemplateParams(Params) {} | |||
186 | ||||
187 | void TemplateDecl::anchor() {} | |||
188 | ||||
189 | void TemplateDecl:: | |||
190 | getAssociatedConstraints(llvm::SmallVectorImpl<const Expr *> &AC) const { | |||
191 | TemplateParams->getAssociatedConstraints(AC); | |||
192 | if (auto *FD = dyn_cast_or_null<FunctionDecl>(getTemplatedDecl())) | |||
193 | if (const Expr *TRC = FD->getTrailingRequiresClause()) | |||
194 | AC.push_back(TRC); | |||
195 | } | |||
196 | ||||
197 | bool TemplateDecl::hasAssociatedConstraints() const { | |||
198 | if (TemplateParams->hasAssociatedConstraints()) | |||
199 | return true; | |||
200 | if (auto *FD = dyn_cast_or_null<FunctionDecl>(getTemplatedDecl())) | |||
201 | return FD->getTrailingRequiresClause(); | |||
202 | return false; | |||
203 | } | |||
204 | ||||
205 | //===----------------------------------------------------------------------===// | |||
206 | // RedeclarableTemplateDecl Implementation | |||
207 | //===----------------------------------------------------------------------===// | |||
208 | ||||
209 | void RedeclarableTemplateDecl::anchor() {} | |||
210 | ||||
211 | RedeclarableTemplateDecl::CommonBase *RedeclarableTemplateDecl::getCommonPtr() const { | |||
212 | if (Common) | |||
213 | return Common; | |||
214 | ||||
215 | // Walk the previous-declaration chain until we either find a declaration | |||
216 | // with a common pointer or we run out of previous declarations. | |||
217 | SmallVector<const RedeclarableTemplateDecl *, 2> PrevDecls; | |||
218 | for (const RedeclarableTemplateDecl *Prev = getPreviousDecl(); Prev; | |||
219 | Prev = Prev->getPreviousDecl()) { | |||
220 | if (Prev->Common) { | |||
221 | Common = Prev->Common; | |||
222 | break; | |||
223 | } | |||
224 | ||||
225 | PrevDecls.push_back(Prev); | |||
226 | } | |||
227 | ||||
228 | // If we never found a common pointer, allocate one now. | |||
229 | if (!Common) { | |||
230 | // FIXME: If any of the declarations is from an AST file, we probably | |||
231 | // need an update record to add the common data. | |||
232 | ||||
233 | Common = newCommon(getASTContext()); | |||
234 | } | |||
235 | ||||
236 | // Update any previous declarations we saw with the common pointer. | |||
237 | for (const RedeclarableTemplateDecl *Prev : PrevDecls) | |||
238 | Prev->Common = Common; | |||
239 | ||||
240 | return Common; | |||
241 | } | |||
242 | ||||
243 | void RedeclarableTemplateDecl::loadLazySpecializationsImpl() const { | |||
244 | // Grab the most recent declaration to ensure we've loaded any lazy | |||
245 | // redeclarations of this template. | |||
246 | CommonBase *CommonBasePtr = getMostRecentDecl()->getCommonPtr(); | |||
247 | if (CommonBasePtr->LazySpecializations) { | |||
248 | ASTContext &Context = getASTContext(); | |||
249 | uint32_t *Specs = CommonBasePtr->LazySpecializations; | |||
250 | CommonBasePtr->LazySpecializations = nullptr; | |||
251 | for (uint32_t I = 0, N = *Specs++; I != N; ++I) | |||
252 | (void)Context.getExternalSource()->GetExternalDecl(Specs[I]); | |||
253 | } | |||
254 | } | |||
255 | ||||
256 | template<class EntryType, typename... ProfileArguments> | |||
257 | typename RedeclarableTemplateDecl::SpecEntryTraits<EntryType>::DeclType * | |||
258 | RedeclarableTemplateDecl::findSpecializationImpl( | |||
259 | llvm::FoldingSetVector<EntryType> &Specs, void *&InsertPos, | |||
260 | ProfileArguments&&... ProfileArgs) { | |||
261 | using SETraits = SpecEntryTraits<EntryType>; | |||
262 | ||||
263 | llvm::FoldingSetNodeID ID; | |||
264 | EntryType::Profile(ID, std::forward<ProfileArguments>(ProfileArgs)..., | |||
265 | getASTContext()); | |||
266 | EntryType *Entry = Specs.FindNodeOrInsertPos(ID, InsertPos); | |||
267 | return Entry ? SETraits::getDecl(Entry)->getMostRecentDecl() : nullptr; | |||
268 | } | |||
269 | ||||
270 | template<class Derived, class EntryType> | |||
271 | void RedeclarableTemplateDecl::addSpecializationImpl( | |||
272 | llvm::FoldingSetVector<EntryType> &Specializations, EntryType *Entry, | |||
273 | void *InsertPos) { | |||
274 | using SETraits = SpecEntryTraits<EntryType>; | |||
275 | ||||
276 | if (InsertPos) { | |||
277 | #ifndef NDEBUG | |||
278 | void *CorrectInsertPos; | |||
279 | assert(!findSpecializationImpl(Specializations,(static_cast <bool> (!findSpecializationImpl(Specializations , CorrectInsertPos, SETraits::getTemplateArgs(Entry)) && InsertPos == CorrectInsertPos && "given incorrect InsertPos for specialization" ) ? void (0) : __assert_fail ("!findSpecializationImpl(Specializations, CorrectInsertPos, SETraits::getTemplateArgs(Entry)) && InsertPos == CorrectInsertPos && \"given incorrect InsertPos for specialization\"" , "/build/llvm-toolchain-snapshot-13~++20210506100649+6304c0836a4d/clang/lib/AST/DeclTemplate.cpp" , 283, __extension__ __PRETTY_FUNCTION__)) | |||
280 | CorrectInsertPos,(static_cast <bool> (!findSpecializationImpl(Specializations , CorrectInsertPos, SETraits::getTemplateArgs(Entry)) && InsertPos == CorrectInsertPos && "given incorrect InsertPos for specialization" ) ? void (0) : __assert_fail ("!findSpecializationImpl(Specializations, CorrectInsertPos, SETraits::getTemplateArgs(Entry)) && InsertPos == CorrectInsertPos && \"given incorrect InsertPos for specialization\"" , "/build/llvm-toolchain-snapshot-13~++20210506100649+6304c0836a4d/clang/lib/AST/DeclTemplate.cpp" , 283, __extension__ __PRETTY_FUNCTION__)) | |||
281 | SETraits::getTemplateArgs(Entry)) &&(static_cast <bool> (!findSpecializationImpl(Specializations , CorrectInsertPos, SETraits::getTemplateArgs(Entry)) && InsertPos == CorrectInsertPos && "given incorrect InsertPos for specialization" ) ? void (0) : __assert_fail ("!findSpecializationImpl(Specializations, CorrectInsertPos, SETraits::getTemplateArgs(Entry)) && InsertPos == CorrectInsertPos && \"given incorrect InsertPos for specialization\"" , "/build/llvm-toolchain-snapshot-13~++20210506100649+6304c0836a4d/clang/lib/AST/DeclTemplate.cpp" , 283, __extension__ __PRETTY_FUNCTION__)) | |||
282 | InsertPos == CorrectInsertPos &&(static_cast <bool> (!findSpecializationImpl(Specializations , CorrectInsertPos, SETraits::getTemplateArgs(Entry)) && InsertPos == CorrectInsertPos && "given incorrect InsertPos for specialization" ) ? void (0) : __assert_fail ("!findSpecializationImpl(Specializations, CorrectInsertPos, SETraits::getTemplateArgs(Entry)) && InsertPos == CorrectInsertPos && \"given incorrect InsertPos for specialization\"" , "/build/llvm-toolchain-snapshot-13~++20210506100649+6304c0836a4d/clang/lib/AST/DeclTemplate.cpp" , 283, __extension__ __PRETTY_FUNCTION__)) | |||
283 | "given incorrect InsertPos for specialization")(static_cast <bool> (!findSpecializationImpl(Specializations , CorrectInsertPos, SETraits::getTemplateArgs(Entry)) && InsertPos == CorrectInsertPos && "given incorrect InsertPos for specialization" ) ? void (0) : __assert_fail ("!findSpecializationImpl(Specializations, CorrectInsertPos, SETraits::getTemplateArgs(Entry)) && InsertPos == CorrectInsertPos && \"given incorrect InsertPos for specialization\"" , "/build/llvm-toolchain-snapshot-13~++20210506100649+6304c0836a4d/clang/lib/AST/DeclTemplate.cpp" , 283, __extension__ __PRETTY_FUNCTION__)); | |||
284 | #endif | |||
285 | Specializations.InsertNode(Entry, InsertPos); | |||
286 | } else { | |||
287 | EntryType *Existing = Specializations.GetOrInsertNode(Entry); | |||
288 | (void)Existing; | |||
289 | assert(SETraits::getDecl(Existing)->isCanonicalDecl() &&(static_cast <bool> (SETraits::getDecl(Existing)->isCanonicalDecl () && "non-canonical specialization?") ? void (0) : __assert_fail ("SETraits::getDecl(Existing)->isCanonicalDecl() && \"non-canonical specialization?\"" , "/build/llvm-toolchain-snapshot-13~++20210506100649+6304c0836a4d/clang/lib/AST/DeclTemplate.cpp" , 290, __extension__ __PRETTY_FUNCTION__)) | |||
290 | "non-canonical specialization?")(static_cast <bool> (SETraits::getDecl(Existing)->isCanonicalDecl () && "non-canonical specialization?") ? void (0) : __assert_fail ("SETraits::getDecl(Existing)->isCanonicalDecl() && \"non-canonical specialization?\"" , "/build/llvm-toolchain-snapshot-13~++20210506100649+6304c0836a4d/clang/lib/AST/DeclTemplate.cpp" , 290, __extension__ __PRETTY_FUNCTION__)); | |||
291 | } | |||
292 | ||||
293 | if (ASTMutationListener *L = getASTMutationListener()) | |||
294 | L->AddedCXXTemplateSpecialization(cast<Derived>(this), | |||
295 | SETraits::getDecl(Entry)); | |||
296 | } | |||
297 | ||||
298 | //===----------------------------------------------------------------------===// | |||
299 | // FunctionTemplateDecl Implementation | |||
300 | //===----------------------------------------------------------------------===// | |||
301 | ||||
302 | FunctionTemplateDecl *FunctionTemplateDecl::Create(ASTContext &C, | |||
303 | DeclContext *DC, | |||
304 | SourceLocation L, | |||
305 | DeclarationName Name, | |||
306 | TemplateParameterList *Params, | |||
307 | NamedDecl *Decl) { | |||
308 | AdoptTemplateParameterList(Params, cast<DeclContext>(Decl)); | |||
309 | return new (C, DC) FunctionTemplateDecl(C, DC, L, Name, Params, Decl); | |||
310 | } | |||
311 | ||||
312 | FunctionTemplateDecl *FunctionTemplateDecl::CreateDeserialized(ASTContext &C, | |||
313 | unsigned ID) { | |||
314 | return new (C, ID) FunctionTemplateDecl(C, nullptr, SourceLocation(), | |||
315 | DeclarationName(), nullptr, nullptr); | |||
316 | } | |||
317 | ||||
318 | RedeclarableTemplateDecl::CommonBase * | |||
319 | FunctionTemplateDecl::newCommon(ASTContext &C) const { | |||
320 | auto *CommonPtr = new (C) Common; | |||
321 | C.addDestruction(CommonPtr); | |||
322 | return CommonPtr; | |||
323 | } | |||
324 | ||||
325 | void FunctionTemplateDecl::LoadLazySpecializations() const { | |||
326 | loadLazySpecializationsImpl(); | |||
327 | } | |||
328 | ||||
329 | llvm::FoldingSetVector<FunctionTemplateSpecializationInfo> & | |||
330 | FunctionTemplateDecl::getSpecializations() const { | |||
331 | LoadLazySpecializations(); | |||
332 | return getCommonPtr()->Specializations; | |||
333 | } | |||
334 | ||||
335 | FunctionDecl * | |||
336 | FunctionTemplateDecl::findSpecialization(ArrayRef<TemplateArgument> Args, | |||
337 | void *&InsertPos) { | |||
338 | return findSpecializationImpl(getSpecializations(), InsertPos, Args); | |||
339 | } | |||
340 | ||||
341 | void FunctionTemplateDecl::addSpecialization( | |||
342 | FunctionTemplateSpecializationInfo *Info, void *InsertPos) { | |||
343 | addSpecializationImpl<FunctionTemplateDecl>(getSpecializations(), Info, | |||
344 | InsertPos); | |||
345 | } | |||
346 | ||||
347 | ArrayRef<TemplateArgument> FunctionTemplateDecl::getInjectedTemplateArgs() { | |||
348 | TemplateParameterList *Params = getTemplateParameters(); | |||
349 | Common *CommonPtr = getCommonPtr(); | |||
350 | if (!CommonPtr->InjectedArgs) { | |||
351 | auto &Context = getASTContext(); | |||
352 | SmallVector<TemplateArgument, 16> TemplateArgs; | |||
353 | Context.getInjectedTemplateArgs(Params, TemplateArgs); | |||
354 | CommonPtr->InjectedArgs = | |||
355 | new (Context) TemplateArgument[TemplateArgs.size()]; | |||
356 | std::copy(TemplateArgs.begin(), TemplateArgs.end(), | |||
357 | CommonPtr->InjectedArgs); | |||
358 | } | |||
359 | ||||
360 | return llvm::makeArrayRef(CommonPtr->InjectedArgs, Params->size()); | |||
361 | } | |||
362 | ||||
363 | void FunctionTemplateDecl::mergePrevDecl(FunctionTemplateDecl *Prev) { | |||
364 | using Base = RedeclarableTemplateDecl; | |||
365 | ||||
366 | // If we haven't created a common pointer yet, then it can just be created | |||
367 | // with the usual method. | |||
368 | if (!Base::Common) | |||
369 | return; | |||
370 | ||||
371 | Common *ThisCommon = static_cast<Common *>(Base::Common); | |||
372 | Common *PrevCommon = nullptr; | |||
373 | SmallVector<FunctionTemplateDecl *, 8> PreviousDecls; | |||
374 | for (; Prev; Prev = Prev->getPreviousDecl()) { | |||
375 | if (Prev->Base::Common) { | |||
376 | PrevCommon = static_cast<Common *>(Prev->Base::Common); | |||
377 | break; | |||
378 | } | |||
379 | PreviousDecls.push_back(Prev); | |||
380 | } | |||
381 | ||||
382 | // If the previous redecl chain hasn't created a common pointer yet, then just | |||
383 | // use this common pointer. | |||
384 | if (!PrevCommon) { | |||
385 | for (auto *D : PreviousDecls) | |||
386 | D->Base::Common = ThisCommon; | |||
387 | return; | |||
388 | } | |||
389 | ||||
390 | // Ensure we don't leak any important state. | |||
391 | assert(ThisCommon->Specializations.size() == 0 &&(static_cast <bool> (ThisCommon->Specializations.size () == 0 && "Can't merge incompatible declarations!") ? void (0) : __assert_fail ("ThisCommon->Specializations.size() == 0 && \"Can't merge incompatible declarations!\"" , "/build/llvm-toolchain-snapshot-13~++20210506100649+6304c0836a4d/clang/lib/AST/DeclTemplate.cpp" , 392, __extension__ __PRETTY_FUNCTION__)) | |||
392 | "Can't merge incompatible declarations!")(static_cast <bool> (ThisCommon->Specializations.size () == 0 && "Can't merge incompatible declarations!") ? void (0) : __assert_fail ("ThisCommon->Specializations.size() == 0 && \"Can't merge incompatible declarations!\"" , "/build/llvm-toolchain-snapshot-13~++20210506100649+6304c0836a4d/clang/lib/AST/DeclTemplate.cpp" , 392, __extension__ __PRETTY_FUNCTION__)); | |||
393 | ||||
394 | Base::Common = PrevCommon; | |||
395 | } | |||
396 | ||||
397 | //===----------------------------------------------------------------------===// | |||
398 | // ClassTemplateDecl Implementation | |||
399 | //===----------------------------------------------------------------------===// | |||
400 | ||||
401 | ClassTemplateDecl *ClassTemplateDecl::Create(ASTContext &C, | |||
402 | DeclContext *DC, | |||
403 | SourceLocation L, | |||
404 | DeclarationName Name, | |||
405 | TemplateParameterList *Params, | |||
406 | NamedDecl *Decl) { | |||
407 | AdoptTemplateParameterList(Params, cast<DeclContext>(Decl)); | |||
408 | ||||
409 | return new (C, DC) ClassTemplateDecl(C, DC, L, Name, Params, Decl); | |||
410 | } | |||
411 | ||||
412 | ClassTemplateDecl *ClassTemplateDecl::CreateDeserialized(ASTContext &C, | |||
413 | unsigned ID) { | |||
414 | return new (C, ID) ClassTemplateDecl(C, nullptr, SourceLocation(), | |||
415 | DeclarationName(), nullptr, nullptr); | |||
416 | } | |||
417 | ||||
418 | void ClassTemplateDecl::LoadLazySpecializations() const { | |||
419 | loadLazySpecializationsImpl(); | |||
420 | } | |||
421 | ||||
422 | llvm::FoldingSetVector<ClassTemplateSpecializationDecl> & | |||
423 | ClassTemplateDecl::getSpecializations() const { | |||
424 | LoadLazySpecializations(); | |||
425 | return getCommonPtr()->Specializations; | |||
426 | } | |||
427 | ||||
428 | llvm::FoldingSetVector<ClassTemplatePartialSpecializationDecl> & | |||
429 | ClassTemplateDecl::getPartialSpecializations() const { | |||
430 | LoadLazySpecializations(); | |||
431 | return getCommonPtr()->PartialSpecializations; | |||
432 | } | |||
433 | ||||
434 | RedeclarableTemplateDecl::CommonBase * | |||
435 | ClassTemplateDecl::newCommon(ASTContext &C) const { | |||
436 | auto *CommonPtr = new (C) Common; | |||
437 | C.addDestruction(CommonPtr); | |||
438 | return CommonPtr; | |||
439 | } | |||
440 | ||||
441 | ClassTemplateSpecializationDecl * | |||
442 | ClassTemplateDecl::findSpecialization(ArrayRef<TemplateArgument> Args, | |||
443 | void *&InsertPos) { | |||
444 | return findSpecializationImpl(getSpecializations(), InsertPos, Args); | |||
445 | } | |||
446 | ||||
447 | void ClassTemplateDecl::AddSpecialization(ClassTemplateSpecializationDecl *D, | |||
448 | void *InsertPos) { | |||
449 | addSpecializationImpl<ClassTemplateDecl>(getSpecializations(), D, InsertPos); | |||
450 | } | |||
451 | ||||
452 | ClassTemplatePartialSpecializationDecl * | |||
453 | ClassTemplateDecl::findPartialSpecialization( | |||
454 | ArrayRef<TemplateArgument> Args, | |||
455 | TemplateParameterList *TPL, void *&InsertPos) { | |||
456 | return findSpecializationImpl(getPartialSpecializations(), InsertPos, Args, | |||
457 | TPL); | |||
458 | } | |||
459 | ||||
460 | static void ProfileTemplateParameterList(ASTContext &C, | |||
461 | llvm::FoldingSetNodeID &ID, const TemplateParameterList *TPL) { | |||
462 | const Expr *RC = TPL->getRequiresClause(); | |||
463 | ID.AddBoolean(RC != nullptr); | |||
464 | if (RC) | |||
465 | RC->Profile(ID, C, /*Canonical=*/true); | |||
466 | ID.AddInteger(TPL->size()); | |||
467 | for (NamedDecl *D : *TPL) { | |||
468 | if (const auto *NTTP = dyn_cast<NonTypeTemplateParmDecl>(D)) { | |||
469 | ID.AddInteger(0); | |||
470 | ID.AddBoolean(NTTP->isParameterPack()); | |||
471 | NTTP->getType().getCanonicalType().Profile(ID); | |||
472 | continue; | |||
473 | } | |||
474 | if (const auto *TTP = dyn_cast<TemplateTypeParmDecl>(D)) { | |||
475 | ID.AddInteger(1); | |||
476 | ID.AddBoolean(TTP->isParameterPack()); | |||
477 | ID.AddBoolean(TTP->hasTypeConstraint()); | |||
478 | if (const TypeConstraint *TC = TTP->getTypeConstraint()) | |||
479 | TC->getImmediatelyDeclaredConstraint()->Profile(ID, C, | |||
480 | /*Canonical=*/true); | |||
481 | continue; | |||
482 | } | |||
483 | const auto *TTP = cast<TemplateTemplateParmDecl>(D); | |||
484 | ID.AddInteger(2); | |||
485 | ID.AddBoolean(TTP->isParameterPack()); | |||
486 | ProfileTemplateParameterList(C, ID, TTP->getTemplateParameters()); | |||
487 | } | |||
488 | } | |||
489 | ||||
490 | void | |||
491 | ClassTemplatePartialSpecializationDecl::Profile(llvm::FoldingSetNodeID &ID, | |||
492 | ArrayRef<TemplateArgument> TemplateArgs, TemplateParameterList *TPL, | |||
493 | ASTContext &Context) { | |||
494 | ID.AddInteger(TemplateArgs.size()); | |||
495 | for (const TemplateArgument &TemplateArg : TemplateArgs) | |||
496 | TemplateArg.Profile(ID, Context); | |||
497 | ProfileTemplateParameterList(Context, ID, TPL); | |||
498 | } | |||
499 | ||||
500 | void ClassTemplateDecl::AddPartialSpecialization( | |||
501 | ClassTemplatePartialSpecializationDecl *D, | |||
502 | void *InsertPos) { | |||
503 | if (InsertPos) | |||
504 | getPartialSpecializations().InsertNode(D, InsertPos); | |||
505 | else { | |||
506 | ClassTemplatePartialSpecializationDecl *Existing | |||
507 | = getPartialSpecializations().GetOrInsertNode(D); | |||
508 | (void)Existing; | |||
509 | assert(Existing->isCanonicalDecl() && "Non-canonical specialization?")(static_cast <bool> (Existing->isCanonicalDecl() && "Non-canonical specialization?") ? void (0) : __assert_fail ( "Existing->isCanonicalDecl() && \"Non-canonical specialization?\"" , "/build/llvm-toolchain-snapshot-13~++20210506100649+6304c0836a4d/clang/lib/AST/DeclTemplate.cpp" , 509, __extension__ __PRETTY_FUNCTION__)); | |||
510 | } | |||
511 | ||||
512 | if (ASTMutationListener *L = getASTMutationListener()) | |||
513 | L->AddedCXXTemplateSpecialization(this, D); | |||
514 | } | |||
515 | ||||
516 | void ClassTemplateDecl::getPartialSpecializations( | |||
517 | SmallVectorImpl<ClassTemplatePartialSpecializationDecl *> &PS) const { | |||
518 | llvm::FoldingSetVector<ClassTemplatePartialSpecializationDecl> &PartialSpecs | |||
519 | = getPartialSpecializations(); | |||
520 | PS.clear(); | |||
521 | PS.reserve(PartialSpecs.size()); | |||
522 | for (ClassTemplatePartialSpecializationDecl &P : PartialSpecs) | |||
523 | PS.push_back(P.getMostRecentDecl()); | |||
524 | } | |||
525 | ||||
526 | ClassTemplatePartialSpecializationDecl * | |||
527 | ClassTemplateDecl::findPartialSpecialization(QualType T) { | |||
528 | ASTContext &Context = getASTContext(); | |||
529 | for (ClassTemplatePartialSpecializationDecl &P : | |||
530 | getPartialSpecializations()) { | |||
531 | if (Context.hasSameType(P.getInjectedSpecializationType(), T)) | |||
532 | return P.getMostRecentDecl(); | |||
533 | } | |||
534 | ||||
535 | return nullptr; | |||
536 | } | |||
537 | ||||
538 | ClassTemplatePartialSpecializationDecl * | |||
539 | ClassTemplateDecl::findPartialSpecInstantiatedFromMember( | |||
540 | ClassTemplatePartialSpecializationDecl *D) { | |||
541 | Decl *DCanon = D->getCanonicalDecl(); | |||
542 | for (ClassTemplatePartialSpecializationDecl &P : getPartialSpecializations()) { | |||
543 | if (P.getInstantiatedFromMember()->getCanonicalDecl() == DCanon) | |||
544 | return P.getMostRecentDecl(); | |||
545 | } | |||
546 | ||||
547 | return nullptr; | |||
548 | } | |||
549 | ||||
550 | QualType | |||
551 | ClassTemplateDecl::getInjectedClassNameSpecialization() { | |||
552 | Common *CommonPtr = getCommonPtr(); | |||
553 | if (!CommonPtr->InjectedClassNameType.isNull()) | |||
554 | return CommonPtr->InjectedClassNameType; | |||
555 | ||||
556 | // C++0x [temp.dep.type]p2: | |||
557 | // The template argument list of a primary template is a template argument | |||
558 | // list in which the nth template argument has the value of the nth template | |||
559 | // parameter of the class template. If the nth template parameter is a | |||
560 | // template parameter pack (14.5.3), the nth template argument is a pack | |||
561 | // expansion (14.5.3) whose pattern is the name of the template parameter | |||
562 | // pack. | |||
563 | ASTContext &Context = getASTContext(); | |||
564 | TemplateParameterList *Params = getTemplateParameters(); | |||
565 | SmallVector<TemplateArgument, 16> TemplateArgs; | |||
566 | Context.getInjectedTemplateArgs(Params, TemplateArgs); | |||
567 | CommonPtr->InjectedClassNameType | |||
568 | = Context.getTemplateSpecializationType(TemplateName(this), | |||
569 | TemplateArgs); | |||
570 | return CommonPtr->InjectedClassNameType; | |||
571 | } | |||
572 | ||||
573 | //===----------------------------------------------------------------------===// | |||
574 | // TemplateTypeParm Allocation/Deallocation Method Implementations | |||
575 | //===----------------------------------------------------------------------===// | |||
576 | ||||
577 | TemplateTypeParmDecl * | |||
578 | TemplateTypeParmDecl::Create(const ASTContext &C, DeclContext *DC, | |||
579 | SourceLocation KeyLoc, SourceLocation NameLoc, | |||
580 | unsigned D, unsigned P, IdentifierInfo *Id, | |||
581 | bool Typename, bool ParameterPack, | |||
582 | bool HasTypeConstraint, | |||
583 | Optional<unsigned> NumExpanded) { | |||
584 | auto *TTPDecl = | |||
585 | new (C, DC, | |||
586 | additionalSizeToAlloc<TypeConstraint>(HasTypeConstraint ? 1 : 0)) | |||
587 | TemplateTypeParmDecl(DC, KeyLoc, NameLoc, Id, Typename, | |||
588 | HasTypeConstraint, NumExpanded); | |||
589 | QualType TTPType = C.getTemplateTypeParmType(D, P, ParameterPack, TTPDecl); | |||
590 | TTPDecl->setTypeForDecl(TTPType.getTypePtr()); | |||
591 | return TTPDecl; | |||
592 | } | |||
593 | ||||
594 | TemplateTypeParmDecl * | |||
595 | TemplateTypeParmDecl::CreateDeserialized(const ASTContext &C, unsigned ID) { | |||
596 | return new (C, ID) TemplateTypeParmDecl(nullptr, SourceLocation(), | |||
597 | SourceLocation(), nullptr, false, | |||
598 | false, None); | |||
599 | } | |||
600 | ||||
601 | TemplateTypeParmDecl * | |||
602 | TemplateTypeParmDecl::CreateDeserialized(const ASTContext &C, unsigned ID, | |||
603 | bool HasTypeConstraint) { | |||
604 | return new (C, ID, | |||
605 | additionalSizeToAlloc<TypeConstraint>(HasTypeConstraint ? 1 : 0)) | |||
606 | TemplateTypeParmDecl(nullptr, SourceLocation(), SourceLocation(), | |||
607 | nullptr, false, HasTypeConstraint, None); | |||
608 | } | |||
609 | ||||
610 | SourceLocation TemplateTypeParmDecl::getDefaultArgumentLoc() const { | |||
611 | return hasDefaultArgument() | |||
612 | ? getDefaultArgumentInfo()->getTypeLoc().getBeginLoc() | |||
613 | : SourceLocation(); | |||
614 | } | |||
615 | ||||
616 | SourceRange TemplateTypeParmDecl::getSourceRange() const { | |||
617 | if (hasDefaultArgument() && !defaultArgumentWasInherited()) | |||
618 | return SourceRange(getBeginLoc(), | |||
619 | getDefaultArgumentInfo()->getTypeLoc().getEndLoc()); | |||
620 | // TypeDecl::getSourceRange returns a range containing name location, which is | |||
621 | // wrong for unnamed template parameters. e.g: | |||
622 | // it will return <[[typename>]] instead of <[[typename]]> | |||
623 | else if (getDeclName().isEmpty()) | |||
624 | return SourceRange(getBeginLoc()); | |||
625 | return TypeDecl::getSourceRange(); | |||
626 | } | |||
627 | ||||
628 | unsigned TemplateTypeParmDecl::getDepth() const { | |||
629 | return getTypeForDecl()->castAs<TemplateTypeParmType>()->getDepth(); | |||
630 | } | |||
631 | ||||
632 | unsigned TemplateTypeParmDecl::getIndex() const { | |||
633 | return getTypeForDecl()->castAs<TemplateTypeParmType>()->getIndex(); | |||
634 | } | |||
635 | ||||
636 | bool TemplateTypeParmDecl::isParameterPack() const { | |||
637 | return getTypeForDecl()->castAs<TemplateTypeParmType>()->isParameterPack(); | |||
638 | } | |||
639 | ||||
640 | void TemplateTypeParmDecl::setTypeConstraint(NestedNameSpecifierLoc NNS, | |||
641 | DeclarationNameInfo NameInfo, NamedDecl *FoundDecl, ConceptDecl *CD, | |||
642 | const ASTTemplateArgumentListInfo *ArgsAsWritten, | |||
643 | Expr *ImmediatelyDeclaredConstraint) { | |||
644 | assert(HasTypeConstraint &&(static_cast <bool> (HasTypeConstraint && "HasTypeConstraint=true must be passed at construction in order to " "call setTypeConstraint") ? void (0) : __assert_fail ("HasTypeConstraint && \"HasTypeConstraint=true must be passed at construction in order to \" \"call setTypeConstraint\"" , "/build/llvm-toolchain-snapshot-13~++20210506100649+6304c0836a4d/clang/lib/AST/DeclTemplate.cpp" , 646, __extension__ __PRETTY_FUNCTION__)) | |||
645 | "HasTypeConstraint=true must be passed at construction in order to "(static_cast <bool> (HasTypeConstraint && "HasTypeConstraint=true must be passed at construction in order to " "call setTypeConstraint") ? void (0) : __assert_fail ("HasTypeConstraint && \"HasTypeConstraint=true must be passed at construction in order to \" \"call setTypeConstraint\"" , "/build/llvm-toolchain-snapshot-13~++20210506100649+6304c0836a4d/clang/lib/AST/DeclTemplate.cpp" , 646, __extension__ __PRETTY_FUNCTION__)) | |||
646 | "call setTypeConstraint")(static_cast <bool> (HasTypeConstraint && "HasTypeConstraint=true must be passed at construction in order to " "call setTypeConstraint") ? void (0) : __assert_fail ("HasTypeConstraint && \"HasTypeConstraint=true must be passed at construction in order to \" \"call setTypeConstraint\"" , "/build/llvm-toolchain-snapshot-13~++20210506100649+6304c0836a4d/clang/lib/AST/DeclTemplate.cpp" , 646, __extension__ __PRETTY_FUNCTION__)); | |||
647 | assert(!TypeConstraintInitialized &&(static_cast <bool> (!TypeConstraintInitialized && "TypeConstraint was already initialized!") ? void (0) : __assert_fail ("!TypeConstraintInitialized && \"TypeConstraint was already initialized!\"" , "/build/llvm-toolchain-snapshot-13~++20210506100649+6304c0836a4d/clang/lib/AST/DeclTemplate.cpp" , 648, __extension__ __PRETTY_FUNCTION__)) | |||
648 | "TypeConstraint was already initialized!")(static_cast <bool> (!TypeConstraintInitialized && "TypeConstraint was already initialized!") ? void (0) : __assert_fail ("!TypeConstraintInitialized && \"TypeConstraint was already initialized!\"" , "/build/llvm-toolchain-snapshot-13~++20210506100649+6304c0836a4d/clang/lib/AST/DeclTemplate.cpp" , 648, __extension__ __PRETTY_FUNCTION__)); | |||
649 | new (getTrailingObjects<TypeConstraint>()) TypeConstraint(NNS, NameInfo, | |||
650 | FoundDecl, CD, ArgsAsWritten, ImmediatelyDeclaredConstraint); | |||
651 | TypeConstraintInitialized = true; | |||
652 | } | |||
653 | ||||
654 | //===----------------------------------------------------------------------===// | |||
655 | // NonTypeTemplateParmDecl Method Implementations | |||
656 | //===----------------------------------------------------------------------===// | |||
657 | ||||
658 | NonTypeTemplateParmDecl::NonTypeTemplateParmDecl( | |||
659 | DeclContext *DC, SourceLocation StartLoc, SourceLocation IdLoc, unsigned D, | |||
660 | unsigned P, IdentifierInfo *Id, QualType T, TypeSourceInfo *TInfo, | |||
661 | ArrayRef<QualType> ExpandedTypes, ArrayRef<TypeSourceInfo *> ExpandedTInfos) | |||
662 | : DeclaratorDecl(NonTypeTemplateParm, DC, IdLoc, Id, T, TInfo, StartLoc), | |||
663 | TemplateParmPosition(D, P), ParameterPack(true), | |||
664 | ExpandedParameterPack(true), NumExpandedTypes(ExpandedTypes.size()) { | |||
665 | if (!ExpandedTypes.empty() && !ExpandedTInfos.empty()) { | |||
666 | auto TypesAndInfos = | |||
667 | getTrailingObjects<std::pair<QualType, TypeSourceInfo *>>(); | |||
668 | for (unsigned I = 0; I != NumExpandedTypes; ++I) { | |||
669 | new (&TypesAndInfos[I].first) QualType(ExpandedTypes[I]); | |||
670 | TypesAndInfos[I].second = ExpandedTInfos[I]; | |||
671 | } | |||
672 | } | |||
673 | } | |||
674 | ||||
675 | NonTypeTemplateParmDecl * | |||
676 | NonTypeTemplateParmDecl::Create(const ASTContext &C, DeclContext *DC, | |||
677 | SourceLocation StartLoc, SourceLocation IdLoc, | |||
678 | unsigned D, unsigned P, IdentifierInfo *Id, | |||
679 | QualType T, bool ParameterPack, | |||
680 | TypeSourceInfo *TInfo) { | |||
681 | AutoType *AT = | |||
682 | C.getLangOpts().CPlusPlus20 ? T->getContainedAutoType() : nullptr; | |||
683 | return new (C, DC, | |||
684 | additionalSizeToAlloc<std::pair<QualType, TypeSourceInfo *>, | |||
685 | Expr *>(0, | |||
686 | AT && AT->isConstrained() ? 1 : 0)) | |||
687 | NonTypeTemplateParmDecl(DC, StartLoc, IdLoc, D, P, Id, T, ParameterPack, | |||
688 | TInfo); | |||
689 | } | |||
690 | ||||
691 | NonTypeTemplateParmDecl *NonTypeTemplateParmDecl::Create( | |||
692 | const ASTContext &C, DeclContext *DC, SourceLocation StartLoc, | |||
693 | SourceLocation IdLoc, unsigned D, unsigned P, IdentifierInfo *Id, | |||
694 | QualType T, TypeSourceInfo *TInfo, ArrayRef<QualType> ExpandedTypes, | |||
695 | ArrayRef<TypeSourceInfo *> ExpandedTInfos) { | |||
696 | AutoType *AT = TInfo->getType()->getContainedAutoType(); | |||
697 | return new (C, DC, | |||
698 | additionalSizeToAlloc<std::pair<QualType, TypeSourceInfo *>, | |||
699 | Expr *>( | |||
700 | ExpandedTypes.size(), AT && AT->isConstrained() ? 1 : 0)) | |||
701 | NonTypeTemplateParmDecl(DC, StartLoc, IdLoc, D, P, Id, T, TInfo, | |||
702 | ExpandedTypes, ExpandedTInfos); | |||
703 | } | |||
704 | ||||
705 | NonTypeTemplateParmDecl * | |||
706 | NonTypeTemplateParmDecl::CreateDeserialized(ASTContext &C, unsigned ID, | |||
707 | bool HasTypeConstraint) { | |||
708 | return new (C, ID, additionalSizeToAlloc<std::pair<QualType, | |||
709 | TypeSourceInfo *>, | |||
710 | Expr *>(0, | |||
711 | HasTypeConstraint ? 1 : 0)) | |||
712 | NonTypeTemplateParmDecl(nullptr, SourceLocation(), SourceLocation(), | |||
713 | 0, 0, nullptr, QualType(), false, nullptr); | |||
714 | } | |||
715 | ||||
716 | NonTypeTemplateParmDecl * | |||
717 | NonTypeTemplateParmDecl::CreateDeserialized(ASTContext &C, unsigned ID, | |||
718 | unsigned NumExpandedTypes, | |||
719 | bool HasTypeConstraint) { | |||
720 | auto *NTTP = | |||
721 | new (C, ID, additionalSizeToAlloc<std::pair<QualType, TypeSourceInfo *>, | |||
722 | Expr *>( | |||
723 | NumExpandedTypes, HasTypeConstraint ? 1 : 0)) | |||
724 | NonTypeTemplateParmDecl(nullptr, SourceLocation(), SourceLocation(), | |||
725 | 0, 0, nullptr, QualType(), nullptr, None, | |||
726 | None); | |||
727 | NTTP->NumExpandedTypes = NumExpandedTypes; | |||
728 | return NTTP; | |||
729 | } | |||
730 | ||||
731 | SourceRange NonTypeTemplateParmDecl::getSourceRange() const { | |||
732 | if (hasDefaultArgument() && !defaultArgumentWasInherited()) | |||
733 | return SourceRange(getOuterLocStart(), | |||
734 | getDefaultArgument()->getSourceRange().getEnd()); | |||
735 | return DeclaratorDecl::getSourceRange(); | |||
736 | } | |||
737 | ||||
738 | SourceLocation NonTypeTemplateParmDecl::getDefaultArgumentLoc() const { | |||
739 | return hasDefaultArgument() | |||
740 | ? getDefaultArgument()->getSourceRange().getBegin() | |||
741 | : SourceLocation(); | |||
742 | } | |||
743 | ||||
744 | //===----------------------------------------------------------------------===// | |||
745 | // TemplateTemplateParmDecl Method Implementations | |||
746 | //===----------------------------------------------------------------------===// | |||
747 | ||||
748 | void TemplateTemplateParmDecl::anchor() {} | |||
749 | ||||
750 | TemplateTemplateParmDecl::TemplateTemplateParmDecl( | |||
751 | DeclContext *DC, SourceLocation L, unsigned D, unsigned P, | |||
752 | IdentifierInfo *Id, TemplateParameterList *Params, | |||
753 | ArrayRef<TemplateParameterList *> Expansions) | |||
754 | : TemplateDecl(TemplateTemplateParm, DC, L, Id, Params), | |||
755 | TemplateParmPosition(D, P), ParameterPack(true), | |||
756 | ExpandedParameterPack(true), NumExpandedParams(Expansions.size()) { | |||
757 | if (!Expansions.empty()) | |||
758 | std::uninitialized_copy(Expansions.begin(), Expansions.end(), | |||
759 | getTrailingObjects<TemplateParameterList *>()); | |||
760 | } | |||
761 | ||||
762 | TemplateTemplateParmDecl * | |||
763 | TemplateTemplateParmDecl::Create(const ASTContext &C, DeclContext *DC, | |||
764 | SourceLocation L, unsigned D, unsigned P, | |||
765 | bool ParameterPack, IdentifierInfo *Id, | |||
766 | TemplateParameterList *Params) { | |||
767 | return new (C, DC) TemplateTemplateParmDecl(DC, L, D, P, ParameterPack, Id, | |||
768 | Params); | |||
769 | } | |||
770 | ||||
771 | TemplateTemplateParmDecl * | |||
772 | TemplateTemplateParmDecl::Create(const ASTContext &C, DeclContext *DC, | |||
773 | SourceLocation L, unsigned D, unsigned P, | |||
774 | IdentifierInfo *Id, | |||
775 | TemplateParameterList *Params, | |||
776 | ArrayRef<TemplateParameterList *> Expansions) { | |||
777 | return new (C, DC, | |||
778 | additionalSizeToAlloc<TemplateParameterList *>(Expansions.size())) | |||
779 | TemplateTemplateParmDecl(DC, L, D, P, Id, Params, Expansions); | |||
780 | } | |||
781 | ||||
782 | TemplateTemplateParmDecl * | |||
783 | TemplateTemplateParmDecl::CreateDeserialized(ASTContext &C, unsigned ID) { | |||
784 | return new (C, ID) TemplateTemplateParmDecl(nullptr, SourceLocation(), 0, 0, | |||
785 | false, nullptr, nullptr); | |||
786 | } | |||
787 | ||||
788 | TemplateTemplateParmDecl * | |||
789 | TemplateTemplateParmDecl::CreateDeserialized(ASTContext &C, unsigned ID, | |||
790 | unsigned NumExpansions) { | |||
791 | auto *TTP = | |||
792 | new (C, ID, additionalSizeToAlloc<TemplateParameterList *>(NumExpansions)) | |||
793 | TemplateTemplateParmDecl(nullptr, SourceLocation(), 0, 0, nullptr, | |||
794 | nullptr, None); | |||
795 | TTP->NumExpandedParams = NumExpansions; | |||
796 | return TTP; | |||
797 | } | |||
798 | ||||
799 | SourceLocation TemplateTemplateParmDecl::getDefaultArgumentLoc() const { | |||
800 | return hasDefaultArgument() ? getDefaultArgument().getLocation() | |||
801 | : SourceLocation(); | |||
802 | } | |||
803 | ||||
804 | void TemplateTemplateParmDecl::setDefaultArgument( | |||
805 | const ASTContext &C, const TemplateArgumentLoc &DefArg) { | |||
806 | if (DefArg.getArgument().isNull()) | |||
807 | DefaultArgument.set(nullptr); | |||
808 | else | |||
809 | DefaultArgument.set(new (C) TemplateArgumentLoc(DefArg)); | |||
810 | } | |||
811 | ||||
812 | //===----------------------------------------------------------------------===// | |||
813 | // TemplateArgumentList Implementation | |||
814 | //===----------------------------------------------------------------------===// | |||
815 | TemplateArgumentList::TemplateArgumentList(ArrayRef<TemplateArgument> Args) | |||
816 | : Arguments(getTrailingObjects<TemplateArgument>()), | |||
817 | NumArguments(Args.size()) { | |||
818 | std::uninitialized_copy(Args.begin(), Args.end(), | |||
819 | getTrailingObjects<TemplateArgument>()); | |||
820 | } | |||
821 | ||||
822 | TemplateArgumentList * | |||
823 | TemplateArgumentList::CreateCopy(ASTContext &Context, | |||
824 | ArrayRef<TemplateArgument> Args) { | |||
825 | void *Mem = Context.Allocate(totalSizeToAlloc<TemplateArgument>(Args.size())); | |||
826 | return new (Mem) TemplateArgumentList(Args); | |||
827 | } | |||
828 | ||||
829 | FunctionTemplateSpecializationInfo *FunctionTemplateSpecializationInfo::Create( | |||
830 | ASTContext &C, FunctionDecl *FD, FunctionTemplateDecl *Template, | |||
831 | TemplateSpecializationKind TSK, const TemplateArgumentList *TemplateArgs, | |||
832 | const TemplateArgumentListInfo *TemplateArgsAsWritten, SourceLocation POI, | |||
833 | MemberSpecializationInfo *MSInfo) { | |||
834 | const ASTTemplateArgumentListInfo *ArgsAsWritten = nullptr; | |||
835 | if (TemplateArgsAsWritten) | |||
836 | ArgsAsWritten = ASTTemplateArgumentListInfo::Create(C, | |||
837 | *TemplateArgsAsWritten); | |||
838 | ||||
839 | void *Mem = | |||
840 | C.Allocate(totalSizeToAlloc<MemberSpecializationInfo *>(MSInfo ? 1 : 0)); | |||
841 | return new (Mem) FunctionTemplateSpecializationInfo( | |||
842 | FD, Template, TSK, TemplateArgs, ArgsAsWritten, POI, MSInfo); | |||
843 | } | |||
844 | ||||
845 | //===----------------------------------------------------------------------===// | |||
846 | // ClassTemplateSpecializationDecl Implementation | |||
847 | //===----------------------------------------------------------------------===// | |||
848 | ||||
849 | ClassTemplateSpecializationDecl:: | |||
850 | ClassTemplateSpecializationDecl(ASTContext &Context, Kind DK, TagKind TK, | |||
851 | DeclContext *DC, SourceLocation StartLoc, | |||
852 | SourceLocation IdLoc, | |||
853 | ClassTemplateDecl *SpecializedTemplate, | |||
854 | ArrayRef<TemplateArgument> Args, | |||
855 | ClassTemplateSpecializationDecl *PrevDecl) | |||
856 | : CXXRecordDecl(DK, TK, Context, DC, StartLoc, IdLoc, | |||
857 | SpecializedTemplate->getIdentifier(), PrevDecl), | |||
858 | SpecializedTemplate(SpecializedTemplate), | |||
859 | TemplateArgs(TemplateArgumentList::CreateCopy(Context, Args)), | |||
860 | SpecializationKind(TSK_Undeclared) { | |||
861 | } | |||
862 | ||||
863 | ClassTemplateSpecializationDecl::ClassTemplateSpecializationDecl(ASTContext &C, | |||
864 | Kind DK) | |||
865 | : CXXRecordDecl(DK, TTK_Struct, C, nullptr, SourceLocation(), | |||
866 | SourceLocation(), nullptr, nullptr), | |||
867 | SpecializationKind(TSK_Undeclared) {} | |||
868 | ||||
869 | ClassTemplateSpecializationDecl * | |||
870 | ClassTemplateSpecializationDecl::Create(ASTContext &Context, TagKind TK, | |||
871 | DeclContext *DC, | |||
872 | SourceLocation StartLoc, | |||
873 | SourceLocation IdLoc, | |||
874 | ClassTemplateDecl *SpecializedTemplate, | |||
875 | ArrayRef<TemplateArgument> Args, | |||
876 | ClassTemplateSpecializationDecl *PrevDecl) { | |||
877 | auto *Result = | |||
878 | new (Context, DC) ClassTemplateSpecializationDecl( | |||
879 | Context, ClassTemplateSpecialization, TK, DC, StartLoc, IdLoc, | |||
880 | SpecializedTemplate, Args, PrevDecl); | |||
881 | Result->setMayHaveOutOfDateDef(false); | |||
882 | ||||
883 | Context.getTypeDeclType(Result, PrevDecl); | |||
884 | return Result; | |||
885 | } | |||
886 | ||||
887 | ClassTemplateSpecializationDecl * | |||
888 | ClassTemplateSpecializationDecl::CreateDeserialized(ASTContext &C, | |||
889 | unsigned ID) { | |||
890 | auto *Result = | |||
891 | new (C, ID) ClassTemplateSpecializationDecl(C, ClassTemplateSpecialization); | |||
892 | Result->setMayHaveOutOfDateDef(false); | |||
893 | return Result; | |||
894 | } | |||
895 | ||||
896 | void ClassTemplateSpecializationDecl::getNameForDiagnostic( | |||
897 | raw_ostream &OS, const PrintingPolicy &Policy, bool Qualified) const { | |||
898 | NamedDecl::getNameForDiagnostic(OS, Policy, Qualified); | |||
899 | ||||
900 | const auto *PS = dyn_cast<ClassTemplatePartialSpecializationDecl>(this); | |||
901 | if (const ASTTemplateArgumentListInfo *ArgsAsWritten = | |||
902 | PS ? PS->getTemplateArgsAsWritten() : nullptr) { | |||
903 | printTemplateArgumentList( | |||
904 | OS, ArgsAsWritten->arguments(), Policy, | |||
905 | getSpecializedTemplate()->getTemplateParameters()); | |||
906 | } else { | |||
907 | const TemplateArgumentList &TemplateArgs = getTemplateArgs(); | |||
908 | printTemplateArgumentList( | |||
909 | OS, TemplateArgs.asArray(), Policy, | |||
910 | getSpecializedTemplate()->getTemplateParameters()); | |||
911 | } | |||
912 | } | |||
913 | ||||
914 | ClassTemplateDecl * | |||
915 | ClassTemplateSpecializationDecl::getSpecializedTemplate() const { | |||
916 | if (const auto *PartialSpec = | |||
917 | SpecializedTemplate.dyn_cast<SpecializedPartialSpecialization*>()) | |||
918 | return PartialSpec->PartialSpecialization->getSpecializedTemplate(); | |||
919 | return SpecializedTemplate.get<ClassTemplateDecl*>(); | |||
920 | } | |||
921 | ||||
922 | SourceRange | |||
923 | ClassTemplateSpecializationDecl::getSourceRange() const { | |||
924 | if (ExplicitInfo) { | |||
925 | SourceLocation Begin = getTemplateKeywordLoc(); | |||
926 | if (Begin.isValid()) { | |||
927 | // Here we have an explicit (partial) specialization or instantiation. | |||
928 | assert(getSpecializationKind() == TSK_ExplicitSpecialization ||(static_cast <bool> (getSpecializationKind() == TSK_ExplicitSpecialization || getSpecializationKind() == TSK_ExplicitInstantiationDeclaration || getSpecializationKind() == TSK_ExplicitInstantiationDefinition ) ? void (0) : __assert_fail ("getSpecializationKind() == TSK_ExplicitSpecialization || getSpecializationKind() == TSK_ExplicitInstantiationDeclaration || getSpecializationKind() == TSK_ExplicitInstantiationDefinition" , "/build/llvm-toolchain-snapshot-13~++20210506100649+6304c0836a4d/clang/lib/AST/DeclTemplate.cpp" , 930, __extension__ __PRETTY_FUNCTION__)) | |||
929 | getSpecializationKind() == TSK_ExplicitInstantiationDeclaration ||(static_cast <bool> (getSpecializationKind() == TSK_ExplicitSpecialization || getSpecializationKind() == TSK_ExplicitInstantiationDeclaration || getSpecializationKind() == TSK_ExplicitInstantiationDefinition ) ? void (0) : __assert_fail ("getSpecializationKind() == TSK_ExplicitSpecialization || getSpecializationKind() == TSK_ExplicitInstantiationDeclaration || getSpecializationKind() == TSK_ExplicitInstantiationDefinition" , "/build/llvm-toolchain-snapshot-13~++20210506100649+6304c0836a4d/clang/lib/AST/DeclTemplate.cpp" , 930, __extension__ __PRETTY_FUNCTION__)) | |||
930 | getSpecializationKind() == TSK_ExplicitInstantiationDefinition)(static_cast <bool> (getSpecializationKind() == TSK_ExplicitSpecialization || getSpecializationKind() == TSK_ExplicitInstantiationDeclaration || getSpecializationKind() == TSK_ExplicitInstantiationDefinition ) ? void (0) : __assert_fail ("getSpecializationKind() == TSK_ExplicitSpecialization || getSpecializationKind() == TSK_ExplicitInstantiationDeclaration || getSpecializationKind() == TSK_ExplicitInstantiationDefinition" , "/build/llvm-toolchain-snapshot-13~++20210506100649+6304c0836a4d/clang/lib/AST/DeclTemplate.cpp" , 930, __extension__ __PRETTY_FUNCTION__)); | |||
931 | if (getExternLoc().isValid()) | |||
932 | Begin = getExternLoc(); | |||
933 | SourceLocation End = getBraceRange().getEnd(); | |||
934 | if (End.isInvalid()) | |||
935 | End = getTypeAsWritten()->getTypeLoc().getEndLoc(); | |||
936 | return SourceRange(Begin, End); | |||
937 | } | |||
938 | // An implicit instantiation of a class template partial specialization | |||
939 | // uses ExplicitInfo to record the TypeAsWritten, but the source | |||
940 | // locations should be retrieved from the instantiation pattern. | |||
941 | using CTPSDecl = ClassTemplatePartialSpecializationDecl; | |||
942 | auto *ctpsd = const_cast<CTPSDecl *>(cast<CTPSDecl>(this)); | |||
943 | CTPSDecl *inst_from = ctpsd->getInstantiatedFromMember(); | |||
944 | assert(inst_from != nullptr)(static_cast <bool> (inst_from != nullptr) ? void (0) : __assert_fail ("inst_from != nullptr", "/build/llvm-toolchain-snapshot-13~++20210506100649+6304c0836a4d/clang/lib/AST/DeclTemplate.cpp" , 944, __extension__ __PRETTY_FUNCTION__)); | |||
945 | return inst_from->getSourceRange(); | |||
946 | } | |||
947 | else { | |||
948 | // No explicit info available. | |||
949 | llvm::PointerUnion<ClassTemplateDecl *, | |||
950 | ClassTemplatePartialSpecializationDecl *> | |||
951 | inst_from = getInstantiatedFrom(); | |||
952 | if (inst_from.isNull()) | |||
953 | return getSpecializedTemplate()->getSourceRange(); | |||
954 | if (const auto *ctd = inst_from.dyn_cast<ClassTemplateDecl *>()) | |||
955 | return ctd->getSourceRange(); | |||
956 | return inst_from.get<ClassTemplatePartialSpecializationDecl *>() | |||
957 | ->getSourceRange(); | |||
958 | } | |||
959 | } | |||
960 | ||||
961 | //===----------------------------------------------------------------------===// | |||
962 | // ConceptDecl Implementation | |||
963 | //===----------------------------------------------------------------------===// | |||
964 | ConceptDecl *ConceptDecl::Create(ASTContext &C, DeclContext *DC, | |||
965 | SourceLocation L, DeclarationName Name, | |||
966 | TemplateParameterList *Params, | |||
967 | Expr *ConstraintExpr) { | |||
968 | AdoptTemplateParameterList(Params, DC); | |||
969 | return new (C, DC) ConceptDecl(DC, L, Name, Params, ConstraintExpr); | |||
970 | } | |||
971 | ||||
972 | ConceptDecl *ConceptDecl::CreateDeserialized(ASTContext &C, | |||
973 | unsigned ID) { | |||
974 | ConceptDecl *Result = new (C, ID) ConceptDecl(nullptr, SourceLocation(), | |||
975 | DeclarationName(), | |||
976 | nullptr, nullptr); | |||
977 | ||||
978 | return Result; | |||
979 | } | |||
980 | ||||
981 | //===----------------------------------------------------------------------===// | |||
982 | // ClassTemplatePartialSpecializationDecl Implementation | |||
983 | //===----------------------------------------------------------------------===// | |||
984 | void ClassTemplatePartialSpecializationDecl::anchor() {} | |||
985 | ||||
986 | ClassTemplatePartialSpecializationDecl:: | |||
987 | ClassTemplatePartialSpecializationDecl(ASTContext &Context, TagKind TK, | |||
988 | DeclContext *DC, | |||
989 | SourceLocation StartLoc, | |||
990 | SourceLocation IdLoc, | |||
991 | TemplateParameterList *Params, | |||
992 | ClassTemplateDecl *SpecializedTemplate, | |||
993 | ArrayRef<TemplateArgument> Args, | |||
994 | const ASTTemplateArgumentListInfo *ArgInfos, | |||
995 | ClassTemplatePartialSpecializationDecl *PrevDecl) | |||
996 | : ClassTemplateSpecializationDecl(Context, | |||
997 | ClassTemplatePartialSpecialization, | |||
998 | TK, DC, StartLoc, IdLoc, | |||
999 | SpecializedTemplate, Args, PrevDecl), | |||
1000 | TemplateParams(Params), ArgsAsWritten(ArgInfos), | |||
1001 | InstantiatedFromMember(nullptr, false) { | |||
1002 | AdoptTemplateParameterList(Params, this); | |||
1003 | } | |||
1004 | ||||
1005 | ClassTemplatePartialSpecializationDecl * | |||
1006 | ClassTemplatePartialSpecializationDecl:: | |||
1007 | Create(ASTContext &Context, TagKind TK,DeclContext *DC, | |||
1008 | SourceLocation StartLoc, SourceLocation IdLoc, | |||
1009 | TemplateParameterList *Params, | |||
1010 | ClassTemplateDecl *SpecializedTemplate, | |||
1011 | ArrayRef<TemplateArgument> Args, | |||
1012 | const TemplateArgumentListInfo &ArgInfos, | |||
1013 | QualType CanonInjectedType, | |||
1014 | ClassTemplatePartialSpecializationDecl *PrevDecl) { | |||
1015 | const ASTTemplateArgumentListInfo *ASTArgInfos = | |||
1016 | ASTTemplateArgumentListInfo::Create(Context, ArgInfos); | |||
1017 | ||||
1018 | auto *Result = new (Context, DC) | |||
1019 | ClassTemplatePartialSpecializationDecl(Context, TK, DC, StartLoc, IdLoc, | |||
1020 | Params, SpecializedTemplate, Args, | |||
1021 | ASTArgInfos, PrevDecl); | |||
1022 | Result->setSpecializationKind(TSK_ExplicitSpecialization); | |||
1023 | Result->setMayHaveOutOfDateDef(false); | |||
1024 | ||||
1025 | Context.getInjectedClassNameType(Result, CanonInjectedType); | |||
1026 | return Result; | |||
1027 | } | |||
1028 | ||||
1029 | ClassTemplatePartialSpecializationDecl * | |||
1030 | ClassTemplatePartialSpecializationDecl::CreateDeserialized(ASTContext &C, | |||
1031 | unsigned ID) { | |||
1032 | auto *Result = new (C, ID) ClassTemplatePartialSpecializationDecl(C); | |||
1033 | Result->setMayHaveOutOfDateDef(false); | |||
1034 | return Result; | |||
1035 | } | |||
1036 | ||||
1037 | //===----------------------------------------------------------------------===// | |||
1038 | // FriendTemplateDecl Implementation | |||
1039 | //===----------------------------------------------------------------------===// | |||
1040 | ||||
1041 | void FriendTemplateDecl::anchor() {} | |||
1042 | ||||
1043 | FriendTemplateDecl * | |||
1044 | FriendTemplateDecl::Create(ASTContext &Context, DeclContext *DC, | |||
1045 | SourceLocation L, | |||
1046 | MutableArrayRef<TemplateParameterList *> Params, | |||
1047 | FriendUnion Friend, SourceLocation FLoc) { | |||
1048 | return new (Context, DC) FriendTemplateDecl(DC, L, Params, Friend, FLoc); | |||
1049 | } | |||
1050 | ||||
1051 | FriendTemplateDecl *FriendTemplateDecl::CreateDeserialized(ASTContext &C, | |||
1052 | unsigned ID) { | |||
1053 | return new (C, ID) FriendTemplateDecl(EmptyShell()); | |||
1054 | } | |||
1055 | ||||
1056 | //===----------------------------------------------------------------------===// | |||
1057 | // TypeAliasTemplateDecl Implementation | |||
1058 | //===----------------------------------------------------------------------===// | |||
1059 | ||||
1060 | TypeAliasTemplateDecl *TypeAliasTemplateDecl::Create(ASTContext &C, | |||
1061 | DeclContext *DC, | |||
1062 | SourceLocation L, | |||
1063 | DeclarationName Name, | |||
1064 | TemplateParameterList *Params, | |||
1065 | NamedDecl *Decl) { | |||
1066 | AdoptTemplateParameterList(Params, DC); | |||
1067 | return new (C, DC) TypeAliasTemplateDecl(C, DC, L, Name, Params, Decl); | |||
1068 | } | |||
1069 | ||||
1070 | TypeAliasTemplateDecl *TypeAliasTemplateDecl::CreateDeserialized(ASTContext &C, | |||
1071 | unsigned ID) { | |||
1072 | return new (C, ID) TypeAliasTemplateDecl(C, nullptr, SourceLocation(), | |||
1073 | DeclarationName(), nullptr, nullptr); | |||
1074 | } | |||
1075 | ||||
1076 | RedeclarableTemplateDecl::CommonBase * | |||
1077 | TypeAliasTemplateDecl::newCommon(ASTContext &C) const { | |||
1078 | auto *CommonPtr = new (C) Common; | |||
1079 | C.addDestruction(CommonPtr); | |||
1080 | return CommonPtr; | |||
1081 | } | |||
1082 | ||||
1083 | //===----------------------------------------------------------------------===// | |||
1084 | // ClassScopeFunctionSpecializationDecl Implementation | |||
1085 | //===----------------------------------------------------------------------===// | |||
1086 | ||||
1087 | void ClassScopeFunctionSpecializationDecl::anchor() {} | |||
1088 | ||||
1089 | ClassScopeFunctionSpecializationDecl * | |||
1090 | ClassScopeFunctionSpecializationDecl::CreateDeserialized(ASTContext &C, | |||
1091 | unsigned ID) { | |||
1092 | return new (C, ID) ClassScopeFunctionSpecializationDecl( | |||
1093 | nullptr, SourceLocation(), nullptr, nullptr); | |||
1094 | } | |||
1095 | ||||
1096 | //===----------------------------------------------------------------------===// | |||
1097 | // VarTemplateDecl Implementation | |||
1098 | //===----------------------------------------------------------------------===// | |||
1099 | ||||
1100 | VarTemplateDecl *VarTemplateDecl::getDefinition() { | |||
1101 | VarTemplateDecl *CurD = this; | |||
1102 | while (CurD) { | |||
1103 | if (CurD->isThisDeclarationADefinition()) | |||
1104 | return CurD; | |||
1105 | CurD = CurD->getPreviousDecl(); | |||
1106 | } | |||
1107 | return nullptr; | |||
1108 | } | |||
1109 | ||||
1110 | VarTemplateDecl *VarTemplateDecl::Create(ASTContext &C, DeclContext *DC, | |||
1111 | SourceLocation L, DeclarationName Name, | |||
1112 | TemplateParameterList *Params, | |||
1113 | VarDecl *Decl) { | |||
1114 | AdoptTemplateParameterList(Params, DC); | |||
1115 | return new (C, DC) VarTemplateDecl(C, DC, L, Name, Params, Decl); | |||
1116 | } | |||
1117 | ||||
1118 | VarTemplateDecl *VarTemplateDecl::CreateDeserialized(ASTContext &C, | |||
1119 | unsigned ID) { | |||
1120 | return new (C, ID) VarTemplateDecl(C, nullptr, SourceLocation(), | |||
1121 | DeclarationName(), nullptr, nullptr); | |||
1122 | } | |||
1123 | ||||
1124 | void VarTemplateDecl::LoadLazySpecializations() const { | |||
1125 | loadLazySpecializationsImpl(); | |||
1126 | } | |||
1127 | ||||
1128 | llvm::FoldingSetVector<VarTemplateSpecializationDecl> & | |||
1129 | VarTemplateDecl::getSpecializations() const { | |||
1130 | LoadLazySpecializations(); | |||
1131 | return getCommonPtr()->Specializations; | |||
1132 | } | |||
1133 | ||||
1134 | llvm::FoldingSetVector<VarTemplatePartialSpecializationDecl> & | |||
1135 | VarTemplateDecl::getPartialSpecializations() const { | |||
1136 | LoadLazySpecializations(); | |||
1137 | return getCommonPtr()->PartialSpecializations; | |||
1138 | } | |||
1139 | ||||
1140 | RedeclarableTemplateDecl::CommonBase * | |||
1141 | VarTemplateDecl::newCommon(ASTContext &C) const { | |||
1142 | auto *CommonPtr = new (C) Common; | |||
1143 | C.addDestruction(CommonPtr); | |||
1144 | return CommonPtr; | |||
1145 | } | |||
1146 | ||||
1147 | VarTemplateSpecializationDecl * | |||
1148 | VarTemplateDecl::findSpecialization(ArrayRef<TemplateArgument> Args, | |||
1149 | void *&InsertPos) { | |||
1150 | return findSpecializationImpl(getSpecializations(), InsertPos, Args); | |||
1151 | } | |||
1152 | ||||
1153 | void VarTemplateDecl::AddSpecialization(VarTemplateSpecializationDecl *D, | |||
1154 | void *InsertPos) { | |||
1155 | addSpecializationImpl<VarTemplateDecl>(getSpecializations(), D, InsertPos); | |||
1156 | } | |||
1157 | ||||
1158 | VarTemplatePartialSpecializationDecl * | |||
1159 | VarTemplateDecl::findPartialSpecialization(ArrayRef<TemplateArgument> Args, | |||
1160 | TemplateParameterList *TPL, void *&InsertPos) { | |||
1161 | return findSpecializationImpl(getPartialSpecializations(), InsertPos, Args, | |||
1162 | TPL); | |||
1163 | } | |||
1164 | ||||
1165 | void | |||
1166 | VarTemplatePartialSpecializationDecl::Profile(llvm::FoldingSetNodeID &ID, | |||
1167 | ArrayRef<TemplateArgument> TemplateArgs, TemplateParameterList *TPL, | |||
1168 | ASTContext &Context) { | |||
1169 | ID.AddInteger(TemplateArgs.size()); | |||
1170 | for (const TemplateArgument &TemplateArg : TemplateArgs) | |||
1171 | TemplateArg.Profile(ID, Context); | |||
1172 | ProfileTemplateParameterList(Context, ID, TPL); | |||
1173 | } | |||
1174 | ||||
1175 | void VarTemplateDecl::AddPartialSpecialization( | |||
1176 | VarTemplatePartialSpecializationDecl *D, void *InsertPos) { | |||
1177 | if (InsertPos) | |||
1178 | getPartialSpecializations().InsertNode(D, InsertPos); | |||
1179 | else { | |||
1180 | VarTemplatePartialSpecializationDecl *Existing = | |||
1181 | getPartialSpecializations().GetOrInsertNode(D); | |||
1182 | (void)Existing; | |||
1183 | assert(Existing->isCanonicalDecl() && "Non-canonical specialization?")(static_cast <bool> (Existing->isCanonicalDecl() && "Non-canonical specialization?") ? void (0) : __assert_fail ( "Existing->isCanonicalDecl() && \"Non-canonical specialization?\"" , "/build/llvm-toolchain-snapshot-13~++20210506100649+6304c0836a4d/clang/lib/AST/DeclTemplate.cpp" , 1183, __extension__ __PRETTY_FUNCTION__)); | |||
1184 | } | |||
1185 | ||||
1186 | if (ASTMutationListener *L = getASTMutationListener()) | |||
1187 | L->AddedCXXTemplateSpecialization(this, D); | |||
1188 | } | |||
1189 | ||||
1190 | void VarTemplateDecl::getPartialSpecializations( | |||
1191 | SmallVectorImpl<VarTemplatePartialSpecializationDecl *> &PS) const { | |||
1192 | llvm::FoldingSetVector<VarTemplatePartialSpecializationDecl> &PartialSpecs = | |||
1193 | getPartialSpecializations(); | |||
1194 | PS.clear(); | |||
1195 | PS.reserve(PartialSpecs.size()); | |||
1196 | for (VarTemplatePartialSpecializationDecl &P : PartialSpecs) | |||
1197 | PS.push_back(P.getMostRecentDecl()); | |||
1198 | } | |||
1199 | ||||
1200 | VarTemplatePartialSpecializationDecl * | |||
1201 | VarTemplateDecl::findPartialSpecInstantiatedFromMember( | |||
1202 | VarTemplatePartialSpecializationDecl *D) { | |||
1203 | Decl *DCanon = D->getCanonicalDecl(); | |||
1204 | for (VarTemplatePartialSpecializationDecl &P : getPartialSpecializations()) { | |||
1205 | if (P.getInstantiatedFromMember()->getCanonicalDecl() == DCanon) | |||
1206 | return P.getMostRecentDecl(); | |||
1207 | } | |||
1208 | ||||
1209 | return nullptr; | |||
1210 | } | |||
1211 | ||||
1212 | //===----------------------------------------------------------------------===// | |||
1213 | // VarTemplateSpecializationDecl Implementation | |||
1214 | //===----------------------------------------------------------------------===// | |||
1215 | ||||
1216 | VarTemplateSpecializationDecl::VarTemplateSpecializationDecl( | |||
1217 | Kind DK, ASTContext &Context, DeclContext *DC, SourceLocation StartLoc, | |||
1218 | SourceLocation IdLoc, VarTemplateDecl *SpecializedTemplate, QualType T, | |||
1219 | TypeSourceInfo *TInfo, StorageClass S, ArrayRef<TemplateArgument> Args) | |||
1220 | : VarDecl(DK, Context, DC, StartLoc, IdLoc, | |||
1221 | SpecializedTemplate->getIdentifier(), T, TInfo, S), | |||
1222 | SpecializedTemplate(SpecializedTemplate), | |||
1223 | TemplateArgs(TemplateArgumentList::CreateCopy(Context, Args)), | |||
1224 | SpecializationKind(TSK_Undeclared), IsCompleteDefinition(false) {} | |||
1225 | ||||
1226 | VarTemplateSpecializationDecl::VarTemplateSpecializationDecl(Kind DK, | |||
1227 | ASTContext &C) | |||
1228 | : VarDecl(DK, C, nullptr, SourceLocation(), SourceLocation(), nullptr, | |||
1229 | QualType(), nullptr, SC_None), | |||
1230 | SpecializationKind(TSK_Undeclared), IsCompleteDefinition(false) {} | |||
1231 | ||||
1232 | VarTemplateSpecializationDecl *VarTemplateSpecializationDecl::Create( | |||
1233 | ASTContext &Context, DeclContext *DC, SourceLocation StartLoc, | |||
1234 | SourceLocation IdLoc, VarTemplateDecl *SpecializedTemplate, QualType T, | |||
1235 | TypeSourceInfo *TInfo, StorageClass S, ArrayRef<TemplateArgument> Args) { | |||
1236 | return new (Context, DC) VarTemplateSpecializationDecl( | |||
1237 | VarTemplateSpecialization, Context, DC, StartLoc, IdLoc, | |||
1238 | SpecializedTemplate, T, TInfo, S, Args); | |||
1239 | } | |||
1240 | ||||
1241 | VarTemplateSpecializationDecl * | |||
1242 | VarTemplateSpecializationDecl::CreateDeserialized(ASTContext &C, unsigned ID) { | |||
1243 | return new (C, ID) | |||
1244 | VarTemplateSpecializationDecl(VarTemplateSpecialization, C); | |||
1245 | } | |||
1246 | ||||
1247 | void VarTemplateSpecializationDecl::getNameForDiagnostic( | |||
1248 | raw_ostream &OS, const PrintingPolicy &Policy, bool Qualified) const { | |||
1249 | NamedDecl::getNameForDiagnostic(OS, Policy, Qualified); | |||
1250 | ||||
1251 | const auto *PS = dyn_cast<VarTemplatePartialSpecializationDecl>(this); | |||
1252 | if (const ASTTemplateArgumentListInfo *ArgsAsWritten = | |||
1253 | PS ? PS->getTemplateArgsAsWritten() : nullptr) { | |||
1254 | printTemplateArgumentList( | |||
1255 | OS, ArgsAsWritten->arguments(), Policy, | |||
1256 | getSpecializedTemplate()->getTemplateParameters()); | |||
1257 | } else { | |||
1258 | const TemplateArgumentList &TemplateArgs = getTemplateArgs(); | |||
1259 | printTemplateArgumentList( | |||
1260 | OS, TemplateArgs.asArray(), Policy, | |||
1261 | getSpecializedTemplate()->getTemplateParameters()); | |||
1262 | } | |||
1263 | } | |||
1264 | ||||
1265 | VarTemplateDecl *VarTemplateSpecializationDecl::getSpecializedTemplate() const { | |||
1266 | if (const auto *PartialSpec = | |||
1267 | SpecializedTemplate.dyn_cast<SpecializedPartialSpecialization *>()) | |||
1268 | return PartialSpec->PartialSpecialization->getSpecializedTemplate(); | |||
1269 | return SpecializedTemplate.get<VarTemplateDecl *>(); | |||
1270 | } | |||
1271 | ||||
1272 | void VarTemplateSpecializationDecl::setTemplateArgsInfo( | |||
1273 | const TemplateArgumentListInfo &ArgsInfo) { | |||
1274 | TemplateArgsInfo.setLAngleLoc(ArgsInfo.getLAngleLoc()); | |||
1275 | TemplateArgsInfo.setRAngleLoc(ArgsInfo.getRAngleLoc()); | |||
1276 | for (const TemplateArgumentLoc &Loc : ArgsInfo.arguments()) | |||
1277 | TemplateArgsInfo.addArgument(Loc); | |||
1278 | } | |||
1279 | ||||
1280 | //===----------------------------------------------------------------------===// | |||
1281 | // VarTemplatePartialSpecializationDecl Implementation | |||
1282 | //===----------------------------------------------------------------------===// | |||
1283 | ||||
1284 | void VarTemplatePartialSpecializationDecl::anchor() {} | |||
1285 | ||||
1286 | VarTemplatePartialSpecializationDecl::VarTemplatePartialSpecializationDecl( | |||
1287 | ASTContext &Context, DeclContext *DC, SourceLocation StartLoc, | |||
1288 | SourceLocation IdLoc, TemplateParameterList *Params, | |||
1289 | VarTemplateDecl *SpecializedTemplate, QualType T, TypeSourceInfo *TInfo, | |||
1290 | StorageClass S, ArrayRef<TemplateArgument> Args, | |||
1291 | const ASTTemplateArgumentListInfo *ArgInfos) | |||
1292 | : VarTemplateSpecializationDecl(VarTemplatePartialSpecialization, Context, | |||
1293 | DC, StartLoc, IdLoc, SpecializedTemplate, T, | |||
1294 | TInfo, S, Args), | |||
1295 | TemplateParams(Params), ArgsAsWritten(ArgInfos), | |||
1296 | InstantiatedFromMember(nullptr, false) { | |||
1297 | // TODO: The template parameters should be in DC by now. Verify. | |||
1298 | // AdoptTemplateParameterList(Params, DC); | |||
1299 | } | |||
1300 | ||||
1301 | VarTemplatePartialSpecializationDecl * | |||
1302 | VarTemplatePartialSpecializationDecl::Create( | |||
1303 | ASTContext &Context, DeclContext *DC, SourceLocation StartLoc, | |||
1304 | SourceLocation IdLoc, TemplateParameterList *Params, | |||
1305 | VarTemplateDecl *SpecializedTemplate, QualType T, TypeSourceInfo *TInfo, | |||
1306 | StorageClass S, ArrayRef<TemplateArgument> Args, | |||
1307 | const TemplateArgumentListInfo &ArgInfos) { | |||
1308 | const ASTTemplateArgumentListInfo *ASTArgInfos | |||
1309 | = ASTTemplateArgumentListInfo::Create(Context, ArgInfos); | |||
1310 | ||||
1311 | auto *Result = | |||
1312 | new (Context, DC) VarTemplatePartialSpecializationDecl( | |||
1313 | Context, DC, StartLoc, IdLoc, Params, SpecializedTemplate, T, TInfo, | |||
1314 | S, Args, ASTArgInfos); | |||
1315 | Result->setSpecializationKind(TSK_ExplicitSpecialization); | |||
1316 | return Result; | |||
1317 | } | |||
1318 | ||||
1319 | VarTemplatePartialSpecializationDecl * | |||
1320 | VarTemplatePartialSpecializationDecl::CreateDeserialized(ASTContext &C, | |||
1321 | unsigned ID) { | |||
1322 | return new (C, ID) VarTemplatePartialSpecializationDecl(C); | |||
1323 | } | |||
1324 | ||||
1325 | static TemplateParameterList * | |||
1326 | createMakeIntegerSeqParameterList(const ASTContext &C, DeclContext *DC) { | |||
1327 | // typename T | |||
1328 | auto *T = TemplateTypeParmDecl::Create( | |||
1329 | C, DC, SourceLocation(), SourceLocation(), /*Depth=*/1, /*Position=*/0, | |||
1330 | /*Id=*/nullptr, /*Typename=*/true, /*ParameterPack=*/false, | |||
1331 | /*HasTypeConstraint=*/false); | |||
1332 | T->setImplicit(true); | |||
1333 | ||||
1334 | // T ...Ints | |||
1335 | TypeSourceInfo *TI = | |||
1336 | C.getTrivialTypeSourceInfo(QualType(T->getTypeForDecl(), 0)); | |||
1337 | auto *N = NonTypeTemplateParmDecl::Create( | |||
1338 | C, DC, SourceLocation(), SourceLocation(), /*Depth=*/0, /*Position=*/1, | |||
1339 | /*Id=*/nullptr, TI->getType(), /*ParameterPack=*/true, TI); | |||
1340 | N->setImplicit(true); | |||
1341 | ||||
1342 | // <typename T, T ...Ints> | |||
1343 | NamedDecl *P[2] = {T, N}; | |||
1344 | auto *TPL = TemplateParameterList::Create( | |||
1345 | C, SourceLocation(), SourceLocation(), P, SourceLocation(), nullptr); | |||
1346 | ||||
1347 | // template <typename T, ...Ints> class IntSeq | |||
1348 | auto *TemplateTemplateParm = TemplateTemplateParmDecl::Create( | |||
1349 | C, DC, SourceLocation(), /*Depth=*/0, /*Position=*/0, | |||
1350 | /*ParameterPack=*/false, /*Id=*/nullptr, TPL); | |||
1351 | TemplateTemplateParm->setImplicit(true); | |||
1352 | ||||
1353 | // typename T | |||
1354 | auto *TemplateTypeParm = TemplateTypeParmDecl::Create( | |||
1355 | C, DC, SourceLocation(), SourceLocation(), /*Depth=*/0, /*Position=*/1, | |||
1356 | /*Id=*/nullptr, /*Typename=*/true, /*ParameterPack=*/false, | |||
1357 | /*HasTypeConstraint=*/false); | |||
1358 | TemplateTypeParm->setImplicit(true); | |||
1359 | ||||
1360 | // T N | |||
1361 | TypeSourceInfo *TInfo = C.getTrivialTypeSourceInfo( | |||
1362 | QualType(TemplateTypeParm->getTypeForDecl(), 0)); | |||
1363 | auto *NonTypeTemplateParm = NonTypeTemplateParmDecl::Create( | |||
1364 | C, DC, SourceLocation(), SourceLocation(), /*Depth=*/0, /*Position=*/2, | |||
1365 | /*Id=*/nullptr, TInfo->getType(), /*ParameterPack=*/false, TInfo); | |||
1366 | NamedDecl *Params[] = {TemplateTemplateParm, TemplateTypeParm, | |||
1367 | NonTypeTemplateParm}; | |||
1368 | ||||
1369 | // template <template <typename T, T ...Ints> class IntSeq, typename T, T N> | |||
1370 | return TemplateParameterList::Create(C, SourceLocation(), SourceLocation(), | |||
1371 | Params, SourceLocation(), nullptr); | |||
1372 | } | |||
1373 | ||||
1374 | static TemplateParameterList * | |||
1375 | createTypePackElementParameterList(const ASTContext &C, DeclContext *DC) { | |||
1376 | // std::size_t Index | |||
1377 | TypeSourceInfo *TInfo = C.getTrivialTypeSourceInfo(C.getSizeType()); | |||
1378 | auto *Index = NonTypeTemplateParmDecl::Create( | |||
1379 | C, DC, SourceLocation(), SourceLocation(), /*Depth=*/0, /*Position=*/0, | |||
1380 | /*Id=*/nullptr, TInfo->getType(), /*ParameterPack=*/false, TInfo); | |||
1381 | ||||
1382 | // typename ...T | |||
1383 | auto *Ts = TemplateTypeParmDecl::Create( | |||
1384 | C, DC, SourceLocation(), SourceLocation(), /*Depth=*/0, /*Position=*/1, | |||
1385 | /*Id=*/nullptr, /*Typename=*/true, /*ParameterPack=*/true, | |||
1386 | /*HasTypeConstraint=*/false); | |||
1387 | Ts->setImplicit(true); | |||
1388 | ||||
1389 | // template <std::size_t Index, typename ...T> | |||
1390 | NamedDecl *Params[] = {Index, Ts}; | |||
1391 | return TemplateParameterList::Create(C, SourceLocation(), SourceLocation(), | |||
1392 | llvm::makeArrayRef(Params), | |||
1393 | SourceLocation(), nullptr); | |||
1394 | } | |||
1395 | ||||
1396 | static TemplateParameterList *createBuiltinTemplateParameterList( | |||
1397 | const ASTContext &C, DeclContext *DC, BuiltinTemplateKind BTK) { | |||
1398 | switch (BTK) { | |||
1399 | case BTK__make_integer_seq: | |||
1400 | return createMakeIntegerSeqParameterList(C, DC); | |||
1401 | case BTK__type_pack_element: | |||
1402 | return createTypePackElementParameterList(C, DC); | |||
1403 | } | |||
1404 | ||||
1405 | llvm_unreachable("unhandled BuiltinTemplateKind!")::llvm::llvm_unreachable_internal("unhandled BuiltinTemplateKind!" , "/build/llvm-toolchain-snapshot-13~++20210506100649+6304c0836a4d/clang/lib/AST/DeclTemplate.cpp" , 1405); | |||
1406 | } | |||
1407 | ||||
1408 | void BuiltinTemplateDecl::anchor() {} | |||
1409 | ||||
1410 | BuiltinTemplateDecl::BuiltinTemplateDecl(const ASTContext &C, DeclContext *DC, | |||
1411 | DeclarationName Name, | |||
1412 | BuiltinTemplateKind BTK) | |||
1413 | : TemplateDecl(BuiltinTemplate, DC, SourceLocation(), Name, | |||
1414 | createBuiltinTemplateParameterList(C, DC, BTK)), | |||
| ||||
1415 | BTK(BTK) {} | |||
1416 | ||||
1417 | void TypeConstraint::print(llvm::raw_ostream &OS, PrintingPolicy Policy) const { | |||
1418 | if (NestedNameSpec) | |||
1419 | NestedNameSpec.getNestedNameSpecifier()->print(OS, Policy); | |||
1420 | ConceptName.printName(OS, Policy); | |||
1421 | if (hasExplicitTemplateArgs()) { | |||
1422 | OS << "<"; | |||
1423 | for (auto &ArgLoc : ArgsAsWritten->arguments()) | |||
1424 | ArgLoc.getArgument().print(Policy, OS); | |||
1425 | OS << ">"; | |||
1426 | } | |||
1427 | } | |||
1428 | ||||
1429 | TemplateParamObjectDecl *TemplateParamObjectDecl::Create(const ASTContext &C, | |||
1430 | QualType T, | |||
1431 | const APValue &V) { | |||
1432 | DeclContext *DC = C.getTranslationUnitDecl(); | |||
1433 | auto *TPOD = new (C, DC) TemplateParamObjectDecl(DC, T, V); | |||
1434 | C.addDestruction(&TPOD->Value); | |||
1435 | return TPOD; | |||
1436 | } | |||
1437 | ||||
1438 | TemplateParamObjectDecl * | |||
1439 | TemplateParamObjectDecl::CreateDeserialized(ASTContext &C, unsigned ID) { | |||
1440 | auto *TPOD = new (C, ID) TemplateParamObjectDecl(nullptr, QualType(), APValue()); | |||
1441 | C.addDestruction(&TPOD->Value); | |||
1442 | return TPOD; | |||
1443 | } | |||
1444 | ||||
1445 | void TemplateParamObjectDecl::printName(llvm::raw_ostream &OS) const { | |||
1446 | OS << "<template param "; | |||
1447 | printAsExpr(OS); | |||
1448 | OS << ">"; | |||
1449 | } | |||
1450 | ||||
1451 | void TemplateParamObjectDecl::printAsExpr(llvm::raw_ostream &OS) const { | |||
1452 | const ASTContext &Ctx = getASTContext(); | |||
1453 | getType().getUnqualifiedType().print(OS, Ctx.getPrintingPolicy()); | |||
1454 | printAsInit(OS); | |||
1455 | } | |||
1456 | ||||
1457 | void TemplateParamObjectDecl::printAsInit(llvm::raw_ostream &OS) const { | |||
1458 | const ASTContext &Ctx = getASTContext(); | |||
1459 | getValue().printPretty(OS, Ctx, getType()); | |||
1460 | } |