| // Part of the Crubit project, under the Apache License v2.0 with LLVM |
| // Exceptions. See /LICENSE for license information. |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| |
| #include "nullability/inference/eligible_ranges.h" |
| |
| #include <optional> |
| #include <string> |
| #include <vector> |
| |
| #include "absl/log/check.h" |
| #include "nullability/inference/augmented_test_inputs.h" |
| #include "nullability/inference/inference.proto.h" |
| #include "nullability/pragma.h" |
| #include "nullability/type_nullability.h" |
| #include "clang/AST/Decl.h" |
| #include "clang/ASTMatchers/ASTMatchFinder.h" |
| #include "clang/ASTMatchers/ASTMatchers.h" |
| #include "clang/Basic/LLVM.h" |
| #include "clang/Testing/TestAST.h" |
| #include "llvm/ADT/StringRef.h" |
| #include "llvm/Testing/Annotations/Annotations.h" |
| #include "third_party/llvm/llvm-project/third-party/unittest/googlemock/include/gmock/gmock.h" // IWYU pragma: keep |
| #include "third_party/llvm/llvm-project/third-party/unittest/googletest/include/gtest/gtest.h" |
| |
| namespace clang::tidy::nullability { |
| namespace { |
| using ::clang::ast_matchers::fieldDecl; |
| using ::clang::ast_matchers::functionDecl; |
| using ::clang::ast_matchers::hasName; |
| using ::clang::ast_matchers::match; |
| using ::clang::ast_matchers::selectFirst; |
| using ::clang::ast_matchers::varDecl; |
| using ::llvm::Annotations; |
| using ::testing::AllOf; |
| using ::testing::ExplainMatchResult; |
| using ::testing::Optional; |
| using ::testing::Pointwise; |
| using ::testing::UnorderedElementsAre; |
| |
| test::EnableSmartPointers Enable; |
| |
| constexpr char MainFileName[] = "input.cc"; |
| |
| MATCHER_P2(SlotRange, SlotID, Range, |
| absl::StrCat("is a SlotRange with ID ", SlotID, |
| " and range equivalent to [", Range.Begin, ",", |
| Range.End, ")")) { |
| return ((SlotID == -1 && !arg.has_slot()) || arg.slot() == SlotID) && |
| Range.Begin == arg.begin() && Range.End == arg.end(); |
| } |
| |
| MATCHER_P2(SlotRangeWithNoExistingAnnotation, SlotID, Range, "") { |
| return !arg.has_existing_annotation() && |
| ExplainMatchResult(SlotRange(SlotID, Range), arg, result_listener); |
| } |
| |
| MATCHER_P3(SlotRange, SlotID, Range, ExistingAnnotation, |
| absl::StrCat("is a SlotRange with ID ", SlotID, |
| " and range equivalent to [", Range.Begin, ",", |
| Range.End, ") and existing annotation ", |
| ExistingAnnotation)) { |
| return ExplainMatchResult(SlotRange(SlotID, Range), arg, result_listener) && |
| arg.has_existing_annotation() && |
| arg.existing_annotation() == ExistingAnnotation; |
| } |
| |
| MATCHER_P2(TypeLocRanges, Path, Ranges, "") { |
| return ExplainMatchResult(Path, arg.path(), result_listener) && |
| ExplainMatchResult(Ranges, arg.range(), result_listener); |
| } |
| |
| MATCHER_P2(TypeLocRangesWithNoPragmaNullability, Path, Ranges, "") { |
| return !arg.has_pragma_nullability() && |
| ExplainMatchResult(TypeLocRanges(Path, Ranges), arg, result_listener); |
| } |
| |
| MATCHER_P3(TypeLocRanges, Path, Ranges, PragmaNullability, "") { |
| return ExplainMatchResult(Path, arg.path(), result_listener) && |
| ExplainMatchResult(Ranges, arg.range(), result_listener) && |
| ExplainMatchResult(PragmaNullability, arg.pragma_nullability(), |
| result_listener); |
| } |
| |
| template <typename DeclT, typename MatcherT> |
| std::optional<clang::tidy::nullability::TypeLocRanges> getRanges( |
| llvm::StringRef Input, MatcherT Matcher) { |
| NullabilityPragmas Pragmas; |
| TestAST TU(getAugmentedTestInputs(Input, Pragmas)); |
| const auto *D = |
| selectFirst<DeclT>("d", match(Matcher.bind("d"), TU.context())); |
| CHECK(D != nullptr); |
| return clang::tidy::nullability::getEligibleRanges( |
| *D, TypeNullabilityDefaults(TU.context(), Pragmas)); |
| } |
| |
| std::optional<clang::tidy::nullability::TypeLocRanges> getFunctionRanges( |
| llvm::StringRef Input, llvm::StringRef FunctionName = "target") { |
| return getRanges<FunctionDecl>(Input, functionDecl(hasName(FunctionName))); |
| } |
| |
| std::optional<clang::tidy::nullability::TypeLocRanges> getFieldRanges( |
| llvm::StringRef Input, llvm::StringRef FieldName = "target") { |
| return getRanges<FieldDecl>( |
| Input, FieldName.empty() ? fieldDecl() : fieldDecl(hasName(FieldName))); |
| } |
| |
| std::optional<clang::tidy::nullability::TypeLocRanges> getVarRanges( |
| llvm::StringRef Input, llvm::StringRef VarName = "target") { |
| return getRanges<VarDecl>(Input, varDecl(hasName(VarName))); |
| } |
| |
| TEST(EligibleRangesTest, ReturnAndOneParameterIdentified) { |
| auto Input = Annotations("$r[[int *]]target($p[[int *]]p) { return p; }"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre( |
| SlotRangeWithNoExistingAnnotation(0, Input.range("r")), |
| SlotRangeWithNoExistingAnnotation(1, Input.range("p")))))); |
| } |
| |
| TEST(EligibleRangesTest, OnlyFirstParameterIdentified) { |
| auto Input = Annotations("void target([[int *]]p1, int p2) { return; }"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, UnorderedElementsAre(SlotRange(1, Input.range()))))); |
| } |
| |
| // Checks that a function decl without a body is handled correctly. |
| TEST(EligibleRangesTest, DeclHandled) { |
| auto Input = Annotations("void target([[int *]]p1, int p2);"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, UnorderedElementsAre(SlotRange(1, Input.range()))))); |
| } |
| |
| TEST(EligibleRangesTest, AllNestedPointersEligible) { |
| auto Input = |
| Annotations("void target($three[[$two[[$one[[int *]]*]]*]]p1, int p2);"); |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(SlotRange(-1, Input.range("one")), |
| SlotRange(-1, Input.range("two")), |
| SlotRange(1, Input.range("three")))))); |
| } |
| |
| TEST(EligibleRangesTest, DeclConstExcluded) { |
| auto Input = Annotations(R"( |
| void target($one[[int *]] const p1, |
| $two_o[[$two_i[[int *]] const *]] const p2); |
| )"); |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(SlotRange(1, Input.range("one")), |
| SlotRange(2, Input.range("two_o")), |
| SlotRange(-1, Input.range("two_i")))))); |
| } |
| |
| TEST(EligibleRangesTest, PointeeConstIncluded) { |
| auto Input = Annotations(R"( |
| void target([[const int *]]p); |
| )"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, UnorderedElementsAre(SlotRange(1, Input.range()))))); |
| } |
| |
| TEST(EligibleRangesTest, NestedPointeeConstIncluded) { |
| auto Input = Annotations("void target($o[[$i[[const int *]] const *]]p);"); |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(SlotRange(1, Input.range("o")), |
| SlotRange(-1, Input.range("i")))))); |
| } |
| |
| TEST(EligibleRangesTest, AnnotatedSlotsGetRangesForPointerTypeOnly) { |
| auto Input = Annotations(R"( |
| void target(Nonnull<$one[[int *]]> nonnull, |
| Nullable<$two[[int *]]> nullable, |
| NullabilityUnknown<$three[[int *]]> unknown); |
| )"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre( |
| SlotRange(1, Input.range("one"), Nullability::NONNULL), |
| SlotRange(2, Input.range("two"), Nullability::NULLABLE), |
| SlotRange(3, Input.range("three"), Nullability::UNKNOWN))))); |
| } |
| |
| TEST(EligibleRangesTest, NamespacedAliasAnnotatedSlotsGetNoRange) { |
| auto Input = Annotations(R"( |
| namespace custom { |
| template <typename T> |
| using CustomNonnull = Nonnull<T>; |
| template <typename T> |
| using CustomNullable = Nullable<T>; |
| template <typename T> |
| using CustomUnknown = NullabilityUnknown<T>; |
| } |
| |
| // Note also that these custom annotations are aliases for the nullability |
| // annotations, not themselves annotated. Aliases of any depth for a |
| // nullability annotation are considered an annotation. |
| void target(custom::CustomNonnull<$one[[int *]]> nonnull, |
| custom::CustomNullable<$two[[int *]]> nullable, |
| custom::CustomUnknown<$three[[int *]]> unknown); |
| )"); |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(SlotRange(1, Input.range("one")), |
| SlotRange(2, Input.range("two")), |
| SlotRange(3, Input.range("three")))))); |
| } |
| |
| TEST(EligibleRangesTest, NestedAnnotationsGetOneRange) { |
| auto Input = Annotations(R"(void target(Nonnull<Nonnull<[[int *]]>> a);)"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, UnorderedElementsAre(SlotRange(1, Input.range()))))); |
| } |
| |
| TEST(EligibleRangesTest, NestedPointersOuterAnnotated) { |
| auto Input = Annotations(R"( |
| namespace std { |
| template <typename T> |
| class unique_ptr; |
| } |
| void target( |
| Nonnull<$one_o[[$one_i[[int *]]*]]> p, |
| Nonnull<$two_o[[std::unique_ptr<$two_i[[int*]]>]]> q, |
| Nonnull<$three_o[[$three_i[[std::unique_ptr<int>]]*]]> r, |
| Nonnull<$four_o[[std::unique_ptr<$four_i[[std::unique_ptr<int>]]>]]> s); |
| )"); |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(SlotRange(1, Input.range("one_o")), |
| SlotRange(-1, Input.range("one_i")), |
| SlotRange(2, Input.range("two_o")), |
| SlotRange(-1, Input.range("two_i")), |
| SlotRange(3, Input.range("three_o")), |
| SlotRange(-1, Input.range("three_i")), |
| SlotRange(4, Input.range("four_o")), |
| SlotRange(-1, Input.range("four_i")))))); |
| } |
| |
| TEST(EligibleRangesTest, NestedPointersInnerAnnotated) { |
| auto Input = Annotations(R"( |
| namespace std { |
| template <typename T> |
| class unique_ptr; |
| } |
| |
| void target( |
| $one_o[[Nonnull<$one_i[[int *]]>*]] p, |
| $two_o[[std::unique_ptr<Nonnull<$two_i[[int*]]>>]] q, |
| $three_o[[Nonnull<$three_i[[std::unique_ptr<int>]]>*]] r, |
| $four_o[[std::unique_ptr<Nonnull<$four_i[[std::unique_ptr<int>]]>>]] s); |
| )"); |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(SlotRange(1, Input.range("one_o")), |
| SlotRange(-1, Input.range("one_i")), |
| SlotRange(2, Input.range("two_o")), |
| SlotRange(-1, Input.range("two_i")), |
| SlotRange(3, Input.range("three_o")), |
| SlotRange(-1, Input.range("three_i")), |
| SlotRange(4, Input.range("four_o")), |
| SlotRange(-1, Input.range("four_i")))))); |
| } |
| |
| TEST(EligibleRangesTest, RefToPointer) { |
| auto Input = Annotations("void target([[int *]]&p);"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, UnorderedElementsAre(SlotRange(1, Input.range()))))); |
| } |
| |
| TEST(EligibleRangesTest, TemplateOfPointers) { |
| auto Input = Annotations(R"( |
| template <typename One, typename Two> |
| struct S {}; |
| |
| void target(S<$one[[int *]], $two[[$two_inner[[bool *]]*]]> p); |
| )"); |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, UnorderedElementsAre( |
| SlotRange(-1, Input.range("one")), |
| SlotRange(-1, Input.range("two")), |
| SlotRange(-1, Input.range("two_inner")))))); |
| } |
| |
| TEST(EligibleRangesTest, TemplateOfConstPointers) { |
| auto Input = Annotations(R"( |
| template <typename One, typename Two> |
| struct S {}; |
| |
| void target( |
| S<$one[[const int *]], $two_o[[$two_i[[const int *]] const *]]> p, |
| S<$three[[int *]] const, $four_o[[$four_i[[int *]] const *]] const> q); |
| )"); |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(SlotRange(-1, Input.range("one")), |
| SlotRange(-1, Input.range("two_o")), |
| SlotRange(-1, Input.range("two_i")), |
| SlotRange(-1, Input.range("three")), |
| SlotRange(-1, Input.range("four_o")), |
| SlotRange(-1, Input.range("four_i")))))); |
| } |
| |
| TEST(EligibleRangesTest, UniquePtr) { |
| auto Input = Annotations(R"( |
| namespace std { |
| template <typename T> |
| class unique_ptr; |
| } |
| |
| void target($one[[std::unique_ptr<int>]] std_smart, |
| Nonnull<$two[[std::unique_ptr<int>]]> nonnull_std_smart); |
| )"); |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(SlotRange(1, Input.range("one")), |
| SlotRange(2, Input.range("two")))))); |
| } |
| |
| TEST(EligibleRangesTest, UserDefinedSmartPointer) { |
| auto Input = Annotations(R"( |
| struct MySmartIntPtr { |
| using absl_nullability_compatible = void; |
| using pointer = int *; |
| }; |
| |
| void target($one[[MySmartIntPtr]] user_defined_smart, |
| Nonnull<$two[[MySmartIntPtr]]> nonnull_user_defined_smart); |
| )"); |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(SlotRange(1, Input.range("one")), |
| SlotRange(2, Input.range("two")))))); |
| } |
| |
| TEST(EligibleRangesTest, UserDefinedTemplatedSmartPointer) { |
| auto Input = Annotations(R"( |
| template <typename T> |
| struct MySmartPtr { |
| using absl_nullability_compatible = void; |
| }; |
| |
| void target($one[[MySmartPtr<int>]] user_defined_smart, |
| Nonnull<$two[[MySmartPtr<int>]]> nonnull_user_defined_smart); |
| )"); |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(SlotRange(1, Input.range("one")), |
| SlotRange(2, Input.range("two")))))); |
| } |
| |
| TEST(EligibleRangesTest, SimpleAlias) { |
| auto Input = Annotations(R"( |
| using IntPtr = int *; |
| |
| void target([[IntPtr]] a); |
| )"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, UnorderedElementsAre(SlotRange(1, Input.range()))))); |
| } |
| |
| TEST(EligibleRangesTest, InaccessibleAlias) { |
| auto Input = Annotations(R"( |
| template <typename T> |
| class TemplateClass {}; |
| using Inaccessible = TemplateClass<int *>; |
| |
| void target(Inaccessible a); |
| )"); |
| EXPECT_EQ(getFunctionRanges(Input.code()), std::nullopt); |
| } |
| |
| TEST(EligibleRangesTest, NestedAlias) { |
| auto Input = Annotations(R"( |
| using Nested = int **; |
| |
| void target($[[Nested]] a); |
| )"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, UnorderedElementsAre(SlotRange(1, Input.range()))))); |
| } |
| |
| TEST(EligibleRangesTest, AliasTemplate) { |
| auto Input = Annotations(R"( |
| template <typename T> |
| using AliasTemplate = T; |
| |
| void target(AliasTemplate<[[int*]]> a, AliasTemplate<int> b); |
| )"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, UnorderedElementsAre(SlotRange(1, Input.range()))))); |
| } |
| |
| TEST(EligibleRangesTest, DependentAliasSimple) { |
| auto Input = Annotations(R"( |
| template <typename T> |
| struct S { |
| using type = T; |
| }; |
| |
| void target(S<[[int *]]>::type a, S<int>::type b); |
| )"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, UnorderedElementsAre(SlotRange(1, Input.range()))))); |
| } |
| |
| TEST(EligibleRangesTest, DependentAliasAnnotated) { |
| auto Input = Annotations(R"( |
| template <typename T> |
| struct S { |
| using type = T; |
| }; |
| |
| void target(S<Nullable<[[int *]]>>::type a); |
| )"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, UnorderedElementsAre(SlotRange(1, Input.range()))))); |
| } |
| |
| TEST(EligibleRangesTest, DependentAliasOfDependentAlias) { |
| auto Input = Annotations(R"( |
| template <typename T> |
| struct vector { |
| using value_type = T; |
| }; |
| template <typename T> |
| struct S { |
| using type = vector<T>::value_type; |
| }; |
| |
| void target(S<[[int *]]>::type a); |
| )"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, UnorderedElementsAre(SlotRange(1, Input.range()))))); |
| } |
| |
| TEST(EligibleRangesTest, DependentAliasTemplate) { |
| auto Input = Annotations(R"( |
| template <typename V> |
| struct vector {}; |
| template <typename T> |
| struct S { |
| template <template<typename> typename U> |
| using type = U<T>; |
| }; |
| |
| void target(S<[[int*]]>::type<vector> a); |
| )"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, UnorderedElementsAre(SlotRange(-1, Input.range()))))); |
| } |
| |
| TEST(EligibleRangesTest, DependentAliasNested) { |
| auto Input = Annotations(R"( |
| template <typename V> |
| struct vector { |
| using value_type = V; |
| }; |
| |
| void target(vector<$one[[$two[[$three[[int*]]*]]*]]>::value_type a); |
| )"); |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(SlotRange(1, Input.range("one")), |
| SlotRange(-1, Input.range("two")), |
| SlotRange(-1, Input.range("three")))))); |
| } |
| |
| TEST(EligibleRangesTest, NoreturnAliasLosesFunctionTypeSourceInfo) { |
| // This previously crashed because the noreturn attribute causes the |
| // TypedefType to be unwrapped and rewritten without the Typedef layer and |
| // the source information below that layer to be dropped. |
| auto Input = Annotations(R"( |
| typedef void (*Alias)(const char *, ...); |
| |
| __attribute__((__noreturn__)) [[Alias]] target; |
| )"); |
| EXPECT_THAT( |
| getVarRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, UnorderedElementsAre(SlotRange(0, Input.range("")))))); |
| } |
| |
| TEST(EligibleRangesTest, TemplatedClassContext) { |
| auto Input = Annotations(R"( |
| template <typename T> |
| struct Outer { |
| struct Inner {}; |
| }; |
| |
| void target(Outer<[[int *]]>::Inner a); |
| )"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, UnorderedElementsAre(SlotRange(-1, Input.range()))))); |
| } |
| |
| TEST(EligibleRangesTest, NestedTemplatedClasses) { |
| auto Input = Annotations(R"( |
| template <typename S> |
| struct Outermost { |
| template <typename T> |
| struct Outer { |
| template <typename U> |
| struct Inner {}; |
| }; |
| }; |
| |
| void target( |
| Outermost<$three[[char *]]>::Outer<$two[[int *]]>::Inner<$one[[bool *]]> |
| a); |
| )"); |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(SlotRange(-1, Input.range("one")), |
| SlotRange(-1, Input.range("two")), |
| SlotRange(-1, Input.range("three")))))); |
| } |
| |
| TEST(EligibleRangesTest, DependentAliasReferencingFurtherOutTemplateParam) { |
| auto Input = Annotations(R"( |
| template <typename S> |
| struct Outermost { |
| template <typename T> |
| struct Outer { |
| template <typename U> |
| using Inner = S; |
| }; |
| }; |
| |
| void target(Outermost<[[int*]]>::Outer<bool>::Inner<char*> a); |
| )"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, UnorderedElementsAre(SlotRange(1, Input.range()))))); |
| } |
| |
| TEST(EligibleRangesTest, DependentAliasForwardingMultipleTemplateArguments) { |
| auto Input = Annotations(R"( |
| template <typename T, class U> |
| struct Pair; |
| template <typename T, class U> |
| struct PairWrapper { |
| using type = Pair<T , U>; |
| }; |
| |
| void target(PairWrapper<$one[[int *]], $two[[bool *]]>::type a); |
| )"); |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(SlotRange(-1, Input.range("one")), |
| SlotRange(-1, Input.range("two")))))); |
| } |
| |
| TEST(EligibleRangesTest, DependentAliasInMultipleNestedClassContexts) { |
| auto Input = Annotations(R"( |
| template <typename A, class B> |
| struct Pair; |
| |
| template <typename T> |
| struct Outer { |
| template <typename U> |
| struct Inner { |
| using type = Pair<T, U>; |
| }; |
| }; |
| |
| void target(Outer<$one[[int *]]>::Inner<$two[[bool *]]>::type a); |
| )"); |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(SlotRange(-1, Input.range("one")), |
| SlotRange(-1, Input.range("two")))))); |
| } |
| |
| TEST(EligibleRangesTest, AliasTemplateInNestedClassContext) { |
| auto Input = Annotations(R"( |
| template <typename A, class B> |
| struct Pair; |
| |
| template <typename T> |
| struct Outer { |
| template <typename U> |
| using Inner = Pair<T, U>; |
| }; |
| |
| void target(Outer<$one[[int *]]>::Inner<$two[[bool *]]> a); |
| )"); |
| |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(SlotRange(-1, Input.range("one")), |
| SlotRange(-1, Input.range("two")))))); |
| } |
| |
| TEST(EligibleRangesTest, DependentAliasOfSmartPointer) { |
| auto Input = Annotations(R"( |
| namespace std { |
| template <typename T> |
| class unique_ptr; |
| } |
| |
| template <typename T> |
| struct S { |
| using type = std::unique_ptr<T>; |
| }; |
| |
| void target($unique_ptr[[S<$inner[[int*]]>::type]] a); |
| )"); |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(SlotRange(1, Input.range("unique_ptr")), |
| SlotRange(-1, Input.range("inner")))))); |
| } |
| |
| TEST(EligibleRangesTest, DependentlyNamedTemplate) { |
| auto Input = Annotations(R"( |
| struct Wrapper { |
| template <typename T> |
| using Alias = T; |
| }; |
| |
| template <typename U, class WrapT> |
| struct S { |
| using type = typename WrapT::template Alias<U> *; |
| }; |
| |
| // a's canonical type is int**. The outer pointer's range is the whole type, |
| // and the inner pointer's range is the first template argument to S. |
| void target($outer[[S<$inner[[int *]], Wrapper>::type]] a); |
| )"); |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(SlotRange(1, Input.range("outer")), |
| SlotRange(-1, Input.range("inner")))))); |
| } |
| |
| TEST(EligibleRangesTest, PartialSpecialization) { |
| auto Input = Annotations(R"( |
| template <typename T> |
| struct S { |
| }; |
| template <typename P> |
| struct S<P *> { |
| using Alias = P; |
| }; |
| |
| // a's canonical type is int * and derives its nullability from the template |
| // argument minus a layer of pointer indirection. But NullabilityWalker |
| // doesn't support resugaring template arguments in partial specializations, |
| // so we only see the pointer type at the alias' Loc. |
| void target([[S<int **>::Alias]] a); |
| )"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, UnorderedElementsAre(SlotRange(1, Input.range()))))); |
| } |
| |
| TEST(EligibleRangesTest, TypeTemplateParamPack) { |
| auto Input = Annotations(R"( |
| template <typename... T> |
| struct Tuple { |
| using type = int; |
| }; |
| |
| void target(Tuple<$one[[int *]], $two[[$three[[int *]]*]]> a, |
| Tuple<int *, int **>::type b); |
| )"); |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(SlotRange(-1, Input.range("one")), |
| SlotRange(-1, Input.range("two")), |
| SlotRange(-1, Input.range("three")))))); |
| } |
| |
| TEST(EligibleRangesTest, DefaultTemplateArgs) { |
| auto Input = Annotations(R"( |
| template <typename T1, typename T2 = int*> |
| struct S {}; |
| template <typename T1, typename T2 = T1> |
| using Alias = T2; |
| |
| void target(S<$one[[int *]]> a, $two[[Alias<int *>]] b); |
| )"); |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre( |
| SlotRange(-1, Input.range("one")), |
| // TODO(b/281474380) Collect the template |
| // argument instead of the whole alias, when we can see |
| // through the layers of default argument redirection |
| SlotRange(2, Input.range("two")))))); |
| } |
| |
| TEST(EligibleRangesTest, MultipleSlotsOneRange) { |
| auto Input = Annotations(R"( |
| template <typename T1, typename T2> |
| struct Pair { |
| T1 first; |
| T2 second; |
| }; |
| template <typename T> |
| using Couple = Pair<T, T>; |
| |
| void target(Couple<[[int *]]> c); |
| )"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| // Eventually, two different valid slot values for the two |
| // ranges, but for now, inference looks at neither of |
| // them, so both have no slot. |
| MainFileName, UnorderedElementsAre(SlotRange(-1, Input.range()), |
| SlotRange(-1, Input.range()))))); |
| } |
| |
| TEST(EligibleRangesTest, Field) { |
| auto Input = Annotations(R"( |
| struct S { |
| $zero[[$one[[int *]]*]] target; |
| }; |
| )"); |
| EXPECT_THAT(getFieldRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(SlotRange(0, Input.range("zero")), |
| SlotRange(-1, Input.range("one")))))); |
| } |
| |
| TEST(EligibleRangesTest, StaticFieldAkaGlobal) { |
| auto Input = Annotations(R"( |
| struct S { |
| static $zero[[$one[[int *]]*]] target; |
| }; |
| )"); |
| EXPECT_THAT(getVarRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(SlotRange(0, Input.range("zero")), |
| SlotRange(-1, Input.range("one")))))); |
| } |
| |
| TEST(EligibleRangesTest, GlobalVariable) { |
| auto Input = Annotations(R"( |
| $zero[[$one[[int *]]*]] target; |
| )"); |
| EXPECT_THAT(getVarRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(SlotRange(0, Input.range("zero")), |
| SlotRange(-1, Input.range("one")))))); |
| } |
| |
| TEST(EligibleRangesTest, Lambda) { |
| auto Input = Annotations(R"( |
| auto lambda = []($one[[int *]]) -> $zero[[int *]] {}; |
| )"); |
| EXPECT_THAT(getFunctionRanges(Input.code(), "operator()"), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(SlotRange(0, Input.range("zero")), |
| SlotRange(1, Input.range("one"))), |
| Nullability::UNKNOWN))); |
| } |
| |
| TEST(EligibleRangesTest, LambdaCaptureWithFunctionTypeInTemplateArg) { |
| std::string Input = R"cc( |
| template <typename T> |
| using ATemplate = T; |
| |
| void WithPartsOfWindowedValue(ATemplate<void(const int)> *f) { |
| [&f]() { f(0); }(); |
| } |
| )cc"; |
| // We expect no ranges for the lambda's implicit FieldDecl, which for some |
| // reason has an incomplete FunctionTypeLoc that has only nullptrs where the |
| // ParmVarDecls should be for the function parameters. |
| EXPECT_EQ(getFieldRanges(Input, ""), std::nullopt); |
| } |
| |
| TEST(EligibleRangesTest, Pragma) { |
| auto Input = Annotations(R"( |
| #pragma nullability file_default nonnull |
| |
| $zero[[$one[[int *]]*]] target($param_one[[int *]], $param_two[[int *]]); |
| )"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre( |
| SlotRange(0, Input.range("zero"), Nullability::NONNULL), |
| SlotRange(-1, Input.range("one"), Nullability::NONNULL), |
| SlotRange(1, Input.range("param_one"), Nullability::NONNULL), |
| SlotRange(2, Input.range("param_two"), Nullability::NONNULL)), |
| Nullability::NONNULL))); |
| |
| Input = Annotations(R"( |
| #pragma nullability file_default nullable |
| [[int*]] target; |
| )"); |
| EXPECT_THAT( |
| getVarRanges(Input.code()), |
| Optional(TypeLocRanges(MainFileName, |
| UnorderedElementsAre(SlotRange( |
| 0, Input.range(), Nullability::NULLABLE)), |
| Nullability::NULLABLE))); |
| |
| Input = Annotations(R"( |
| [[int*]] target; |
| )"); |
| EXPECT_THAT( |
| getVarRanges(Input.code()), |
| Optional(TypeLocRangesWithNoPragmaNullability( |
| MainFileName, UnorderedElementsAre(SlotRangeWithNoExistingAnnotation( |
| 0, Input.range()))))); |
| } |
| |
| MATCHER(NoPreRangeLength, "") { |
| return !arg.has_existing_annotation_pre_range_length(); |
| } |
| |
| MATCHER(NoPostRangeLength, "") { |
| return !arg.has_existing_annotation_post_range_length(); |
| } |
| |
| MATCHER_P(PreRangeLength, Length, "") { |
| return arg.has_existing_annotation_pre_range_length() && |
| arg.existing_annotation_pre_range_length() == Length; |
| } |
| |
| MATCHER_P(PostRangeLength, Length, "") { |
| return arg.has_existing_annotation_post_range_length() && |
| arg.existing_annotation_post_range_length() == Length; |
| } |
| |
| TEST(ExistingAnnotationLengthTest, AbslTemplate) { |
| auto Input = Annotations(R"( |
| namespace absl { |
| template <typename T> |
| using NullabilityUnknown = ::NullabilityUnknown<T>; |
| template <typename T> |
| using Nullable = ::Nullable<T>; |
| template <typename T> |
| using Nonnull = ::Nonnull<T>; |
| } |
| void target($no[[int*]] p, absl::NullabilityUnknown<$yes[[int*]]> q, |
| absl::/* a comment*/NullabilityUnknown< /* a comment */ |
| $with_comments[[int*]] /* a comment */ > r, |
| absl::Nullable<$nullable[[int*]]> s, |
| absl::Nonnull<$nonnull[[int*]]> t); |
| )"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, testing::ElementsAre( |
| AllOf(SlotRange(1, Input.range("no")), |
| NoPreRangeLength(), NoPostRangeLength()), |
| AllOf(SlotRange(2, Input.range("yes")), |
| PreRangeLength(25), PostRangeLength(1)), |
| AllOf(SlotRange(3, Input.range("with_comments")), |
| PreRangeLength(70), PostRangeLength(19)), |
| AllOf(SlotRange(4, Input.range("nullable")), |
| PreRangeLength(15), PostRangeLength(1)), |
| AllOf(SlotRange(5, Input.range("nonnull")), |
| PreRangeLength(14), PostRangeLength(1)))))); |
| } |
| |
| TEST(ExistingAnnotationLengthTest, AnnotationInMacro) { |
| auto Input = Annotations(R"( |
| namespace absl { |
| template <typename T> |
| using NullabilityUnknown = ::NullabilityUnknown<T>; |
| } |
| |
| #define UNKNOWN(T) absl::NullabilityUnknown<T> |
| |
| void target(UNKNOWN([[int *]]) x); |
| )"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, UnorderedElementsAre(AllOf( |
| SlotRange(1, Input.range("")), |
| // The token checks looking for annotations are done |
| // without expansion of macros, so we see a left |
| // paren as the preceding token and report no |
| // existing pre-range/post-range annotation. |
| NoPreRangeLength(), NoPostRangeLength()))))); |
| } |
| |
| TEST(ExistingAnnotationLengthTest, UniquePtr) { |
| auto Input = Annotations(R"( |
| namespace std { |
| template <typename T> |
| class unique_ptr; |
| } |
| namespace absl { |
| template <typename T> |
| using NullabilityUnknown = ::NullabilityUnknown<T>; |
| } |
| |
| void target(absl::NullabilityUnknown<[[std::unique_ptr<int>]]> x); |
| )"); |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, UnorderedElementsAre(AllOf( |
| SlotRange(1, Input.range("")), |
| PreRangeLength(25), PostRangeLength(1)))))); |
| } |
| |
| TEST(ExistingAnnotationLengthTest, DoubleClosingAngleBrackets) { |
| auto Input = Annotations(R"( |
| namespace absl { |
| template <typename T> |
| using NullabilityUnknown = ::NullabilityUnknown<T>; |
| } |
| |
| template <typename T> |
| using MyTemplateAlias = T; |
| |
| void target(MyTemplateAlias<absl::NullabilityUnknown<$nothing[[int *]]>> x, |
| MyTemplateAlias<absl::NullabilityUnknown<$comment[[int *]]>/* a comment */> y, |
| MyTemplateAlias<absl::NullabilityUnknown<$whitespace[[int *]]> |
| > z); |
| )"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, UnorderedElementsAre( |
| AllOf(SlotRange(1, Input.range("nothing")), |
| PreRangeLength(25), PostRangeLength(1)), |
| AllOf(SlotRange(2, Input.range("comment")), |
| PreRangeLength(25), PostRangeLength(1)), |
| AllOf(SlotRange(3, Input.range("whitespace")), |
| PreRangeLength(25), PostRangeLength(1)))))); |
| } |
| |
| TEST(ExistingAnnotationLengthTest, ClangAttribute) { |
| auto Input = Annotations(R"( |
| void target($no[[int*]] p, $yes[[int*]] _Null_unspecified q, |
| $with_comment[[int*]]/* a comment */_Null_unspecified r, |
| $nullable[[int*]] _Nullable s, $nonnull[[int*]] _Nonnull t); |
| )"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(AllOf(SlotRange(1, Input.range("no")), |
| NoPreRangeLength(), NoPostRangeLength()), |
| AllOf(SlotRange(2, Input.range("yes")), |
| PreRangeLength(0), PostRangeLength(18)), |
| AllOf(SlotRange(3, Input.range("with_comment")), |
| PreRangeLength(0), PostRangeLength(32)), |
| AllOf(SlotRange(4, Input.range("nullable")), |
| PreRangeLength(0), PostRangeLength(10)), |
| AllOf(SlotRange(5, Input.range("nonnull")), |
| PreRangeLength(0), PostRangeLength(9)))))); |
| } |
| |
| MATCHER(EquivalentRanges, "") { |
| return std::get<0>(arg).begin() == std::get<1>(arg).Begin && |
| std::get<0>(arg).end() == std::get<1>(arg).End; |
| } |
| |
| MATCHER_P2(ComplexDeclaratorImpl, FollowingAnnotation, Ranges, "") { |
| if (!arg.has_complex_declarator_ranges()) { |
| *result_listener << "no complex declarator ranges present"; |
| return false; |
| } |
| ComplexDeclaratorRanges ArgRanges = arg.complex_declarator_ranges(); |
| return ExplainMatchResult(FollowingAnnotation, |
| ArgRanges.following_annotation(), |
| result_listener) && |
| ExplainMatchResult(Pointwise(EquivalentRanges(), Ranges), |
| ArgRanges.removal(), result_listener); |
| } |
| |
| auto ComplexDeclarator(llvm::StringRef FollowingAnnotation, |
| std::vector<Annotations::Range> Ranges) { |
| return ComplexDeclaratorImpl(FollowingAnnotation, Ranges); |
| } |
| |
| MATCHER(NoComplexDeclarator, "") { |
| return !arg.has_complex_declarator_ranges(); |
| } |
| |
| TEST(ComplexDeclaratorTest, FunctionPointer) { |
| auto Input = Annotations(R"( |
| void target($func_pointer[[int (*$remove_from_type[[p]])(int, $pointer_param[[int*]])]]); |
| )"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre( |
| AllOf(SlotRange(1, Input.range("func_pointer")), |
| ComplexDeclarator("p", {Input.range("remove_from_type")})), |
| AllOf(SlotRange(-1, Input.range("pointer_param")), |
| NoComplexDeclarator()))))); |
| |
| Input = Annotations("void target($unnamed[[int (*)(int)]]);"); |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, UnorderedElementsAre( |
| AllOf(SlotRange(1, Input.range("unnamed")), |
| NoComplexDeclarator()))))); |
| } |
| |
| TEST(ComplexDeclaratorTest, ArrayOfNonPointersHasNoRanges) { |
| std::string Input = "void target(int p[]);"; |
| EXPECT_EQ(getFunctionRanges(Input), std::nullopt); |
| } |
| |
| TEST(ComplexDeclaratorTest, ArrayOfSimplePointers) { |
| auto Input = Annotations("void target([[int*]] p[]);"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, UnorderedElementsAre(AllOf(SlotRange(-1, Input.range()), |
| NoComplexDeclarator()))))); |
| } |
| |
| TEST(ComplexDeclaratorTest, ArrayOfFunctionPointers) { |
| // Can't use ranges marked by [[...]] around arrays because of the adjacent |
| // closing square bracket at the end of the array length and the unfortunate |
| // syntax of Annotations, so use individual points. |
| auto Input = Annotations("void target([[int (*$1^p[3]$2^)(float)]]);"); |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(AllOf( |
| SlotRange(-1, Input.range()), |
| ComplexDeclarator( |
| "p[3]", {Annotations::Range(Input.point("1"), |
| Input.point("2"))})))))); |
| |
| // An unnamed array of function pointers. The array brackets are still moved. |
| Input = Annotations("void target([[void(*$1^[]$2^)(int)]]);"); |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(AllOf( |
| SlotRange(-1, Input.range()), |
| ComplexDeclarator( |
| "[]", {Annotations::Range(Input.point("1"), |
| Input.point("2"))})))))); |
| } |
| |
| TEST(ComplexDeclaratorTest, ArrayOfArrayOfPointersToArray) { |
| // Can't use ranges marked by [[...]] around arrays because of the adjacent |
| // closing square bracket at the end of the array length and the unfortunate |
| // syntax of Annotations, so use individual points. |
| auto Input = Annotations(R"( |
| void target($1^$range[[int*]] (*$3^p[3][2]$4^)[1]$2^);)"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre( |
| AllOf(SlotRange(-1, Input.range("range")), NoComplexDeclarator()), |
| AllOf(SlotRange(-1, Annotations::Range(Input.point("1"), |
| Input.point("2"))), |
| ComplexDeclarator( |
| "p[3][2]", {Annotations::Range(Input.point("3"), |
| Input.point("4"))})))))); |
| } |
| |
| TEST(ComplexDeclaratorTest, PointerToArray) { |
| // Can't use ranges marked by [[...]] around arrays because of the adjacent |
| // closing square bracket at the end of the array length and the unfortunate |
| // syntax of Annotations, so use individual points. |
| auto Input = |
| Annotations(R"(void target($1^int (*$remove_from_type[[p]])[]$2^);)"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(AllOf( |
| SlotRange(1, |
| Annotations::Range(Input.point("1"), Input.point("2"))), |
| ComplexDeclarator("p", {Input.range("remove_from_type")})))))); |
| |
| // An unnamed pointer to an array. There's nothing to move. |
| Input = Annotations(R"(void target($1^int (*)[]$2^);)"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, UnorderedElementsAre(AllOf( |
| SlotRange(1, Annotations::Range(Input.point("1"), |
| Input.point("2"))), |
| NoComplexDeclarator()))))); |
| } |
| |
| TEST(ComplexDeclaratorTest, |
| ArrayOfPointersWithExtraParensAroundNameAndInSizeBrackets) { |
| // Can't use ranges marked by [[...]] around arrays because of the adjacent |
| // closing square bracket at the end of the array length and the unfortunate |
| // syntax of Annotations, so use individual points. |
| auto Input = Annotations(R"(void target([[int (*$3^((p))[(1 + 2)]$4^)]]);)"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre(AllOf( |
| SlotRange(-1, Input.range()), |
| ComplexDeclarator("((p))[(1 + 2)]", |
| {Annotations::Range(Input.point("3"), |
| Input.point("4"))})))))); |
| } |
| |
| TEST(ComplexDeclaratorTest, PointerToPointerToArray) { |
| // Can't use ranges marked by [[...]] around arrays because of the adjacent |
| // closing square bracket at the end of the array length and the unfortunate |
| // syntax of Annotations, so use individual points. |
| auto Input = |
| Annotations(R"(void target($1^int (*$star[[*]]$q[[q]])[1]$2^);)"); |
| EXPECT_THAT(getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre( |
| AllOf(SlotRange(1, Annotations::Range(Input.point("1"), |
| Input.point("2"))), |
| ComplexDeclarator("q", {Input.range("q")})), |
| AllOf(SlotRange(-1, Annotations::Range(Input.point("1"), |
| Input.point("2"))), |
| ComplexDeclarator("*", {Input.range("star")})))))); |
| } |
| |
| TEST(ComplexDeclaratorTest, PointerToArrayOfFunctionPointers) { |
| // Can't use ranges marked by [[...]] around arrays because of the adjacent |
| // closing square bracket at the end of the array length and the unfortunate |
| // syntax of Annotations, so use individual points. |
| auto Input = Annotations( |
| R"(void target($whole[[void (*$1^(*$f[[(f)]])[]$2^)(int)]]);)"); |
| EXPECT_THAT( |
| getFunctionRanges(Input.code()), |
| Optional(TypeLocRanges( |
| MainFileName, |
| UnorderedElementsAre( |
| AllOf(SlotRange(1, Input.range("whole")), |
| ComplexDeclarator("(f)", {Input.range("f")})), |
| AllOf(SlotRange(-1, Input.range("whole")), |
| ComplexDeclarator( |
| "(*)[]", {Annotations::Range(Input.point("1"), |
| Input.range("f").Begin), |
| Annotations::Range(Input.range("f").End, |
| Input.point("2"))})))))); |
| } |
| |
| } // namespace |
| } // namespace clang::tidy::nullability |