Clang Project

clang_source_code/lib/StaticAnalyzer/Checkers/MPI-Checker/MPIBugReporter.h
1//===-- MPIBugReporter.h - bug reporter -----------------------*- C++ -*-===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8///
9/// \file
10/// This file defines prefabricated reports which are emitted in
11/// case of MPI related bugs, detected by path-sensitive analysis.
12///
13//===----------------------------------------------------------------------===//
14
15#ifndef LLVM_CLANG_LIB_STATICANALYZER_CHECKERS_MPICHECKER_MPIBUGREPORTER_H
16#define LLVM_CLANG_LIB_STATICANALYZER_CHECKERS_MPICHECKER_MPIBUGREPORTER_H
17
18#include "MPITypes.h"
19#include "clang/StaticAnalyzer/Core/BugReporter/BugType.h"
20
21namespace clang {
22namespace ento {
23namespace mpi {
24
25class MPIBugReporter {
26public:
27  MPIBugReporter(const CheckerBase &CB) {
28    UnmatchedWaitBugType.reset(new BugType(&CB, "Unmatched wait", MPIError));
29    DoubleNonblockingBugType.reset(
30        new BugType(&CB, "Double nonblocking", MPIError));
31    MissingWaitBugType.reset(new BugType(&CB, "Missing wait", MPIError));
32  }
33
34  /// Report duplicate request use by nonblocking calls without intermediate
35  /// wait.
36  ///
37  /// \param MPICallEvent MPI call that caused the double nonblocking
38  /// \param Req request that was used by two nonblocking calls in sequence
39  /// \param RequestRegion memory region of the request
40  /// \param ExplNode node in the graph the bug appeared at
41  /// \param BReporter bug reporter for current context
42  void reportDoubleNonblocking(const CallEvent &MPICallEvent,
43                               const Request &Req,
44                               const MemRegion *const RequestRegion,
45                               const ExplodedNode *const ExplNode,
46                              BugReporter &BReporterconst;
47
48  /// Report a missing wait for a nonblocking call.
49  ///
50  /// \param Req request that is not matched by a wait
51  /// \param RequestRegion memory region of the request
52  /// \param ExplNode node in the graph the bug appeared at
53  /// \param BReporter bug reporter for current context
54  void reportMissingWait(const Request &Req,
55                         const MemRegion *const RequestRegion,
56                         const ExplodedNode *const ExplNode,
57                         BugReporter &BReporterconst;
58
59  /// Report a wait on a request that has not been used at all before.
60  ///
61  /// \param CE wait call that uses the request
62  /// \param RequestRegion memory region of the request
63  /// \param ExplNode node in the graph the bug appeared at
64  /// \param BReporter bug reporter for current context
65  void reportUnmatchedWait(const CallEvent &CE,
66                           const MemRegion *const RequestRegion,
67                           const ExplodedNode *const ExplNode,
68                           BugReporter &BReporterconst;
69
70private:
71  const std::string MPIError = "MPI Error";
72
73  // path-sensitive bug types
74  std::unique_ptr<BugTypeUnmatchedWaitBugType;
75  std::unique_ptr<BugTypeMissingWaitBugType;
76  std::unique_ptr<BugTypeDoubleNonblockingBugType;
77
78  /// Bug visitor class to find the node where the request region was previously
79  /// used in order to include it into the BugReport path.
80  class RequestNodeVisitor : public BugReporterVisitor {
81  public:
82    RequestNodeVisitor(const MemRegion *const MemoryRegion,
83                       const std::string &ErrText)
84        : RequestRegion(MemoryRegion), ErrorText(ErrText) {}
85
86    void Profile(llvm::FoldingSetNodeID &IDconst override {
87      static int X = 0;
88      ID.AddPointer(&X);
89      ID.AddPointer(RequestRegion);
90    }
91
92    std::shared_ptr<PathDiagnosticPieceVisitNode(const ExplodedNode *N,
93                                                   BugReporterContext &BRC,
94                                                   BugReport &BR) override;
95
96  private:
97    const MemRegion *const RequestRegion;
98    bool IsNodeFound = false;
99    std::string ErrorText;
100  };
101};
102
103// end of namespace: mpi
104// end of namespace: ento
105// end of namespace: clang
106
107#endif
108
clang::ento::mpi::MPIBugReporter::reportDoubleNonblocking
clang::ento::mpi::MPIBugReporter::reportMissingWait
clang::ento::mpi::MPIBugReporter::reportUnmatchedWait
clang::ento::mpi::MPIBugReporter::MPIError
clang::ento::mpi::MPIBugReporter::UnmatchedWaitBugType
clang::ento::mpi::MPIBugReporter::MissingWaitBugType
clang::ento::mpi::MPIBugReporter::DoubleNonblockingBugType
clang::ento::mpi::MPIBugReporter::RequestNodeVisitor
clang::ento::mpi::MPIBugReporter::RequestNodeVisitor::Profile
clang::ento::mpi::MPIBugReporter::RequestNodeVisitor::VisitNode
clang::ento::mpi::MPIBugReporter::RequestNodeVisitor::RequestRegion
clang::ento::mpi::MPIBugReporter::RequestNodeVisitor::IsNodeFound
clang::ento::mpi::MPIBugReporter::RequestNodeVisitor::ErrorText