Collect evidence for function parameters from observed call sites.
Records Nonnull, Nullable, and Unknown arguments for use in flexible heuristics.
PiperOrigin-RevId: 549105355
Change-Id: Iea59d3ff84987446357d84cdba11df6d6ba8478f
diff --git a/nullability/pointer_nullability_test.cc b/nullability/pointer_nullability_test.cc
index cf35fe0..f410358 100644
--- a/nullability/pointer_nullability_test.cc
+++ b/nullability/pointer_nullability_test.cc
@@ -11,43 +11,66 @@
#include "clang/Analysis/FlowSensitive/Formula.h"
#include "clang/Analysis/FlowSensitive/Value.h"
#include "clang/Analysis/FlowSensitive/WatchedLiteralsSolver.h"
+#include "clang/Basic/Specifiers.h"
#include "third_party/llvm/llvm-project/third-party/unittest/googletest/include/gtest/gtest.h"
namespace clang::tidy::nullability {
namespace {
-TEST(NullabilityPropertiesTest, Test) {
- dataflow::DataflowAnalysisContext DACtx(
- std::make_unique<dataflow::WatchedLiteralsSolver>());
- dataflow::Environment Env(DACtx);
-
- auto &True = DACtx.arena().makeLiteral(true);
- auto &False = DACtx.arena().makeLiteral(false);
- auto &False2 = DACtx.arena().makeNot(True);
-
- auto MakePointer =
- [&](const dataflow::Formula &FromNullable,
- const dataflow::Formula &Null) -> dataflow::PointerValue & {
+class NullabilityPropertiesTest : public ::testing::Test {
+ protected:
+ dataflow::PointerValue &makePointer(const dataflow::Formula &FromNullable,
+ const dataflow::Formula &Null) {
auto &P = Env.create<dataflow::PointerValue>(
DACtx.createStorageLocation(QualType()));
initPointerNullState(P, Env, &Env.arena().makeBoolValue(FromNullable),
&Env.arena().makeBoolValue(Null));
return P;
- };
+ }
+
+ dataflow::DataflowAnalysisContext DACtx = dataflow::DataflowAnalysisContext(
+ std::make_unique<dataflow::WatchedLiteralsSolver>());
+ dataflow::Environment Env = dataflow::Environment(DACtx);
+};
+
+TEST_F(NullabilityPropertiesTest, Test) {
+ auto &True = DACtx.arena().makeLiteral(true);
+ auto &False = DACtx.arena().makeLiteral(false);
+ auto &False2 = DACtx.arena().makeNot(True);
EXPECT_TRUE(
- isNullable(MakePointer(/*FromNullable=*/True, /*Null=*/True), Env));
+ isNullable(makePointer(/*FromNullable=*/True, /*Null=*/True), Env));
EXPECT_FALSE(
- isNullable(MakePointer(/*FromNullable=*/True, /*Null=*/False), Env));
+ isNullable(makePointer(/*FromNullable=*/True, /*Null=*/False), Env));
EXPECT_FALSE(
- isNullable(MakePointer(/*FromNullable=*/False, /*Null=*/True), Env));
+ isNullable(makePointer(/*FromNullable=*/False, /*Null=*/True), Env));
EXPECT_FALSE(
- isNullable(MakePointer(/*FromNullable=*/False, /*Null=*/False), Env));
+ isNullable(makePointer(/*FromNullable=*/False, /*Null=*/False), Env));
EXPECT_FALSE(
- isNullable(MakePointer(/*FromNullable=*/True, /*Null=*/False2), Env));
+ isNullable(makePointer(/*FromNullable=*/True, /*Null=*/False2), Env));
EXPECT_FALSE(
- isNullable(MakePointer(/*FromNullable=*/False2, /*Null=*/True), Env));
+ isNullable(makePointer(/*FromNullable=*/False2, /*Null=*/True), Env));
+}
+
+TEST_F(NullabilityPropertiesTest, IsNullableAdditionalConstraints) {
+ auto &FromNullable = Env.makeAtomicBoolValue().formula();
+ auto &Null = Env.makeAtomicBoolValue().formula();
+ EXPECT_TRUE(isNullable(makePointer(FromNullable, Null), Env));
+
+ auto *NotNull = &DACtx.arena().makeNot(Null);
+ EXPECT_FALSE(isNullable(makePointer(FromNullable, Null), Env, NotNull));
+}
+
+TEST_F(NullabilityPropertiesTest, GetNullabilityAdditionalConstraints) {
+ auto &FromNullable = Env.makeAtomicBoolValue().formula();
+ auto &Null = Env.makeAtomicBoolValue().formula();
+ EXPECT_EQ(getNullability(makePointer(FromNullable, Null), Env),
+ NullabilityKind::Nullable);
+
+ auto *NotNull = &DACtx.arena().makeNot(Null);
+ EXPECT_EQ(getNullability(makePointer(FromNullable, Null), Env, NotNull),
+ NullabilityKind::NonNull);
}
} // namespace