| // 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 |
| |
| #ifndef CRUBIT_NULLABILITY_INFERENCE_COLLECT_EVIDENCE_H_ |
| #define CRUBIT_NULLABILITY_INFERENCE_COLLECT_EVIDENCE_H_ |
| |
| #include <string> |
| #include <string_view> |
| |
| #include "nullability/inference/inference.proto.h" |
| #include "nullability/inference/slot_fingerprint.h" |
| #include "nullability/pointer_nullability_analysis.h" |
| #include "nullability/pragma.h" |
| #include "clang/AST/DeclBase.h" |
| #include "clang/Analysis/FlowSensitive/Solver.h" |
| #include "clang/Basic/SourceLocation.h" |
| #include "llvm/ADT/DenseMap.h" |
| #include "llvm/ADT/DenseSet.h" |
| #include "llvm/ADT/FunctionExtras.h" |
| #include "llvm/ADT/STLFunctionalExtras.h" |
| #include "llvm/Support/Error.h" |
| |
| namespace clang::tidy::nullability { |
| |
| using USRCache = llvm::DenseMap<const Decl *, std::string>; |
| |
| std::string_view getOrGenerateUSR(USRCache &Cache, const Decl &); |
| |
| /// Callback used to report collected nullability evidence. |
| using EvidenceEmitter = void(const Decl &Target, Slot, Evidence::Kind, |
| SourceLocation); |
| /// Creates an EvidenceEmitter that serializes the evidence as Evidence protos. |
| /// This emitter caches USR generation, and should be reused for the whole AST. |
| llvm::unique_function<EvidenceEmitter> evidenceEmitter( |
| llvm::unique_function<void(const Evidence &) const>, USRCache &USRCache); |
| |
| struct PreviousInferences { |
| const llvm::DenseSet<SlotFingerprint> &Nullable = {}; |
| const llvm::DenseSet<SlotFingerprint> &Nonnull = {}; |
| }; |
| |
| /// Creates a solver with default parameters that is suitable for passing to |
| /// `collectEvidenceFromDefinition()`. |
| std::unique_ptr<dataflow::Solver> makeDefaultSolverForInference(); |
| |
| /// Analyze code (such as a function body or variable initializer) to infer |
| /// nullability. |
| /// |
| /// Produces Evidence constraining the nullability slots of the symbols that |
| /// the code interacts with, such as the function's own parameters. |
| /// This is based on the code's behavior and our definition of null-safety. |
| /// |
| /// It is up to the caller to ensure the definition is eligible for inference |
| /// (function has a body, is not dependent, etc). |
| llvm::Error collectEvidenceFromDefinition( |
| const Decl &, llvm::function_ref<EvidenceEmitter>, USRCache &USRCache, |
| const NullabilityPragmas &Pragmas, |
| PreviousInferences PreviousInferences = {}, |
| const SolverFactory &MakeSolver = makeDefaultSolverForInference); |
| |
| /// Gathers evidence of a symbol's nullability from a declaration of it. |
| /// |
| /// These are trivial "inferences" of what's already written in the code. e.g: |
| /// void foo(Nullable<int*>); |
| /// The first parameter of foo must be nullable. |
| /// |
| /// It is the caller's responsibility to ensure that the symbol is inferable. |
| void collectEvidenceFromTargetDeclaration(const clang::Decl &, |
| llvm::function_ref<EvidenceEmitter>, |
| const NullabilityPragmas &Pragmas); |
| |
| /// Describes locations within an AST that provide evidence for use in |
| /// inference. |
| struct EvidenceSites { |
| /// Declarations of inferable symbols. |
| llvm::DenseSet<const Decl *> Declarations; |
| /// Definitions (e.g. function body, variable initializer) that can be |
| /// analyzed. |
| /// This will always be concrete code, not a template pattern. These may be |
| /// passed to collectEvidenceFromDefinition(). |
| llvm::DenseSet<const Decl *> Definitions; |
| |
| /// Find the evidence sites within the provided AST. |
| static EvidenceSites discover(ASTContext &); |
| }; |
| |
| /// Returns the slot number for the I'th parameter (0-based). |
| inline Slot paramSlot(unsigned I) { return static_cast<Slot>(SLOT_PARAM + I); } |
| |
| } // namespace clang::tidy::nullability |
| |
| #endif // CRUBIT_NULLABILITY_INFERENCE_COLLECT_EVIDENCE_H_ |