Wei Yi Tee | 8b58e19 | 2022-08-02 10:15:40 -0700 | [diff] [blame] | 1 | // 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 | |
Googler | 7f19b2b | 2023-05-01 09:44:57 -0700 | [diff] [blame] | 5 | #include "nullability/pointer_nullability.h" |
Wei Yi Tee | 8b58e19 | 2022-08-02 10:15:40 -0700 | [diff] [blame] | 6 | |
Sam McCall | a868026 | 2023-09-21 05:55:27 -0700 | [diff] [blame] | 7 | #include <cassert> |
| 8 | #include <optional> |
Dmitri Gribenko | 742c4c3 | 2023-07-31 12:32:09 -0700 | [diff] [blame] | 9 | |
Sam McCall | a868026 | 2023-09-21 05:55:27 -0700 | [diff] [blame] | 10 | #include "nullability/type_nullability.h" |
Dmitri Gribenko | 742c4c3 | 2023-07-31 12:32:09 -0700 | [diff] [blame] | 11 | #include "clang/AST/Type.h" |
Sam McCall | a868026 | 2023-09-21 05:55:27 -0700 | [diff] [blame] | 12 | #include "clang/Analysis/FlowSensitive/DataflowAnalysisContext.h" |
Dani Ferreira Franco Moura | 92d72bd | 2023-01-26 04:46:10 -0800 | [diff] [blame] | 13 | #include "clang/Analysis/FlowSensitive/DataflowEnvironment.h" |
Googler | e1504a6 | 2023-07-18 14:03:23 -0700 | [diff] [blame] | 14 | #include "clang/Analysis/FlowSensitive/Formula.h" |
Wei Yi Tee | 8b58e19 | 2022-08-02 10:15:40 -0700 | [diff] [blame] | 15 | #include "clang/Analysis/FlowSensitive/Value.h" |
Martin Brænne | 620d019 | 2023-07-31 03:54:40 -0700 | [diff] [blame] | 16 | #include "clang/Basic/LLVM.h" |
Sam McCall | ef6dc92 | 2023-06-06 05:50:23 -0700 | [diff] [blame] | 17 | #include "clang/Basic/Specifiers.h" |
Wei Yi Tee | 8b58e19 | 2022-08-02 10:15:40 -0700 | [diff] [blame] | 18 | #include "llvm/ADT/StringRef.h" |
| 19 | |
Sam McCall | 4f6be42 | 2023-06-27 02:51:22 -0700 | [diff] [blame] | 20 | namespace clang::tidy::nullability { |
Wei Yi Tee | 8b58e19 | 2022-08-02 10:15:40 -0700 | [diff] [blame] | 21 | |
| 22 | using dataflow::AtomicBoolValue; |
| 23 | using dataflow::BoolValue; |
Sam McCall | a868026 | 2023-09-21 05:55:27 -0700 | [diff] [blame] | 24 | using dataflow::DataflowAnalysisContext; |
Wei Yi Tee | 8b58e19 | 2022-08-02 10:15:40 -0700 | [diff] [blame] | 25 | using dataflow::Environment; |
Sam McCall | a868026 | 2023-09-21 05:55:27 -0700 | [diff] [blame] | 26 | using dataflow::Formula; |
Wei Yi Tee | 8b58e19 | 2022-08-02 10:15:40 -0700 | [diff] [blame] | 27 | using dataflow::PointerValue; |
Martin Brænne | 6716c46 | 2023-05-23 07:32:21 -0700 | [diff] [blame] | 28 | using dataflow::Value; |
Wei Yi Tee | 8b58e19 | 2022-08-02 10:15:40 -0700 | [diff] [blame] | 29 | |
| 30 | /// The nullness information of a pointer is represented by two properties |
Sam McCall | c003cbd | 2023-07-17 05:12:18 -0700 | [diff] [blame] | 31 | /// which indicate if its source was nullable, and if its value is null. |
| 32 | constexpr llvm::StringLiteral kFromNullable = "from_nullable"; |
Dani Ferreira Franco Moura | a064982 | 2022-11-11 07:55:05 -0800 | [diff] [blame] | 33 | constexpr llvm::StringLiteral kNull = "is_null"; |
Wei Yi Tee | 8b58e19 | 2022-08-02 10:15:40 -0700 | [diff] [blame] | 34 | |
Sam McCall | 7d9afee | 2023-06-27 01:43:24 -0700 | [diff] [blame] | 35 | NullabilityKind getNullabilityKind(QualType Type, ASTContext &Ctx) { |
Dani Ferreira Franco Moura | 1d41c11 | 2022-12-21 08:43:32 -0800 | [diff] [blame] | 36 | return Type->getNullability().value_or(NullabilityKind::Unspecified); |
Wei Yi Tee | 9c16161 | 2022-08-19 14:20:02 -0700 | [diff] [blame] | 37 | } |
| 38 | |
Sam McCall | 7d9afee | 2023-06-27 01:43:24 -0700 | [diff] [blame] | 39 | PointerValue *getPointerValueFromExpr(const Expr *PointerExpr, |
| 40 | const Environment &Env) { |
Martin Brænne | 620d019 | 2023-07-31 03:54:40 -0700 | [diff] [blame] | 41 | return cast_or_null<PointerValue>(Env.getValue(*PointerExpr)); |
Wei Yi Tee | 721ee97 | 2022-08-11 01:14:54 -0700 | [diff] [blame] | 42 | } |
| 43 | |
Sam McCall | 7d9afee | 2023-06-27 01:43:24 -0700 | [diff] [blame] | 44 | bool hasPointerNullState(const dataflow::PointerValue &PointerVal) { |
Sam McCall | c003cbd | 2023-07-17 05:12:18 -0700 | [diff] [blame] | 45 | return PointerVal.getProperty(kFromNullable) != nullptr && |
Googler | f164e03 | 2023-05-18 08:38:51 -0700 | [diff] [blame] | 46 | PointerVal.getProperty(kNull) != nullptr; |
| 47 | } |
| 48 | |
Sam McCall | a868026 | 2023-09-21 05:55:27 -0700 | [diff] [blame] | 49 | PointerNullState getPointerNullState(const PointerValue &PointerVal) { |
Sam McCall | 2b19e53 | 2023-09-22 07:10:58 -0700 | [diff] [blame^] | 50 | auto &FromNullable = *cast<BoolValue>(PointerVal.getProperty(kFromNullable)); |
| 51 | auto &Null = *cast<BoolValue>(PointerVal.getProperty(kNull)); |
Sam McCall | a868026 | 2023-09-21 05:55:27 -0700 | [diff] [blame] | 52 | return {FromNullable.formula(), Null.formula()}; |
Wei Yi Tee | 8b58e19 | 2022-08-02 10:15:40 -0700 | [diff] [blame] | 53 | } |
| 54 | |
Sam McCall | a868026 | 2023-09-21 05:55:27 -0700 | [diff] [blame] | 55 | static 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 | |
| 69 | void 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 Tee | 8b58e19 | 2022-08-02 10:15:40 -0700 | [diff] [blame] | 83 | } |
| 84 | } |
| 85 | |
Sam McCall | a868026 | 2023-09-21 05:55:27 -0700 | [diff] [blame] | 86 | void initNullPointer(PointerValue &PointerVal, DataflowAnalysisContext &Ctx) { |
| 87 | tryCreatePointerNullState(PointerVal, Ctx.arena(), |
| 88 | /*FromNullable=*/&Ctx.arena().makeLiteral(true), |
| 89 | /*IsNull=*/&Ctx.arena().makeLiteral(true)); |
Wei Yi Tee | 721ee97 | 2022-08-11 01:14:54 -0700 | [diff] [blame] | 90 | } |
| 91 | |
Googler | e1504a6 | 2023-07-18 14:03:23 -0700 | [diff] [blame] | 92 | bool isNullable(const PointerValue &PointerVal, const Environment &Env, |
| 93 | const dataflow::Formula *AdditionalConstraints) { |
Sam McCall | 9b62a40 | 2023-07-11 22:24:26 -0700 | [diff] [blame] | 94 | auto &A = Env.getDataflowAnalysisContext().arena(); |
Sam McCall | c003cbd | 2023-07-17 05:12:18 -0700 | [diff] [blame] | 95 | auto [FromNullable, Null] = getPointerNullState(PointerVal); |
Sam McCall | a868026 | 2023-09-21 05:55:27 -0700 | [diff] [blame] | 96 | auto *ForseeablyNull = &A.makeAnd(FromNullable, Null); |
Googler | e1504a6 | 2023-07-18 14:03:23 -0700 | [diff] [blame] | 97 | if (AdditionalConstraints) |
| 98 | ForseeablyNull = &A.makeAnd(*AdditionalConstraints, *ForseeablyNull); |
| 99 | return !Env.flowConditionImplies(A.makeNot(*ForseeablyNull)); |
Wei Yi Tee | 036efdf | 2022-08-19 14:16:26 -0700 | [diff] [blame] | 100 | } |
| 101 | |
Sam McCall | 7d9afee | 2023-06-27 01:43:24 -0700 | [diff] [blame] | 102 | NullabilityKind getNullability(const dataflow::PointerValue &PointerVal, |
Googler | e1504a6 | 2023-07-18 14:03:23 -0700 | [diff] [blame] | 103 | const dataflow::Environment &Env, |
| 104 | const dataflow::Formula *AdditionalConstraints) { |
Sam McCall | 9b62a40 | 2023-07-11 22:24:26 -0700 | [diff] [blame] | 105 | auto &A = Env.getDataflowAnalysisContext().arena(); |
Sam McCall | a868026 | 2023-09-21 05:55:27 -0700 | [diff] [blame] | 106 | auto *Null = &getPointerNullState(PointerVal).IsNull; |
Googler | e1504a6 | 2023-07-18 14:03:23 -0700 | [diff] [blame] | 107 | if (AdditionalConstraints) Null = &A.makeAnd(*AdditionalConstraints, *Null); |
| 108 | if (Env.flowConditionImplies(A.makeNot(*Null))) |
Sam McCall | ef6dc92 | 2023-06-06 05:50:23 -0700 | [diff] [blame] | 109 | return NullabilityKind::NonNull; |
Googler | e1504a6 | 2023-07-18 14:03:23 -0700 | [diff] [blame] | 110 | return isNullable(PointerVal, Env, AdditionalConstraints) |
| 111 | ? NullabilityKind::Nullable |
| 112 | : NullabilityKind::Unspecified; |
Sam McCall | ef6dc92 | 2023-06-06 05:50:23 -0700 | [diff] [blame] | 113 | } |
| 114 | |
Sam McCall | 4f6be42 | 2023-06-27 02:51:22 -0700 | [diff] [blame] | 115 | } // namespace clang::tidy::nullability |