blob: 4a9b8be90b9a9380ec344576ac3f3aeb1cee10d8 [file] [log] [blame]
// 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/infer_tu.h"
#include <optional>
#include <vector>
#include "nullability/inference/augmented_test_inputs.h"
#include "nullability/inference/inference.proto.h"
#include "nullability/pragma.h"
#include "nullability/proto_matchers.h"
#include "nullability/type_nullability.h"
#include "clang/AST/Decl.h"
#include "clang/ASTMatchers/ASTMatchFinder.h"
#include "clang/ASTMatchers/ASTMatchers.h"
#include "clang/ASTMatchers/ASTMatchersMacros.h"
#include "clang/Basic/LLVM.h"
#include "clang/Index/USRGeneration.h"
#include "clang/Testing/TestAST.h"
#include "llvm/ADT/SmallString.h"
#include "llvm/ADT/StringRef.h"
#include "third_party/llvm/llvm-project/third-party/unittest/googlemock/include/gmock/gmock.h"
#include "third_party/llvm/llvm-project/third-party/unittest/googletest/include/gtest/gtest.h"
namespace clang::tidy::nullability {
namespace {
using ast_matchers::hasName;
using testing::_;
using testing::ElementsAre;
using testing::IsSupersetOf;
using testing::UnorderedElementsAre;
test::EnableSmartPointers Enable;
MATCHER_P2(inferredSlot, I, Nullability, "") {
return arg.slot() == I && arg.nullability() == Nullability;
}
MATCHER_P3(inferredSlot, I, Nullability, Conflict, "") {
return arg.slot() == I && arg.nullability() == Nullability &&
arg.conflict() == Conflict;
}
MATCHER_P2(inferenceMatcher, USR, SlotsMatcher, "") {
if (arg.symbol().usr() != USR) return false;
return testing::ExplainMatchResult(SlotsMatcher, arg.slot_inference(),
result_listener);
}
AST_MATCHER(Decl, isCanonical) { return Node.isCanonicalDecl(); }
class InferTUTest : public ::testing::Test {
protected:
std::optional<TestAST> AST;
NullabilityPragmas Pragmas;
void build(llvm::StringRef Code) {
AST.emplace(getAugmentedTestInputs(Code, Pragmas));
}
auto infer() { return inferTU(AST->context(), Pragmas); }
// Returns a matcher for an Inference.
// The DeclMatcher should uniquely identify the symbol being described.
// (We use this to compute the USR we expect to find in the inference proto).
// Slots should describe the slots that were inferred.
template <typename MatcherT>
testing::Matcher<const Inference &> inference(
MatcherT DeclMatcher,
std::vector<testing::Matcher<const Inference::SlotInference &>> Slots) {
llvm::SmallString<128> USR;
auto Matches = ast_matchers::match(
ast_matchers::namedDecl(isCanonical(), DeclMatcher).bind("decl"),
AST->context());
EXPECT_EQ(Matches.size(), 1);
if (auto *D = ast_matchers::selectFirst<Decl>("decl", Matches))
EXPECT_FALSE(index::generateUSRForDecl(D, USR));
return inferenceMatcher(USR, testing::ElementsAreArray(Slots));
}
};
TEST_F(InferTUTest, UncheckedDeref) {
build(R"cc(
void target(int *p, bool cond) {
if (cond) *p;
}
void guarded(int *p) {
if (p) *p;
}
)cc");
EXPECT_THAT(infer(),
ElementsAre(inference(hasName("target"),
{inferredSlot(1, Nullability::NONNULL)})));
}
TEST_F(InferTUTest, Samples) {
llvm::StringRef Code =
"void target(int * p) { *p + *p; }\n"
"void another(int x) { target(&x); }";
// 123456789012345678901234567890123456789
// 0 1 2 3
build(Code);
auto Results = infer();
ASSERT_THAT(Results,
ElementsAre(inference(hasName("target"),
{inferredSlot(1, Nullability::NONNULL)})));
EXPECT_THAT(Results.front().slot_inference(0).sample_evidence(),
testing::UnorderedElementsAre(
EqualsProto(R"pb(location: "input.cc:2:30"
kind: NONNULL_ARGUMENT)pb"),
EqualsProto(R"pb(location: "input.cc:1:24"
kind: UNCHECKED_DEREFERENCE)pb"),
EqualsProto(R"pb(location: "input.cc:1:29"
kind: UNCHECKED_DEREFERENCE)pb")));
}
TEST_F(InferTUTest, Annotations) {
build(R"cc(
Nonnull<int *> target(int *a, int *b);
Nonnull<int *> target(int *a, Nullable<int *> p) { *p; }
)cc");
EXPECT_THAT(infer(),
ElementsAre(inference(hasName("target"),
{
inferredSlot(0, Nullability::NONNULL),
inferredSlot(2, Nullability::NULLABLE),
})));
}
TEST_F(InferTUTest, AnnotationsConflict) {
build(R"cc(
Nonnull<int *> target();
Nullable<int *> target();
)cc");
EXPECT_THAT(infer(),
ElementsAre(inference(hasName("target"),
{inferredSlot(0, Nullability::UNKNOWN)})));
}
TEST_F(InferTUTest, ParamsFromCallSite) {
build(R"cc(
void callee(int* p, int* q, int* r);
void target(int* a, Nonnull<int*> b, Nullable<int*> c) { callee(a, b, c); }
)cc");
ASSERT_THAT(infer(),
Contains(inference(hasName("callee"),
{
inferredSlot(1, Nullability::UNKNOWN),
inferredSlot(2, Nullability::NONNULL),
inferredSlot(3, Nullability::NULLABLE),
})));
}
TEST_F(InferTUTest, ReturnTypeNullable) {
build(R"cc(
int* target() { return nullptr; }
)cc");
EXPECT_THAT(infer(),
ElementsAre(inference(hasName("target"),
{inferredSlot(0, Nullability::NULLABLE)})));
}
TEST_F(InferTUTest, ReturnTypeNonnull) {
build(R"cc(
Nonnull<int*> providesNonnull();
int* target() { return providesNonnull(); }
)cc");
EXPECT_THAT(infer(),
Contains(inference(hasName("target"),
{inferredSlot(0, Nullability::NONNULL)})));
}
TEST_F(InferTUTest, ReturnTypeNonnullAndUnknown) {
build(R"cc(
Nonnull<int*> providesNonnull();
int* target(bool b, int* q) {
if (b) return q;
return providesNonnull();
}
)cc");
EXPECT_THAT(infer(),
Contains(inference(hasName("target"),
{inferredSlot(0, Nullability::UNKNOWN)})));
}
TEST_F(InferTUTest, ReturnTypeNonnullAndNullable) {
build(R"cc(
Nonnull<int*> providesNonnull();
int* target(bool b) {
if (b) return nullptr;
return providesNonnull();
}
)cc");
EXPECT_THAT(infer(),
Contains(inference(hasName("target"),
{inferredSlot(0, Nullability::NULLABLE)})));
}
TEST_F(InferTUTest, ReturnTypeDereferenced) {
build(R"cc(
struct S {
void member();
};
S* makePtr();
void target() { makePtr()->member(); }
)cc");
EXPECT_THAT(infer(),
ElementsAre(inference(hasName("makePtr"),
{inferredSlot(0, Nullability::NONNULL)})));
}
TEST_F(InferTUTest, PassedToNonnull) {
build(R"cc(
void takesNonnull(Nonnull<int*>);
void target(int* p) { takesNonnull(p); }
)cc");
EXPECT_THAT(infer(),
Contains(inference(hasName("target"),
{inferredSlot(1, Nullability::NONNULL)})));
}
TEST_F(InferTUTest, PassedToMutableNullableRef) {
build(R"cc(
void takesMutableNullableRef(Nullable<int*>&);
void target(int* p) { takesMutableNullableRef(p); }
)cc");
EXPECT_THAT(infer(),
Contains(inference(hasName("target"),
{inferredSlot(1, Nullability::NULLABLE)})));
}
TEST_F(InferTUTest, AssignedFromNullable) {
build(R"cc(
void target(int* p) { p = nullptr; }
)cc");
EXPECT_THAT(infer(),
Contains(inference(hasName("target"),
{inferredSlot(1, Nullability::NULLABLE)})));
}
TEST_F(InferTUTest, CHECKMacro) {
build(R"cc(
// macro must use the parameter, but otherwise body doesn't matter
#define CHECK(x) x
void target(int* p) { CHECK(p); }
)cc");
EXPECT_THAT(infer(),
Contains(inference(hasName("target"),
{inferredSlot(1, Nullability::NONNULL)})));
}
TEST_F(InferTUTest, CHECKNEMacro) {
build(R"cc(
// macro must use the first parameter, but otherwise body doesn't matter
#define CHECK_NE(x, y) x
void target(int* p, int* q, int* r, int* s) {
CHECK_NE(p, nullptr);
CHECK_NE(nullptr, q);
int* a = nullptr;
CHECK_NE(a, r);
CHECK_NE(s, a);
}
)cc");
EXPECT_THAT(
infer(),
UnorderedElementsAre(
inference(hasName("target"), {inferredSlot(1, Nullability::NONNULL),
inferredSlot(2, Nullability::NONNULL),
inferredSlot(3, Nullability::NONNULL),
inferredSlot(4, Nullability::NONNULL)}),
inference(hasName("a"), {inferredSlot(0, Nullability::NULLABLE)})));
}
TEST_F(InferTUTest, Fields) {
build(R"cc(
int* getIntPtr();
struct S {
int* unchecked_deref;
int* default_null_and_unchecked_deref = nullptr;
int* uninitialized;
int NotATarget = *getIntPtr();
void method() {
*unchecked_deref;
*default_null_and_unchecked_deref;
}
};
void foo() {
// Use the implicitly-declared default constructor so that it will be
// generated.
S s;
}
class C {
public:
C() : null_constructor_init(nullptr) {
null_in_constructor_and_unchecked_deref = nullptr;
null_in_constructor = nullptr;
}
void method() { *null_in_constructor_and_unchecked_deref; }
private:
int* null_in_constructor_and_unchecked_deref;
int* null_constructor_init;
int* null_in_constructor;
};
)cc");
EXPECT_THAT(
infer(),
UnorderedElementsAre(
inference(hasName("unchecked_deref"),
{inferredSlot(0, Nullability::NONNULL)}),
// Unchecked deref is strong evidence and a default null
// member initializer is weak.
inference(hasName("default_null_and_unchecked_deref"),
{inferredSlot(0, Nullability::NONNULL)}),
// No inference for uninitialized.,
inference(hasName("getIntPtr"),
{inferredSlot(0, Nullability::NONNULL)}),
// Initialization to null in the constructor or another
// function body is strong, producing a conflict.
inference(hasName("null_in_constructor_and_unchecked_deref"),
{inferredSlot(0, Nullability::NONNULL, /*Conflict*/ true)}),
inference(hasName("null_constructor_init"),
{inferredSlot(0, Nullability::NULLABLE)}),
inference(hasName("null_in_constructor"),
{inferredSlot(0, Nullability::NULLABLE)})));
}
TEST_F(InferTUTest, FieldsImplicitlyDeclaredConstructorNeverUsed) {
build(R"cc(
Nullable<bool *> getNullable();
struct S {
int *I = nullptr;
bool *B = getNullable();
char *C = static_cast<char *>(nullptr);
};
void foo(S s);
)cc");
// Because the implicitly-declared default constructor is never used, it is
// not present in the AST and we never analyze it. So, we collect no evidence
// from default member initializers.
EXPECT_THAT(infer(),
AllOf(AllOf(Not(Contains(inference(hasName("I"), {_}))),
Not(Contains(inference(hasName("B"), {_}))),
Not(Contains(inference(hasName("C"), {_}))))));
}
TEST_F(InferTUTest, FieldsImplicitlyDeclaredConstructorUsed) {
build(R"cc(
Nullable<bool *> getNullable();
struct S {
int *I = nullptr;
bool *B = getNullable();
char *C = static_cast<char *>(nullptr);
};
// A use of the implicitly-declared default constructor, so it is generated
// and included in the AST for us to analyze, allowing us to infer from
// default member initializers.
void foo() { S s; }
)cc");
EXPECT_THAT(
infer(),
IsSupersetOf(
{inference(hasName("I"), {inferredSlot(0, Nullability::NULLABLE)}),
inference(hasName("B"), {inferredSlot(0, Nullability::NULLABLE)}),
inference(hasName("C"), {inferredSlot(0, Nullability::NULLABLE)})}));
}
TEST_F(InferTUTest, GlobalVariables) {
build(R"cc(
int* getIntPtr();
int* I;
bool* B;
int NotATarget = *getIntPtr();
void target() {
I = nullptr;
*B;
}
)cc");
EXPECT_THAT(
infer(),
UnorderedElementsAre(
inference(hasName("I"), {inferredSlot(0, Nullability::NULLABLE)}),
inference(hasName("B"), {inferredSlot(0, Nullability::NONNULL)}),
inference(hasName("getIntPtr"),
{inferredSlot(0, Nullability::NONNULL)})));
}
TEST_F(InferTUTest, StaticMemberVariables) {
build(R"cc(
struct S {
static int* SI;
static bool* SB;
};
void target() {
*S::SI;
S::SB = nullptr;
}
)cc");
EXPECT_THAT(
infer(),
UnorderedElementsAre(
inference(hasName("SI"), {inferredSlot(0, Nullability::NONNULL)}),
inference(hasName("SB"), {inferredSlot(0, Nullability::NULLABLE)})));
}
TEST_F(InferTUTest, Locals) {
build(R"cc(
void target() {
int* A = nullptr;
static int* B = nullptr;
}
)cc");
EXPECT_THAT(
infer(),
UnorderedElementsAre(
inference(hasName("A"), {inferredSlot(0, Nullability::NULLABLE)}),
inference(hasName("B"), {inferredSlot(0, Nullability::NULLABLE)})));
}
TEST_F(InferTUTest, Filter) {
build(R"cc(
int* target1() { return nullptr; }
int* target2() { return nullptr; }
)cc");
EXPECT_THAT(inferTU(AST->context(), Pragmas, /*Iterations=*/1,
[&](const Decl &D) {
return cast<NamedDecl>(D).getNameAsString() !=
"target2";
}),
ElementsAre(inference(hasName("target1"), {_})));
}
TEST_F(InferTUTest, IterationsPropagateInferences) {
build(R"cc(
void takesToBeNonnull(int* x) { *x; }
int* returnsToBeNonnull(int* a) { return a; }
int* target(int* p, int* q, int* r) {
*p;
takesToBeNonnull(q);
q = r;
return returnsToBeNonnull(p);
}
)cc");
EXPECT_THAT(
inferTU(AST->context(), Pragmas, /*Iterations=*/1),
UnorderedElementsAre(
inference(hasName("target"), {inferredSlot(0, Nullability::UNKNOWN),
inferredSlot(1, Nullability::NONNULL),
inferredSlot(2, Nullability::UNKNOWN)}),
inference(hasName("returnsToBeNonnull"),
{inferredSlot(0, Nullability::UNKNOWN),
inferredSlot(1, Nullability::UNKNOWN)}),
inference(hasName("takesToBeNonnull"),
{inferredSlot(1, Nullability::NONNULL)})));
EXPECT_THAT(
inferTU(AST->context(), Pragmas, /*Iterations=*/2),
UnorderedElementsAre(
inference(hasName("target"), {inferredSlot(0, Nullability::UNKNOWN),
inferredSlot(1, Nullability::NONNULL),
inferredSlot(2, Nullability::NONNULL)}),
inference(hasName("returnsToBeNonnull"),
{inferredSlot(0, Nullability::UNKNOWN),
inferredSlot(1, Nullability::NONNULL)}),
inference(hasName("takesToBeNonnull"),
{inferredSlot(1, Nullability::NONNULL)})));
EXPECT_THAT(
inferTU(AST->context(), Pragmas, /*Iterations=*/3),
UnorderedElementsAre(
inference(hasName("target"), {inferredSlot(0, Nullability::UNKNOWN),
inferredSlot(1, Nullability::NONNULL),
inferredSlot(2, Nullability::NONNULL),
inferredSlot(3, Nullability::NONNULL)}),
inference(hasName("returnsToBeNonnull"),
{inferredSlot(0, Nullability::NONNULL),
inferredSlot(1, Nullability::NONNULL)}),
inference(hasName("takesToBeNonnull"),
{inferredSlot(1, Nullability::NONNULL)})));
EXPECT_THAT(
inferTU(AST->context(), Pragmas, /*Iterations=*/4),
UnorderedElementsAre(
inference(hasName("target"), {inferredSlot(0, Nullability::NONNULL),
inferredSlot(1, Nullability::NONNULL),
inferredSlot(2, Nullability::NONNULL),
inferredSlot(3, Nullability::NONNULL)}),
inference(hasName("returnsToBeNonnull"),
{inferredSlot(0, Nullability::NONNULL),
inferredSlot(1, Nullability::NONNULL)}),
inference(hasName("takesToBeNonnull"),
{inferredSlot(1, Nullability::NONNULL)})));
}
TEST_F(InferTUTest, Pragma) {
build(R"cc(
#pragma nullability file_default nonnull
void target(int* default_nonnull, NullabilityUnknown<int*> inferred_nonnull,
Nullable<int*> nullable,
NullabilityUnknown<int*> inferred_nullable,
NullabilityUnknown<int*> unknown) {
default_nonnull = inferred_nonnull;
default_nonnull = nullptr;
inferred_nullable = nullable;
}
)cc");
EXPECT_THAT(infer(),
UnorderedElementsAre(inference(
hasName("target"),
{
// annotation by pragma beats assignment from null, so
// default_nonnull should still be inferred NONNULL
inferredSlot(1, Nullability::NONNULL),
// an explicit unknown does not override a Nonnull
// inference, even if it overrides the pragma
inferredSlot(2, Nullability::NONNULL),
// an explicit nullable overrides pragma default
inferredSlot(3, Nullability::NULLABLE),
// an explicit unknown does not override a Nullable
// inference, which does override the pragma
inferredSlot(4, Nullability::NULLABLE)
// an explicit unknown overrides the pragma, but produces
// no inference, so nothing for slot 5.
})));
}
using InferTUSmartPointerTest = InferTUTest;
TEST_F(InferTUSmartPointerTest, Annotations) {
build(R"cc(
#include <memory>
Nonnull<std::unique_ptr<int>> target(std::unique_ptr<int> a,
std::unique_ptr<int> b);
Nonnull<std::unique_ptr<int>> target(std::unique_ptr<int> a,
Nullable<std::unique_ptr<int>> p) {
*p;
}
)cc");
EXPECT_THAT(infer(),
ElementsAre(inference(hasName("target"),
{
inferredSlot(0, Nullability::NONNULL),
inferredSlot(2, Nullability::NULLABLE),
})));
}
TEST_F(InferTUSmartPointerTest, ParamsFromCallSite) {
build(R"cc(
#include <memory>
#include <utility>
void callee(std::unique_ptr<int> p, std::unique_ptr<int> q,
std::unique_ptr<int> r);
void target(std::unique_ptr<int> a, Nonnull<std::unique_ptr<int>> b,
Nullable<std::unique_ptr<int>> c) {
callee(std::move(a), std::move(b), std::move(c));
}
)cc");
EXPECT_THAT(infer(),
Contains(inference(hasName("callee"),
{
inferredSlot(1, Nullability::UNKNOWN),
inferredSlot(2, Nullability::NONNULL),
inferredSlot(3, Nullability::NULLABLE),
})));
}
TEST_F(InferTUSmartPointerTest, ReturnTypeNullable) {
build(R"cc(
#include <memory>
std::unique_ptr<int> target() { return std::unique_ptr<int>(); }
)cc");
EXPECT_THAT(infer(),
ElementsAre(inference(hasName("target"),
{inferredSlot(0, Nullability::NULLABLE)})));
}
TEST_F(InferTUSmartPointerTest, ReturnTypeNonnull) {
build(R"cc(
#include <memory>
std::unique_ptr<int> target() { return std::make_unique<int>(0); }
)cc");
EXPECT_THAT(infer(),
ElementsAre(inference(hasName("target"),
{inferredSlot(0, Nullability::NONNULL)})));
}
using InferTUVirtualMethodsTest = InferTUTest;
TEST_F(InferTUVirtualMethodsTest, SafeVarianceNoConflicts) {
build(R"cc(
struct Base {
virtual int* foo(int* p) {
*p;
return nullptr;
}
};
struct Derived : public Base {
int* foo(int* p) override {
static int i = 0;
p = nullptr;
return &i;
}
};
)cc");
EXPECT_THAT(infer(),
UnorderedElementsAre(
inference(hasName("Base::foo"),
{inferredSlot(0, Nullability::NULLABLE),
inferredSlot(1, Nullability::NONNULL)}),
inference(hasName("Derived::foo"),
{inferredSlot(0, Nullability::NONNULL),
inferredSlot(1, Nullability::NULLABLE)})));
}
TEST_F(InferTUVirtualMethodsTest, BaseConstrainsDerived) {
build(R"cc(
struct Base {
virtual Nonnull<int*> foo(int* p) {
static int i = 0;
p = nullptr;
return &i;
}
};
struct Derived : public Base {
int* foo(int* p) override;
};
)cc");
EXPECT_THAT(infer(),
UnorderedElementsAre(
inference(hasName("Base::foo"),
{inferredSlot(0, Nullability::NONNULL),
inferredSlot(1, Nullability::NULLABLE)}),
inference(hasName("Derived::foo"),
{inferredSlot(0, Nullability::NONNULL),
inferredSlot(1, Nullability::NULLABLE)})));
}
TEST_F(InferTUVirtualMethodsTest, DerivedConstrainsBase) {
build(R"cc(
struct Base {
virtual int* foo(int* p);
};
struct Derived : public Base {
int* foo(int* p) override {
*p;
return nullptr;
}
};
)cc");
EXPECT_THAT(infer(), UnorderedElementsAre(
inference(hasName("Base::foo"),
{inferredSlot(0, Nullability::NULLABLE),
inferredSlot(1, Nullability::NONNULL)}),
inference(hasName("Derived::foo"),
{inferredSlot(0, Nullability::NULLABLE),
inferredSlot(1, Nullability::NONNULL)})));
}
TEST_F(InferTUVirtualMethodsTest, Conflict) {
build(R"cc(
struct Base {
virtual int* foo(int* p);
};
struct Derived : public Base {
int* foo(int* p) override {
*p;
return nullptr;
}
};
void usage() {
Base B;
// Conflict-producing nonnull return type evidence is only possible
// from a usage site. Since we need a usage, produce the parameter
// evidence here as well.
*B.foo(nullptr);
}
)cc");
EXPECT_THAT(
infer(),
UnorderedElementsAre(
inference(hasName("Base::foo"),
{inferredSlot(0, Nullability::NONNULL, /*Conflict*/ true),
inferredSlot(1, Nullability::NONNULL, /*Conflict*/ true)}),
inference(
hasName("Derived::foo"),
{inferredSlot(0, Nullability::NONNULL, /*Conflict*/ true),
inferredSlot(1, Nullability::NONNULL, /*Conflict*/ true)})));
}
TEST_F(InferTUVirtualMethodsTest, MultipleDerived) {
build(R"cc(
struct Base {
virtual void foo(int* p) { p = nullptr; }
};
struct DerivedA : public Base {
void foo(int* p) override;
};
struct DerivedB : public Base {
void foo(int* p) override;
};
)cc");
EXPECT_THAT(infer(),
UnorderedElementsAre(
inference(hasName("Base::foo"),
{inferredSlot(1, Nullability::NULLABLE)}),
inference(hasName("DerivedA::foo"),
{inferredSlot(1, Nullability::NULLABLE)}),
inference(hasName("DerivedB::foo"),
{inferredSlot(1, Nullability::NULLABLE)})));
}
TEST_F(InferTUVirtualMethodsTest, MultipleBase) {
build(R"cc(
struct BaseA {
virtual void foo(int* p);
};
struct BaseB {
virtual void foo(int* p);
};
struct Derived : public BaseA, public BaseB {
void foo(int* p) override { *p; }
};
)cc");
EXPECT_THAT(infer(), UnorderedElementsAre(
inference(hasName("BaseA::foo"),
{inferredSlot(1, Nullability::NONNULL)}),
inference(hasName("BaseB::foo"),
{inferredSlot(1, Nullability::NONNULL)}),
inference(hasName("Derived::foo"),
{inferredSlot(1, Nullability::NONNULL)})));
}
} // namespace
} // namespace clang::tidy::nullability