1 | |
2 | |
3 | |
4 | |
5 | |
6 | |
7 | |
8 | |
9 | |
10 | |
11 | |
12 | |
13 | #include "clang/StaticAnalyzer/Frontend/CheckerRegistration.h" |
14 | #include "clang/Basic/Diagnostic.h" |
15 | #include "clang/Frontend/FrontendDiagnostic.h" |
16 | #include "clang/StaticAnalyzer/Core/AnalyzerOptions.h" |
17 | #include "clang/StaticAnalyzer/Core/CheckerManager.h" |
18 | #include "clang/StaticAnalyzer/Frontend/CheckerRegistry.h" |
19 | #include "clang/StaticAnalyzer/Frontend/FrontendActions.h" |
20 | #include "llvm/ADT/SmallVector.h" |
21 | #include "llvm/Support/FormattedStream.h" |
22 | #include "llvm/Support/raw_ostream.h" |
23 | #include <memory> |
24 | |
25 | using namespace clang; |
26 | using namespace ento; |
27 | |
28 | std::unique_ptr<CheckerManager> ento::createCheckerManager( |
29 | ASTContext &context, |
30 | AnalyzerOptions &opts, |
31 | ArrayRef<std::string> plugins, |
32 | ArrayRef<std::function<void(CheckerRegistry &)>> checkerRegistrationFns, |
33 | DiagnosticsEngine &diags) { |
34 | auto checkerMgr = llvm::make_unique<CheckerManager>(context, opts); |
35 | |
36 | CheckerRegistry allCheckers(plugins, diags, opts, context.getLangOpts(), |
37 | checkerRegistrationFns); |
38 | |
39 | allCheckers.initializeManager(*checkerMgr); |
40 | allCheckers.validateCheckerOptions(); |
41 | checkerMgr->finishedCheckerRegistration(); |
42 | |
43 | return checkerMgr; |
44 | } |
45 | |
46 | void ento::printCheckerHelp(raw_ostream &out, ArrayRef<std::string> plugins, |
47 | AnalyzerOptions &anopts, |
48 | DiagnosticsEngine &diags, |
49 | const LangOptions &langOpts) { |
50 | out << "OVERVIEW: Clang Static Analyzer Checkers List\n\n"; |
51 | out << "USAGE: -analyzer-checker <CHECKER or PACKAGE,...>\n\n"; |
52 | |
53 | CheckerRegistry(plugins, diags, anopts, langOpts).printHelp(out); |
54 | } |
55 | |
56 | void ento::printEnabledCheckerList(raw_ostream &out, |
57 | ArrayRef<std::string> plugins, |
58 | AnalyzerOptions &anopts, |
59 | DiagnosticsEngine &diags, |
60 | const LangOptions &langOpts) { |
61 | out << "OVERVIEW: Clang Static Analyzer Enabled Checkers List\n\n"; |
62 | |
63 | CheckerRegistry(plugins, diags, anopts, langOpts).printList(out); |
64 | } |
65 | |
66 | void ento::printAnalyzerConfigList(raw_ostream &out) { |
67 | out << "OVERVIEW: Clang Static Analyzer -analyzer-config Option List\n\n"; |
68 | out << "USAGE: clang -cc1 [CLANG_OPTIONS] -analyzer-config " |
69 | "<OPTION1=VALUE,OPTION2=VALUE,...>\n\n"; |
70 | out << " clang -cc1 [CLANG_OPTIONS] -analyzer-config OPTION1=VALUE, " |
71 | "-analyzer-config OPTION2=VALUE, ...\n\n"; |
72 | out << " clang [CLANG_OPTIONS] -Xclang -analyzer-config -Xclang" |
73 | "<OPTION1=VALUE,OPTION2=VALUE,...>\n\n"; |
74 | out << " clang [CLANG_OPTIONS] -Xclang -analyzer-config -Xclang " |
75 | "OPTION1=VALUE, -Xclang -analyzer-config -Xclang " |
76 | "OPTION2=VALUE, ...\n\n"; |
77 | out << "OPTIONS:\n\n"; |
78 | |
79 | using OptionAndDescriptionTy = std::pair<StringRef, std::string>; |
80 | OptionAndDescriptionTy PrintableOptions[] = { |
81 | #define ANALYZER_OPTION(TYPE, NAME, CMDFLAG, DESC, DEFAULT_VAL) \ |
82 | { \ |
83 | CMDFLAG, \ |
84 | llvm::Twine(llvm::Twine() + "(" + \ |
85 | (StringRef(#TYPE) == "StringRef" ? "string" : #TYPE ) + \ |
86 | ") " DESC \ |
87 | " (default: " #DEFAULT_VAL ")").str() \ |
88 | }, |
89 | |
90 | #define ANALYZER_OPTION_DEPENDS_ON_USER_MODE(TYPE, NAME, CMDFLAG, DESC, \ |
91 | SHALLOW_VAL, DEEP_VAL) \ |
92 | { \ |
93 | CMDFLAG, \ |
94 | llvm::Twine(llvm::Twine() + "(" + \ |
95 | (StringRef(#TYPE) == "StringRef" ? "string" : #TYPE ) + \ |
96 | ") " DESC \ |
97 | " (default: " #SHALLOW_VAL " in shallow mode, " #DEEP_VAL \ |
98 | " in deep mode)").str() \ |
99 | }, |
100 | #include "clang/StaticAnalyzer/Core/AnalyzerOptions.def" |
101 | #undef ANALYZER_OPTION |
102 | #undef ANALYZER_OPTION_DEPENDS_ON_USER_MODE |
103 | }; |
104 | |
105 | llvm::sort(PrintableOptions, [](const OptionAndDescriptionTy &LHS, |
106 | const OptionAndDescriptionTy &RHS) { |
107 | return LHS.first < RHS.first; |
108 | }); |
109 | |
110 | constexpr size_t MinLineWidth = 70; |
111 | constexpr size_t PadForOpt = 2; |
112 | constexpr size_t OptionWidth = 30; |
113 | constexpr size_t PadForDesc = PadForOpt + OptionWidth; |
114 | static_assert(MinLineWidth > PadForDesc, "MinLineWidth must be greater!"); |
115 | |
116 | llvm::formatted_raw_ostream FOut(out); |
117 | |
118 | for (const auto &Pair : PrintableOptions) { |
119 | FOut.PadToColumn(PadForOpt) << Pair.first; |
120 | |
121 | |
122 | if (FOut.getColumn() > PadForDesc) |
123 | FOut << '\n'; |
124 | |
125 | FOut.PadToColumn(PadForDesc); |
126 | |
127 | for (char C : Pair.second) { |
128 | if (FOut.getColumn() > MinLineWidth && C == ' ') { |
129 | FOut << '\n'; |
130 | FOut.PadToColumn(PadForDesc); |
131 | continue; |
132 | } |
133 | FOut << C; |
134 | } |
135 | FOut << "\n\n"; |
136 | } |
137 | } |
138 | |