blob: af998f3d1ea85b1c0a716c6995b727ee5a313cdc [file] [log] [blame]
Sam McCall70942e52023-04-28 12:45:18 -07001// Part of the Crubit project, under the Apache License v2.0 with LLVM
2// Exceptions. See /LICENSE for license information.
3// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
4
Sam McCall5fc2a802023-05-02 05:41:27 -07005#include "nullability/type_nullability.h"
6
Sam McCall26940a22023-05-16 07:44:21 -07007#include <optional>
Dmitri Gribenko742c4c32023-07-31 12:32:09 -07008#include <string>
9#include <utility>
Sam McCalle644e1d2023-07-18 19:19:12 -070010#include <vector>
Sam McCall26940a22023-05-16 07:44:21 -070011
Sam McCall70942e52023-04-28 12:45:18 -070012#include "absl/log/check.h"
Sam McCall70942e52023-04-28 12:45:18 -070013#include "clang/AST/ASTContext.h"
Sam McCalla6b35b62023-06-20 01:48:38 -070014#include "clang/AST/ASTFwd.h"
Sam McCall26940a22023-05-16 07:44:21 -070015#include "clang/AST/Attr.h"
Sam McCall70942e52023-04-28 12:45:18 -070016#include "clang/AST/DeclTemplate.h"
Sam McCall26940a22023-05-16 07:44:21 -070017#include "clang/AST/Expr.h"
18#include "clang/AST/TemplateName.h"
Sam McCall70942e52023-04-28 12:45:18 -070019#include "clang/AST/Type.h"
20#include "clang/AST/TypeVisitor.h"
Sam McCalle644e1d2023-07-18 19:19:12 -070021#include "clang/Analysis/FlowSensitive/Arena.h"
Sam McCall70942e52023-04-28 12:45:18 -070022#include "clang/Basic/LLVM.h"
23#include "clang/Basic/Specifiers.h"
Dmitri Gribenko742c4c32023-07-31 12:32:09 -070024#include "llvm/ADT/STLExtras.h"
25#include "llvm/ADT/STLFunctionalExtras.h"
Sam McCall5fc2a802023-05-02 05:41:27 -070026#include "llvm/Support/SaveAndRestore.h"
Sam McCalle644e1d2023-07-18 19:19:12 -070027#include "llvm/Support/ScopedPrinter.h"
Sam McCall70942e52023-04-28 12:45:18 -070028
29namespace clang::tidy::nullability {
Sam McCall5fc2a802023-05-02 05:41:27 -070030
Martin Brænne7a8d25c2023-08-23 03:52:30 -070031bool isSupportedPointerType(QualType T) { return T->isPointerType(); }
Sam McCallb70563a2023-07-28 08:31:23 -070032
Sam McCalle644e1d2023-07-18 19:19:12 -070033PointerTypeNullability PointerTypeNullability::createSymbolic(
34 dataflow::Arena &A) {
35 PointerTypeNullability Symbolic;
36 Symbolic.Symbolic = true;
37 Symbolic.Nonnull = A.makeAtom();
38 Symbolic.Nullable = A.makeAtom();
39 return Symbolic;
40}
41
42llvm::raw_ostream &operator<<(llvm::raw_ostream &OS,
43 const PointerTypeNullability &PN) {
44 // TODO: should symbolic nullabilities have names?
45 if (PN.isSymbolic())
46 return OS << "Symbolic(Nonnull=" << PN.Nonnull << ", "
47 << "Nullable=" << PN.Nullable << ")";
48 return OS << PN.concrete();
49}
50
Sam McCall7d9afee2023-06-27 01:43:24 -070051std::string nullabilityToString(const TypeNullability &Nullability) {
Sam McCall5fc2a802023-05-02 05:41:27 -070052 std::string Result = "[";
53 llvm::interleave(
54 Nullability,
Sam McCalle644e1d2023-07-18 19:19:12 -070055 [&](const PointerTypeNullability &PN) { Result += llvm::to_string(PN); },
Sam McCall5fc2a802023-05-02 05:41:27 -070056 [&] { Result += ", "; });
57 Result += "]";
58 return Result;
59}
60
Sam McCall26940a22023-05-16 07:44:21 -070061// Recognize aliases e.g. Nonnull<T> as equivalent to T _Nonnull, etc.
62// These aliases should be annotated with [[clang::annotate("Nullable")]] etc.
63//
64// TODO: Ideally such aliases could apply the _Nonnull attribute themselves.
65// This requires resolving compatibilty issues with clang, such as use with
66// user-defined pointer-like types.
Sam McCall7d9afee2023-06-27 01:43:24 -070067std::optional<NullabilityKind> getAliasNullability(const TemplateName &TN) {
68 if (const auto *TD = TN.getAsTemplateDecl()) {
Sam McCallc18ef6b2023-06-19 01:14:21 -070069 if (!TD->getTemplatedDecl()) return std::nullopt; // BuiltinTemplateDecl
Sam McCall7d9afee2023-06-27 01:43:24 -070070 if (const auto *A = TD->getTemplatedDecl()->getAttr<AnnotateAttr>()) {
Sam McCall26940a22023-05-16 07:44:21 -070071 if (A->getAnnotation() == "Nullable") return NullabilityKind::Nullable;
72 if (A->getAnnotation() == "Nonnull") return NullabilityKind::NonNull;
73 if (A->getAnnotation() == "Nullability_Unspecified")
74 return NullabilityKind::Unspecified;
75 }
76 }
77 return std::nullopt;
78}
79
Sam McCall5fc2a802023-05-02 05:41:27 -070080namespace {
Sam McCall5fc2a802023-05-02 05:41:27 -070081// Traverses a Type to find the points where it might be nullable.
82// This will visit the contained PointerType in the correct order to produce
83// the TypeNullability vector.
84//
85// Subclasses must provide `void report(const PointerType*, NullabilityKind)`,
86// and may override TypeVisitor Visit*Type methods to customize the traversal.
87//
88// Canonically-equivalent Types produce equivalent sequences of report() calls:
89// - corresponding PointerTypes are canonically-equivalent
90// - the NullabilityKind may be different, as it derives from type sugar
91template <class Impl>
92class NullabilityWalker : public TypeVisitor<Impl> {
93 using Base = TypeVisitor<Impl>;
Sam McCall7d9afee2023-06-27 01:43:24 -070094 Impl &derived() { return *static_cast<Impl *>(this); }
Sam McCall5fc2a802023-05-02 05:41:27 -070095
96 // A nullability attribute we've seen, waiting to attach to a pointer type.
97 // There may be sugar in between: Attributed -> Typedef -> Typedef -> Pointer.
98 // All non-sugar types must consume nullability, most will ignore it.
99 std::optional<NullabilityKind> PendingNullability;
100
Sam McCall26940a22023-05-16 07:44:21 -0700101 void sawNullability(NullabilityKind NK) {
102 // If we see nullability applied twice, the outer one wins.
103 if (!PendingNullability.has_value()) PendingNullability = NK;
104 }
105
Sam McCall5fc2a802023-05-02 05:41:27 -0700106 void ignoreUnexpectedNullability() {
107 // TODO: Can we upgrade this to an assert?
108 // clang is pretty thorough about ensuring we can't put _Nullable on
109 // non-pointers, even failing template instantiation on this basis.
110 PendingNullability.reset();
111 }
112
Sam McCall901c35b2023-05-10 06:14:51 -0700113 // While walking types instantiated from templates, e.g.:
114 // - the underlying type of alias TemplateSpecializationTypes
115 // - type aliases inside class template instantiations
Sam McCall5fc2a802023-05-02 05:41:27 -0700116 // we see SubstTemplateTypeParmTypes where type parameters were referenced.
117 // The directly-available underlying types lack sugar, but we can retrieve the
Sam McCall901c35b2023-05-10 06:14:51 -0700118 // sugar from the arguments of the original e.g. TemplateSpecializationType.
Sam McCall5fc2a802023-05-02 05:41:27 -0700119 //
Sam McCall901c35b2023-05-10 06:14:51 -0700120 // The "template context" associates template params with the
121 // corresponding args, to allow this retrieval.
122 // In general, not just the directly enclosing template params but also those
123 // of outer classes are accessible.
124 // So conceptually this maps (depth, index, pack_index) => TemplateArgument.
125 // To avoid copying these maps, inner contexts *extend* from outer ones.
126 //
127 // When we start to walk a TemplateArgument (in place of a SubstTTPType), we
128 // must do so in the template instantiation context where the argument was
129 // written. Then when we're done, we must restore the old context.
130 struct TemplateContext {
131 // A decl that owns an arg list, per SubstTTPType::getAssociatedDecl.
132 // For aliases: TypeAliasTemplateDecl.
133 // For classes: ClassTemplateSpecializationDecl.
Sam McCall7d9afee2023-06-27 01:43:24 -0700134 const Decl *AssociatedDecl = nullptr;
Sam McCall901c35b2023-05-10 06:14:51 -0700135 // The sugared template arguments to AssociatedDecl, as written in the code.
Sam McCalla6b35b62023-06-20 01:48:38 -0700136 // If absent, the arguments could not be reconstructed.
137 std::optional<ArrayRef<TemplateArgument>> Args;
Sam McCall901c35b2023-05-10 06:14:51 -0700138 // In general, multiple template params are in scope (nested templates).
139 // These are a linked list: *this describes one, *Extends describes the
140 // next. In practice, this is the enclosing class template.
Sam McCall7d9afee2023-06-27 01:43:24 -0700141 const TemplateContext *Extends = nullptr;
Sam McCall901c35b2023-05-10 06:14:51 -0700142 // The template context in which the args were written.
143 // The args may reference params visible in this context.
Sam McCall7d9afee2023-06-27 01:43:24 -0700144 const TemplateContext *ArgContext = nullptr;
Sam McCall901c35b2023-05-10 06:14:51 -0700145
146 // Example showing a TemplateContext graph:
147 //
148 // // (some sugar and nested templates for the example)
149 // using INT = int; using FLOAT = float;
150 // template <class T> struct Outer {
151 // template <class U> struct Inner {
152 // using Pair = std::pair<T, U>;
153 // }
154 // }
155 //
156 // template <class X>
157 // struct S {
158 // using Type = typename Outer<INT>::Inner<X>::Pair;
159 // }
160 //
161 // using Target = S<FLOAT>::Type;
162 //
163 // Per clang's AST, instantiated Type is std::pair<int, float> with only
164 // SubstTemplateTypeParmTypes for sugar, we're trying to recover INT, FLOAT.
165 //
166 // When walking the ElaboratedType for the S<FLOAT>:: qualifier we set up:
167 //
168 // Current -> {Associated=S<float>, Args=<FLOAT>, Extends=null, ArgCtx=null}
169 //
170 // This means that when resolving ::Type:
171 // - we can resugar occurrences of X (float -> FLOAT)
172 // - ArgContext=null: the arg FLOAT may not refer to template params
173 // (or at least we can't resugar them)
174 // - Extends=null: there are no other template params we can resugar
175 //
176 // Skipping up to ::Pair inside S<FLOAT>'s instantiation, we have the graph:
177 //
178 // Current -> {Associated=Outer<int>::Inner<float>, Args=<X>}
179 // | Extends |
180 // A{Associated=Outer<int>, Args=<INT>, Extends=null} | ArgContext
181 // | ArgContext |
182 // B{Associated=S<float>, Args=<FLOAT>, Extends=null, ArgContext=null}
183 //
184 // (Note that B here is the original TemplateContext we set up above).
185 //
186 // This means that when resolving ::Pair:
187 // - we can resugar instances of U (float -> X)
188 // - ArgContext=B: when resugaring U, we can resugar X (float -> FLOAT)
189 // - Extends=A: we can also resugar T (int -> INT)
190 // - A.ArgContext=B: when resugaring T, we can resugar X.
191 // (we never do, because INT doesn't mention X)
192 // - A.Extends=null: there are no other template params te resugar
193 // - B.ArgContext=null: FLOAT may not refer to any template params
194 // - B.Extends=null: there are no other template params to resugar
195 // (e.g. Type's definition cannot refer to T)
Sam McCall5fc2a802023-05-02 05:41:27 -0700196 };
Sam McCall901c35b2023-05-10 06:14:51 -0700197 // The context that provides sugared args for the template params that are
198 // accessible to the type we're currently walking.
Sam McCall7d9afee2023-06-27 01:43:24 -0700199 const TemplateContext *CurrentTemplateContext = nullptr;
Sam McCall5fc2a802023-05-02 05:41:27 -0700200
Sam McCalla6b35b62023-06-20 01:48:38 -0700201 // Adjusts args list from those of primary template => template pattern.
202 //
203 // A template arg list corresponds 1:1 to primary template params.
204 // In partial specializations, the correspondence may differ:
205 // template <int, class> struct S;
206 // template <class T> struct S<0, T> {
207 // using Alias = T; // T refers to param #0
208 // };
209 // S<0, int*>::Alias X; // T is bound to arg #1
210 // or
211 // template <class> struct S;
212 // template <class T> struct S<T*> { using Alias = T; }
213 // S<int*>::Alias X; // arg #0 is int*, param #0 is bound to int
Sam McCall7d9afee2023-06-27 01:43:24 -0700214 void translateTemplateArgsForSpecialization(TemplateContext &Ctx) {
Sam McCalla6b35b62023-06-20 01:48:38 -0700215 // Only relevant where partial specialization is used.
216 // - Full specializations may not refer to template params at all.
217 // - For primary templates, the input is already correct.
Sam McCall7d9afee2023-06-27 01:43:24 -0700218 const TemplateArgumentList *PartialArgs = nullptr;
219 if (const ClassTemplateSpecializationDecl *CTSD =
Sam McCalla6b35b62023-06-20 01:48:38 -0700220 llvm::dyn_cast<ClassTemplateSpecializationDecl>(
221 Ctx.AssociatedDecl)) {
Dmitri Gribenko742c4c32023-07-31 12:32:09 -0700222 if (isa_and_nonnull<ClassTemplatePartialSpecializationDecl>(
Sam McCalla6b35b62023-06-20 01:48:38 -0700223 CTSD->getTemplateInstantiationPattern()))
224 PartialArgs = &CTSD->getTemplateInstantiationArgs();
Sam McCall7d9afee2023-06-27 01:43:24 -0700225 } else if (const VarTemplateSpecializationDecl *VTSD =
Sam McCalla6b35b62023-06-20 01:48:38 -0700226 llvm::dyn_cast<VarTemplateSpecializationDecl>(
227 Ctx.AssociatedDecl)) {
Dmitri Gribenko742c4c32023-07-31 12:32:09 -0700228 if (isa_and_nonnull<VarTemplatePartialSpecializationDecl>(
Sam McCalla6b35b62023-06-20 01:48:38 -0700229 VTSD->getTemplateInstantiationPattern()))
230 PartialArgs = &VTSD->getTemplateInstantiationArgs();
231 }
232 if (!PartialArgs) return;
233
234 // To get from the template arg list to the partial-specialization arg list
235 // means running much of the template argument deduction algorithm.
236 // This is complex in general. [temp.deduct] For now, bail out.
237 // In future, hopefully we can handle at least simple cases.
238 Ctx.Args.reset();
239 }
240
Sam McCall5fc2a802023-05-02 05:41:27 -0700241 public:
242 void Visit(QualType T) { Base::Visit(T.getTypePtr()); }
Sam McCall7d9afee2023-06-27 01:43:24 -0700243 void Visit(const TemplateArgument &TA) {
Sam McCall5fc2a802023-05-02 05:41:27 -0700244 if (TA.getKind() == TemplateArgument::Type) Visit(TA.getAsType());
245 if (TA.getKind() == TemplateArgument::Pack)
Sam McCall7d9afee2023-06-27 01:43:24 -0700246 for (const auto &PackElt : TA.getPackAsArray()) Visit(PackElt);
Sam McCall5fc2a802023-05-02 05:41:27 -0700247 }
Sam McCall7d9afee2023-06-27 01:43:24 -0700248 void Visit(const DeclContext *DC) {
Sam McCall5fc2a802023-05-02 05:41:27 -0700249 // For now, only consider enclosing classes.
250 // TODO: The nullability of template functions can affect local classes too,
251 // this can be relevant e.g. when instantiating templates with such types.
Dmitri Gribenko742c4c32023-07-31 12:32:09 -0700252 if (auto *CRD = dyn_cast<CXXRecordDecl>(DC))
Sam McCall5fc2a802023-05-02 05:41:27 -0700253 Visit(DC->getParentASTContext().getRecordType(CRD));
254 }
255
Sam McCall7d9afee2023-06-27 01:43:24 -0700256 void VisitType(const Type *T) {
Sam McCall5fc2a802023-05-02 05:41:27 -0700257 // For sugar not explicitly handled below, desugar and continue.
258 // (We need to walk the full structure of the canonical type.)
Sam McCall7d9afee2023-06-27 01:43:24 -0700259 if (auto *Desugar =
Sam McCall5fc2a802023-05-02 05:41:27 -0700260 T->getLocallyUnqualifiedSingleStepDesugaredType().getTypePtr();
261 Desugar != T)
262 return Base::Visit(Desugar);
263
264 // We don't expect to see any nullable non-sugar types except PointerType.
265 ignoreUnexpectedNullability();
266 Base::VisitType(T);
267 }
268
Sam McCall7d9afee2023-06-27 01:43:24 -0700269 void VisitFunctionProtoType(const FunctionProtoType *FPT) {
Sam McCall5fc2a802023-05-02 05:41:27 -0700270 ignoreUnexpectedNullability();
271 Visit(FPT->getReturnType());
272 for (auto ParamType : FPT->getParamTypes()) Visit(ParamType);
273 }
274
Sam McCall7d9afee2023-06-27 01:43:24 -0700275 void VisitTemplateSpecializationType(const TemplateSpecializationType *TST) {
Sam McCall5fc2a802023-05-02 05:41:27 -0700276 if (TST->isTypeAlias()) {
Sam McCall26940a22023-05-16 07:44:21 -0700277 if (auto NK = getAliasNullability(TST->getTemplateName()))
278 sawNullability(*NK);
279
Sam McCall5fc2a802023-05-02 05:41:27 -0700280 // Aliases are sugar, visit the underlying type.
281 // Record template args so we can resugar substituted params.
Martin Brænnee8a33622023-05-08 02:04:59 -0700282 //
283 // TODO(b/281474380): `TemplateSpecializationType::template_arguments()`
284 // doesn't contain defaulted arguments. Can we fetch or compute these in
285 // sugared form?
Sam McCall901c35b2023-05-10 06:14:51 -0700286 const TemplateContext Ctx{
287 /*AssociatedDecl=*/TST->getTemplateName().getAsTemplateDecl(),
288 /*Args=*/TST->template_arguments(),
289 /*Extends=*/CurrentTemplateContext,
290 /*ArgContext=*/CurrentTemplateContext,
291 };
292 llvm::SaveAndRestore UseAlias(CurrentTemplateContext, &Ctx);
Sam McCall5fc2a802023-05-02 05:41:27 -0700293 VisitType(TST);
294 return;
295 }
296
Sam McCall7d9afee2023-06-27 01:43:24 -0700297 auto *CRD = TST->getAsCXXRecordDecl();
Sam McCall5fc2a802023-05-02 05:41:27 -0700298 CHECK(CRD) << "Expected an alias or class specialization in concrete code";
299 ignoreUnexpectedNullability();
300 Visit(CRD->getDeclContext());
301 for (auto TA : TST->template_arguments()) Visit(TA);
Martin Brænnea354fb72023-06-26 06:16:22 -0700302 // `TST->template_arguments()` doesn't contain any default arguments.
303 // Retrieve these (though in unsugared form) from the
304 // `ClassTemplateSpecializationDecl`.
305 // TODO(b/281474380): Can we fetch or compute default arguments in sugared
306 // form?
Sam McCall7d9afee2023-06-27 01:43:24 -0700307 if (auto *CTSD = dyn_cast<ClassTemplateSpecializationDecl>(CRD)) {
Martin Brænnea354fb72023-06-26 06:16:22 -0700308 for (unsigned i = TST->template_arguments().size();
309 i < CTSD->getTemplateArgs().size(); ++i)
310 Visit(CTSD->getTemplateArgs()[i]);
311 }
Sam McCall5fc2a802023-05-02 05:41:27 -0700312 }
313
Sam McCall7d9afee2023-06-27 01:43:24 -0700314 void VisitSubstTemplateTypeParmType(const SubstTemplateTypeParmType *T) {
Sam McCall901c35b2023-05-10 06:14:51 -0700315 // The underlying type of T in the AST has no sugar, as the template has
316 // only one body instantiated per canonical args.
317 // Instead, try to find the (sugared) template argument that T is bound to.
Sam McCall7d9afee2023-06-27 01:43:24 -0700318 for (const auto *Ctx = CurrentTemplateContext; Ctx; Ctx = Ctx->Extends) {
Sam McCall901c35b2023-05-10 06:14:51 -0700319 if (T->getAssociatedDecl() != Ctx->AssociatedDecl) continue;
Sam McCalla6b35b62023-06-20 01:48:38 -0700320 // If args are not available, fall back to un-sugared arg.
321 if (!Ctx->Args.has_value()) break;
Sam McCall901c35b2023-05-10 06:14:51 -0700322 unsigned Index = T->getIndex();
323 // Valid because pack must be the last param in non-function templates.
324 // TODO: if we support function templates, we need to be smarter here.
325 if (auto PackIndex = T->getPackIndex())
Sam McCalla6b35b62023-06-20 01:48:38 -0700326 Index = Ctx->Args->size() - 1 - *PackIndex;
Sam McCall5fc2a802023-05-02 05:41:27 -0700327
Sam McCall901c35b2023-05-10 06:14:51 -0700328 // TODO(b/281474380): `Args` may be too short if `Index` refers to an
329 // arg that was defaulted. We eventually want to populate
330 // `CurrentAliasTemplate->Args` with the default arguments in this case,
331 // but for now, we just walk the underlying type without sugar.
Sam McCalla6b35b62023-06-20 01:48:38 -0700332 if (Index < Ctx->Args->size()) {
Sam McCall7d9afee2023-06-27 01:43:24 -0700333 const TemplateArgument &Arg = (*Ctx->Args)[Index];
Sam McCall901c35b2023-05-10 06:14:51 -0700334 // When we start to walk a sugared TemplateArgument (in place of T),
335 // we must do so in the template instantiation context where the
336 // argument was written.
337 llvm::SaveAndRestore OriginalContext(
338 CurrentTemplateContext, CurrentTemplateContext->ArgContext);
339 return Visit(Arg);
Sam McCall5fc2a802023-05-02 05:41:27 -0700340 }
341 }
Sam McCall901c35b2023-05-10 06:14:51 -0700342 // Our top-level type references an unbound type param.
343 // Our original input was the underlying type of an instantiation, we
344 // lack the context needed to resugar it.
345 // TODO: maybe this could be an assert in some cases (alias params)?
346 // We would need to trust all callers are obtaining types appropriately,
347 // and that clang never partially-desugars in a problematic way.
Sam McCall5fc2a802023-05-02 05:41:27 -0700348 VisitType(T);
349 }
350
Sam McCall901c35b2023-05-10 06:14:51 -0700351 // If we see foo<args>::ty then we may need sugar from args to resugar ty.
Sam McCall7d9afee2023-06-27 01:43:24 -0700352 void VisitElaboratedType(const ElaboratedType *ET) {
Sam McCall901c35b2023-05-10 06:14:51 -0700353 std::vector<TemplateContext> BoundTemplateArgs;
354 // Iterate over qualifiers right-to-left, looking for template args.
Sam McCall7d9afee2023-06-27 01:43:24 -0700355 for (auto *NNS = ET->getQualifier(); NNS; NNS = NNS->getPrefix()) {
Sam McCall901c35b2023-05-10 06:14:51 -0700356 // TODO: there are other ways a NNS could bind template args:
357 // template <typename T> foo { struct bar { using baz = T; }; };
358 // using T = foo<int * _Nullable>::bar;
359 // using U = T::baz;
360 // Here T:: is not a TemplateSpecializationType (directly or indirectly).
361 // Nevertheless it provides sugar that is referenced from baz.
362 // Probably we need another type visitor to collect bindings in general.
Dmitri Gribenko742c4c32023-07-31 12:32:09 -0700363 if (const auto *TST =
364 dyn_cast_or_null<TemplateSpecializationType>(NNS->getAsType())) {
Sam McCall901c35b2023-05-10 06:14:51 -0700365 TemplateContext Ctx;
366 Ctx.Args = TST->template_arguments();
367 Ctx.ArgContext = CurrentTemplateContext;
368 // `Extends` is initialized below: we chain BoundTemplateArgs together.
369 Ctx.AssociatedDecl =
370 TST->isTypeAlias() ? TST->getTemplateName().getAsTemplateDecl()
Sam McCall7d9afee2023-06-27 01:43:24 -0700371 : static_cast<Decl *>(TST->getAsCXXRecordDecl());
Sam McCalla6b35b62023-06-20 01:48:38 -0700372 translateTemplateArgsForSpecialization(Ctx);
Sam McCall901c35b2023-05-10 06:14:51 -0700373 BoundTemplateArgs.push_back(Ctx);
374 }
375 }
Sam McCall7d9afee2023-06-27 01:43:24 -0700376 std::optional<llvm::SaveAndRestore<const TemplateContext *>> Restore;
Sam McCall901c35b2023-05-10 06:14:51 -0700377 if (!BoundTemplateArgs.empty()) {
378 // Wire up the inheritance chain so all the contexts are visible.
379 BoundTemplateArgs.back().Extends = CurrentTemplateContext;
380 for (int I = 0; I < BoundTemplateArgs.size() - 1; ++I)
381 BoundTemplateArgs[I].Extends = &BoundTemplateArgs[I + 1];
382 Restore.emplace(CurrentTemplateContext, &BoundTemplateArgs.front());
383 }
384 Visit(ET->getNamedType());
385 }
386
Sam McCall7d9afee2023-06-27 01:43:24 -0700387 void VisitRecordType(const RecordType *RT) {
Sam McCall5fc2a802023-05-02 05:41:27 -0700388 ignoreUnexpectedNullability();
389 Visit(RT->getDecl()->getDeclContext());
Sam McCall7d9afee2023-06-27 01:43:24 -0700390 if (auto *CTSD = dyn_cast<ClassTemplateSpecializationDecl>(RT->getDecl())) {
Sam McCall5fc2a802023-05-02 05:41:27 -0700391 // TODO: if this is an instantiation, these args lack sugar.
392 // We can try to retrieve it from the current template context.
Sam McCall7d9afee2023-06-27 01:43:24 -0700393 for (auto &TA : CTSD->getTemplateArgs().asArray()) Visit(TA);
Sam McCall5fc2a802023-05-02 05:41:27 -0700394 }
395 }
396
Sam McCall7d9afee2023-06-27 01:43:24 -0700397 void VisitAttributedType(const AttributedType *AT) {
Sam McCall26940a22023-05-16 07:44:21 -0700398 if (auto NK = AT->getImmediateNullability()) sawNullability(*NK);
Sam McCall5fc2a802023-05-02 05:41:27 -0700399 Visit(AT->getModifiedType());
400 CHECK(!PendingNullability.has_value())
401 << "Should have been consumed by modified type! "
402 << AT->getModifiedType().getAsString();
403 }
404
Sam McCall7d9afee2023-06-27 01:43:24 -0700405 void VisitPointerType(const PointerType *PT) {
Sam McCall5fc2a802023-05-02 05:41:27 -0700406 derived().report(PT,
407 PendingNullability.value_or(NullabilityKind::Unspecified));
408 PendingNullability.reset();
409 Visit(PT->getPointeeType());
410 }
Sam McCall9a893de2023-05-02 06:06:55 -0700411
Sam McCall7d9afee2023-06-27 01:43:24 -0700412 void VisitReferenceType(const ReferenceType *RT) {
Sam McCall9a893de2023-05-02 06:06:55 -0700413 ignoreUnexpectedNullability();
414 Visit(RT->getPointeeTypeAsWritten());
415 }
Sam McCallf2b62b32023-05-02 06:45:40 -0700416
Sam McCall7d9afee2023-06-27 01:43:24 -0700417 void VisitArrayType(const ArrayType *AT) {
Sam McCallf2b62b32023-05-02 06:45:40 -0700418 ignoreUnexpectedNullability();
419 Visit(AT->getElementType());
420 }
Sam McCall5fc2a802023-05-02 05:41:27 -0700421};
422
423template <typename T>
Sam McCall7d9afee2023-06-27 01:43:24 -0700424unsigned countPointers(const T &Object) {
Sam McCall5fc2a802023-05-02 05:41:27 -0700425 struct Walker : public NullabilityWalker<Walker> {
426 unsigned Count = 0;
Sam McCall7d9afee2023-06-27 01:43:24 -0700427 void report(const PointerType *, NullabilityKind) { ++Count; }
Sam McCall5fc2a802023-05-02 05:41:27 -0700428 } PointerCountWalker;
429 PointerCountWalker.Visit(Object);
430 return PointerCountWalker.Count;
431}
432
433} // namespace
434
435unsigned countPointersInType(QualType T) { return countPointers(T); }
436
Sam McCall7d9afee2023-06-27 01:43:24 -0700437unsigned countPointersInType(const DeclContext *DC) {
Sam McCall5fc2a802023-05-02 05:41:27 -0700438 return countPointers(DC);
439}
440unsigned countPointersInType(TemplateArgument TA) { return countPointers(TA); }
441
Sam McCall7d9afee2023-06-27 01:43:24 -0700442QualType exprType(const Expr *E) {
Sam McCall5fc2a802023-05-02 05:41:27 -0700443 if (E->hasPlaceholderType(BuiltinType::BoundMember))
444 return Expr::findBoundMemberType(E);
445 return E->getType();
446}
447
Sam McCall7d9afee2023-06-27 01:43:24 -0700448unsigned countPointersInType(const Expr *E) {
Sam McCall5fc2a802023-05-02 05:41:27 -0700449 return countPointersInType(exprType(E));
450}
451
Sam McCalld127f932023-05-02 07:15:27 -0700452TypeNullability getNullabilityAnnotationsFromType(
Sam McCall5fc2a802023-05-02 05:41:27 -0700453 QualType T,
454 llvm::function_ref<GetTypeParamNullability> SubstituteTypeParam) {
Sam McCalldba68972023-07-19 11:01:10 -0700455 CHECK(!T->isDependentType()) << T.getAsString();
Sam McCall5fc2a802023-05-02 05:41:27 -0700456 struct Walker : NullabilityWalker<Walker> {
Sam McCalle644e1d2023-07-18 19:19:12 -0700457 std::vector<PointerTypeNullability> Annotations;
Sam McCall5fc2a802023-05-02 05:41:27 -0700458 llvm::function_ref<GetTypeParamNullability> SubstituteTypeParam;
459
Sam McCall7d9afee2023-06-27 01:43:24 -0700460 void report(const PointerType *, NullabilityKind NK) {
Sam McCall5fc2a802023-05-02 05:41:27 -0700461 Annotations.push_back(NK);
462 }
463
Sam McCall7d9afee2023-06-27 01:43:24 -0700464 void VisitSubstTemplateTypeParmType(const SubstTemplateTypeParmType *ST) {
Sam McCall5fc2a802023-05-02 05:41:27 -0700465 if (SubstituteTypeParam) {
466 if (auto Subst = SubstituteTypeParam(ST)) {
467 DCHECK_EQ(Subst->size(),
468 countPointersInType(ST->getCanonicalTypeInternal()))
469 << "Substituted nullability has the wrong structure: "
470 << QualType(ST, 0).getAsString();
471 llvm::append_range(Annotations, *Subst);
472 return;
473 }
474 }
475 NullabilityWalker::VisitSubstTemplateTypeParmType(ST);
476 }
477 } AnnotationVisitor;
478 AnnotationVisitor.SubstituteTypeParam = SubstituteTypeParam;
479 AnnotationVisitor.Visit(T);
480 return std::move(AnnotationVisitor.Annotations);
481}
482
Sam McCall7d9afee2023-06-27 01:43:24 -0700483TypeNullability unspecifiedNullability(const Expr *E) {
Sam McCalld127f932023-05-02 07:15:27 -0700484 return TypeNullability(countPointersInType(E), NullabilityKind::Unspecified);
Sam McCall5fc2a802023-05-02 05:41:27 -0700485}
486
Sam McCall70942e52023-04-28 12:45:18 -0700487namespace {
488
489// Visitor to rebuild a QualType with explicit nullability.
490// Extra AttributedType nodes are added wrapping interior PointerTypes, and
491// other sugar is added as needed to allow this (e.g. TypeSpecializationType).
492//
493// We only have to handle types that have nontrivial nullability vectors, i.e.
494// those handled by NullabilityWalker.
495// Additionally, we only operate on canonical types (otherwise the sugar we're
496// adding could conflict with existing sugar).
497//
498// This needs to stay in sync with the other algorithms that manipulate
499// nullability data structures for particular types: the non-flow-sensitive
500// transfer and NullabilityWalker.
Sam McCall5fc2a802023-05-02 05:41:27 -0700501struct Rebuilder : public TypeVisitor<Rebuilder, QualType> {
Sam McCall7d9afee2023-06-27 01:43:24 -0700502 Rebuilder(const TypeNullability &Nullability, ASTContext &Ctx)
Sam McCall70942e52023-04-28 12:45:18 -0700503 : Nullability(Nullability), Ctx(Ctx) {}
504
505 bool done() const { return Nullability.empty(); }
506
Sam McCall5fc2a802023-05-02 05:41:27 -0700507 using Base = TypeVisitor<Rebuilder, QualType>;
Sam McCall70942e52023-04-28 12:45:18 -0700508 using Base::Visit;
509 QualType Visit(QualType T) {
510 if (T.isNull()) return T;
511 return Ctx.getQualifiedType(Visit(T.getTypePtr()), T.getLocalQualifiers());
512 }
513 TemplateArgument Visit(TemplateArgument TA) {
514 if (TA.getKind() == TemplateArgument::Type)
515 return TemplateArgument(Visit(TA.getAsType()));
516 return TA;
517 }
518
519 // Default behavior for unhandled types: do not transform.
Sam McCall7d9afee2023-06-27 01:43:24 -0700520 QualType VisitType(const Type *T) { return QualType(T, 0); }
Sam McCall70942e52023-04-28 12:45:18 -0700521
Sam McCall7d9afee2023-06-27 01:43:24 -0700522 QualType VisitPointerType(const PointerType *PT) {
Sam McCall70942e52023-04-28 12:45:18 -0700523 CHECK(!Nullability.empty())
524 << "Nullability vector too short at " << QualType(PT, 0).getAsString();
Sam McCalle644e1d2023-07-18 19:19:12 -0700525 NullabilityKind NK = Nullability.front().concrete();
Sam McCall70942e52023-04-28 12:45:18 -0700526 Nullability = Nullability.drop_front();
527
528 QualType Rebuilt = Ctx.getPointerType(Visit(PT->getPointeeType()));
529 if (NK == NullabilityKind::Unspecified) return Rebuilt;
530 return Ctx.getAttributedType(AttributedType::getNullabilityAttrKind(NK),
531 Rebuilt, Rebuilt);
532 }
533
Sam McCall7d9afee2023-06-27 01:43:24 -0700534 QualType VisitRecordType(const RecordType *RT) {
535 if (const auto *CTSD =
Sam McCall70942e52023-04-28 12:45:18 -0700536 dyn_cast<ClassTemplateSpecializationDecl>(RT->getDecl())) {
537 std::vector<TemplateArgument> TransformedArgs;
Sam McCall7d9afee2023-06-27 01:43:24 -0700538 for (const auto &Arg : CTSD->getTemplateArgs().asArray())
Sam McCall70942e52023-04-28 12:45:18 -0700539 TransformedArgs.push_back(Visit(Arg));
540 return Ctx.getTemplateSpecializationType(
541 TemplateName(CTSD->getSpecializedTemplate()), TransformedArgs,
542 QualType(RT, 0));
543 }
544 return QualType(RT, 0);
545 }
546
Sam McCall7d9afee2023-06-27 01:43:24 -0700547 QualType VisitFunctionProtoType(const FunctionProtoType *T) {
Sam McCall70942e52023-04-28 12:45:18 -0700548 QualType Ret = Visit(T->getReturnType());
549 std::vector<QualType> Params;
Sam McCall7d9afee2023-06-27 01:43:24 -0700550 for (const auto &Param : T->getParamTypes()) Params.push_back(Visit(Param));
Sam McCall70942e52023-04-28 12:45:18 -0700551 return Ctx.getFunctionType(Ret, Params, T->getExtProtoInfo());
552 }
553
Sam McCall7d9afee2023-06-27 01:43:24 -0700554 QualType VisitLValueReferenceType(const LValueReferenceType *T) {
Sam McCall9a893de2023-05-02 06:06:55 -0700555 return Ctx.getLValueReferenceType(Visit(T->getPointeeType()));
556 }
Sam McCall7d9afee2023-06-27 01:43:24 -0700557 QualType VisitRValueReferenceType(const RValueReferenceType *T) {
Sam McCall9a893de2023-05-02 06:06:55 -0700558 return Ctx.getRValueReferenceType(Visit(T->getPointeeType()));
559 }
560
Sam McCall7d9afee2023-06-27 01:43:24 -0700561 QualType VisitConstantArrayType(const ConstantArrayType *AT) {
Sam McCallf2b62b32023-05-02 06:45:40 -0700562 return Ctx.getConstantArrayType(Visit(AT->getElementType()), AT->getSize(),
563 AT->getSizeExpr(), AT->getSizeModifier(),
564 AT->getIndexTypeCVRQualifiers());
565 }
Sam McCall7d9afee2023-06-27 01:43:24 -0700566 QualType VisitIncompleteArrayType(const IncompleteArrayType *AT) {
Sam McCallf2b62b32023-05-02 06:45:40 -0700567 return Ctx.getIncompleteArrayType(Visit(AT->getElementType()),
568 AT->getSizeModifier(),
569 AT->getIndexTypeCVRQualifiers());
570 }
Sam McCall7d9afee2023-06-27 01:43:24 -0700571 QualType VisitVariableArrayType(const VariableArrayType *AT) {
Sam McCallf2b62b32023-05-02 06:45:40 -0700572 return Ctx.getVariableArrayType(
573 Visit(AT->getElementType()), AT->getSizeExpr(), AT->getSizeModifier(),
574 AT->getIndexTypeCVRQualifiers(), AT->getBracketsRange());
575 }
576
Sam McCall70942e52023-04-28 12:45:18 -0700577 private:
Sam McCalle644e1d2023-07-18 19:19:12 -0700578 ArrayRef<PointerTypeNullability> Nullability;
Sam McCall7d9afee2023-06-27 01:43:24 -0700579 ASTContext &Ctx;
Sam McCall70942e52023-04-28 12:45:18 -0700580};
581
582} // namespace
583
Sam McCall7d9afee2023-06-27 01:43:24 -0700584QualType rebuildWithNullability(QualType T, const TypeNullability &Nullability,
585 ASTContext &Ctx) {
Sam McCall5fc2a802023-05-02 05:41:27 -0700586 Rebuilder V(Nullability, Ctx);
Sam McCall70942e52023-04-28 12:45:18 -0700587 QualType Result = V.Visit(T.getCanonicalType());
588 CHECK(V.done()) << "Nullability vector[" << Nullability.size()
589 << "] too long for " << T.getAsString();
590 return Result;
591}
592
Sam McCall7d9afee2023-06-27 01:43:24 -0700593std::string printWithNullability(QualType T, const TypeNullability &Nullability,
594 ASTContext &Ctx) {
Sam McCall70942e52023-04-28 12:45:18 -0700595 return rebuildWithNullability(T, Nullability, Ctx)
596 .getAsString(Ctx.getPrintingPolicy());
597}
598
599} // namespace clang::tidy::nullability