blob: 54daf2bd5f49fba40910f397450867c5d93c5798 [file] [log] [blame]
Wei Yi Tee8b58e192022-08-02 10:15:40 -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
Googler7f19b2b2023-05-01 09:44:57 -07005#include "nullability/pointer_nullability.h"
Wei Yi Tee8b58e192022-08-02 10:15:40 -07006
Sam McCalla8680262023-09-21 05:55:27 -07007#include <cassert>
8#include <optional>
Dmitri Gribenko742c4c32023-07-31 12:32:09 -07009
Sam McCalla8680262023-09-21 05:55:27 -070010#include "nullability/type_nullability.h"
Dmitri Gribenko742c4c32023-07-31 12:32:09 -070011#include "clang/AST/Type.h"
Sam McCalla8680262023-09-21 05:55:27 -070012#include "clang/Analysis/FlowSensitive/DataflowAnalysisContext.h"
Dani Ferreira Franco Moura92d72bd2023-01-26 04:46:10 -080013#include "clang/Analysis/FlowSensitive/DataflowEnvironment.h"
Googlere1504a62023-07-18 14:03:23 -070014#include "clang/Analysis/FlowSensitive/Formula.h"
Wei Yi Tee8b58e192022-08-02 10:15:40 -070015#include "clang/Analysis/FlowSensitive/Value.h"
Martin Brænne620d0192023-07-31 03:54:40 -070016#include "clang/Basic/LLVM.h"
Sam McCallef6dc922023-06-06 05:50:23 -070017#include "clang/Basic/Specifiers.h"
Wei Yi Tee8b58e192022-08-02 10:15:40 -070018#include "llvm/ADT/StringRef.h"
19
Sam McCall4f6be422023-06-27 02:51:22 -070020namespace clang::tidy::nullability {
Wei Yi Tee8b58e192022-08-02 10:15:40 -070021
22using dataflow::AtomicBoolValue;
23using dataflow::BoolValue;
Sam McCalla8680262023-09-21 05:55:27 -070024using dataflow::DataflowAnalysisContext;
Wei Yi Tee8b58e192022-08-02 10:15:40 -070025using dataflow::Environment;
Sam McCalla8680262023-09-21 05:55:27 -070026using dataflow::Formula;
Wei Yi Tee8b58e192022-08-02 10:15:40 -070027using dataflow::PointerValue;
Martin Brænne6716c462023-05-23 07:32:21 -070028using dataflow::Value;
Wei Yi Tee8b58e192022-08-02 10:15:40 -070029
30/// The nullness information of a pointer is represented by two properties
Sam McCallc003cbd2023-07-17 05:12:18 -070031/// which indicate if its source was nullable, and if its value is null.
32constexpr llvm::StringLiteral kFromNullable = "from_nullable";
Dani Ferreira Franco Mouraa0649822022-11-11 07:55:05 -080033constexpr llvm::StringLiteral kNull = "is_null";
Wei Yi Tee8b58e192022-08-02 10:15:40 -070034
Sam McCall7d9afee2023-06-27 01:43:24 -070035NullabilityKind getNullabilityKind(QualType Type, ASTContext &Ctx) {
Dani Ferreira Franco Moura1d41c112022-12-21 08:43:32 -080036 return Type->getNullability().value_or(NullabilityKind::Unspecified);
Wei Yi Tee9c161612022-08-19 14:20:02 -070037}
38
Sam McCall7d9afee2023-06-27 01:43:24 -070039PointerValue *getPointerValueFromExpr(const Expr *PointerExpr,
40 const Environment &Env) {
Martin Brænne620d0192023-07-31 03:54:40 -070041 return cast_or_null<PointerValue>(Env.getValue(*PointerExpr));
Wei Yi Tee721ee972022-08-11 01:14:54 -070042}
43
Sam McCall7d9afee2023-06-27 01:43:24 -070044bool hasPointerNullState(const dataflow::PointerValue &PointerVal) {
Sam McCallc003cbd2023-07-17 05:12:18 -070045 return PointerVal.getProperty(kFromNullable) != nullptr &&
Googlerf164e032023-05-18 08:38:51 -070046 PointerVal.getProperty(kNull) != nullptr;
47}
48
Sam McCalla8680262023-09-21 05:55:27 -070049PointerNullState getPointerNullState(const PointerValue &PointerVal) {
Sam McCall2b19e532023-09-22 07:10:58 -070050 auto &FromNullable = *cast<BoolValue>(PointerVal.getProperty(kFromNullable));
51 auto &Null = *cast<BoolValue>(PointerVal.getProperty(kNull));
Sam McCalla8680262023-09-21 05:55:27 -070052 return {FromNullable.formula(), Null.formula()};
Wei Yi Tee8b58e192022-08-02 10:15:40 -070053}
54
Sam McCalla8680262023-09-21 05:55:27 -070055static bool tryCreatePointerNullState(PointerValue &PointerVal,
56 dataflow::Arena &A,
57 const Formula *FromNullable = nullptr,
58 const Formula *IsNull = nullptr) {
59 // TODO: for now we assume that we have both nullability properties, or none.
60 // We'll need to relax this when properties can be independently widened away.
61 if (hasPointerNullState(PointerVal)) return false;
62 if (!FromNullable) FromNullable = &A.makeAtomRef(A.makeAtom());
63 if (!IsNull) IsNull = &A.makeAtomRef(A.makeAtom());
64 PointerVal.setProperty(kFromNullable, A.makeBoolValue(*FromNullable));
65 PointerVal.setProperty(kNull, A.makeBoolValue(*IsNull));
66 return true;
67}
68
69void initPointerNullState(PointerValue &PointerVal,
70 DataflowAnalysisContext &Ctx,
71 std::optional<PointerTypeNullability> Source) {
72 auto &A = Ctx.arena();
73 if (tryCreatePointerNullState(PointerVal, A,
74 Source ? &Source->isNullable(A) : nullptr)) {
75 // The symbolic/nonnull check is not needed for correctness, but it avoids
76 // adding meaningless (false => !null) invariant clauses.
77 // TODO: remove this once such clauses are recognized and dropped.
78 if (Source &&
79 (Source->isSymbolic() || Source == NullabilityKind::NonNull)) {
80 const Formula &IsNull = getPointerNullState(PointerVal).IsNull;
81 Ctx.addInvariant(A.makeImplies(Source->isNonnull(A), A.makeNot(IsNull)));
82 }
Wei Yi Tee8b58e192022-08-02 10:15:40 -070083 }
84}
85
Sam McCalla8680262023-09-21 05:55:27 -070086void initNullPointer(PointerValue &PointerVal, DataflowAnalysisContext &Ctx) {
87 tryCreatePointerNullState(PointerVal, Ctx.arena(),
88 /*FromNullable=*/&Ctx.arena().makeLiteral(true),
89 /*IsNull=*/&Ctx.arena().makeLiteral(true));
Wei Yi Tee721ee972022-08-11 01:14:54 -070090}
91
Googlere1504a62023-07-18 14:03:23 -070092bool isNullable(const PointerValue &PointerVal, const Environment &Env,
93 const dataflow::Formula *AdditionalConstraints) {
Sam McCall9b62a402023-07-11 22:24:26 -070094 auto &A = Env.getDataflowAnalysisContext().arena();
Sam McCallc003cbd2023-07-17 05:12:18 -070095 auto [FromNullable, Null] = getPointerNullState(PointerVal);
Sam McCalla8680262023-09-21 05:55:27 -070096 auto *ForseeablyNull = &A.makeAnd(FromNullable, Null);
Googlere1504a62023-07-18 14:03:23 -070097 if (AdditionalConstraints)
98 ForseeablyNull = &A.makeAnd(*AdditionalConstraints, *ForseeablyNull);
99 return !Env.flowConditionImplies(A.makeNot(*ForseeablyNull));
Wei Yi Tee036efdf2022-08-19 14:16:26 -0700100}
101
Sam McCall7d9afee2023-06-27 01:43:24 -0700102NullabilityKind getNullability(const dataflow::PointerValue &PointerVal,
Googlere1504a62023-07-18 14:03:23 -0700103 const dataflow::Environment &Env,
104 const dataflow::Formula *AdditionalConstraints) {
Sam McCall9b62a402023-07-11 22:24:26 -0700105 auto &A = Env.getDataflowAnalysisContext().arena();
Sam McCalla8680262023-09-21 05:55:27 -0700106 auto *Null = &getPointerNullState(PointerVal).IsNull;
Googlere1504a62023-07-18 14:03:23 -0700107 if (AdditionalConstraints) Null = &A.makeAnd(*AdditionalConstraints, *Null);
108 if (Env.flowConditionImplies(A.makeNot(*Null)))
Sam McCallef6dc922023-06-06 05:50:23 -0700109 return NullabilityKind::NonNull;
Googlere1504a62023-07-18 14:03:23 -0700110 return isNullable(PointerVal, Env, AdditionalConstraints)
111 ? NullabilityKind::Nullable
112 : NullabilityKind::Unspecified;
Sam McCallef6dc922023-06-06 05:50:23 -0700113}
114
Sam McCall4f6be422023-06-27 02:51:22 -0700115} // namespace clang::tidy::nullability