blob: 3e54939eb057a99691321ce28cc7ee70225681ab [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/pointer_nullability.h"
#include <memory>
#include "nullability/type_nullability.h"
#include "clang/AST/Type.h"
#include "clang/Analysis/FlowSensitive/DataflowAnalysisContext.h"
#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 {
class NullabilityPropertiesTest : public ::testing::Test {
protected:
dataflow::PointerValue &makePointer(PointerTypeNullability N) {
auto &P = Env.create<dataflow::PointerValue>(
DACtx.createStorageLocation(QualType()));
initPointerNullState(P, DACtx, N);
return P;
}
dataflow::PointerValue &makeNullPointer() {
auto &P = Env.create<dataflow::PointerValue>(
DACtx.createStorageLocation(QualType()));
initNullPointer(P, DACtx);
return P;
}
dataflow::DataflowAnalysisContext DACtx = dataflow::DataflowAnalysisContext(
std::make_unique<dataflow::WatchedLiteralsSolver>());
dataflow::Environment Env = dataflow::Environment(DACtx);
};
TEST_F(NullabilityPropertiesTest, Test) {
auto &A = DACtx.arena();
EXPECT_TRUE(isNullable(makeNullPointer(), Env));
{
auto &NullableButNotNull = makePointer(NullabilityKind::Nullable);
EXPECT_TRUE(isNullable(NullableButNotNull, Env));
Env.addToFlowCondition(
A.makeNot(getPointerNullState(NullableButNotNull).IsNull));
EXPECT_FALSE(isNullable(NullableButNotNull, Env));
}
{
auto &NullableAndNull = makePointer(NullabilityKind::Nullable);
Env.addToFlowCondition(getPointerNullState(NullableAndNull).IsNull);
EXPECT_TRUE(isNullable(NullableAndNull, Env));
}
{
auto &NonnullAndNotNull = makePointer(NullabilityKind::NonNull);
EXPECT_FALSE(isNullable(NonnullAndNotNull, Env));
Env.addToFlowCondition(
A.makeNot(getPointerNullState(NonnullAndNotNull).IsNull));
EXPECT_FALSE(isNullable(NonnullAndNotNull, Env));
}
{
// This is a little surprising: if a pointer comes from a non-null source
// but is dynamically discovered to be definitely null, we still don't
// consider it nullable.
auto &NonnullAndNull = makePointer(NullabilityKind::NonNull);
Env.addToFlowCondition(getPointerNullState(NonnullAndNull).IsNull);
EXPECT_FALSE(isNullable(NonnullAndNull, Env));
}
}
TEST_F(NullabilityPropertiesTest, IsNullableAdditionalConstraints) {
auto &P = makePointer(NullabilityKind::Nullable);
EXPECT_TRUE(isNullable(P, Env));
auto *NotNull = &DACtx.arena().makeNot(getPointerNullState(P).IsNull);
EXPECT_FALSE(isNullable(P, Env, NotNull));
}
TEST_F(NullabilityPropertiesTest, GetNullabilityAdditionalConstraints) {
auto &P = makePointer(NullabilityKind::Nullable);
EXPECT_EQ(getNullability(P, Env), NullabilityKind::Nullable);
auto *NotNull = &DACtx.arena().makeNot(getPointerNullState(P).IsNull);
EXPECT_EQ(getNullability(P, Env, NotNull), NullabilityKind::NonNull);
}
} // namespace
} // namespace clang::tidy::nullability