34
34
#include " clang/Frontend/FrontendActions.h"
35
35
#include " clang/Lex/Preprocessor.h"
36
36
#include " clang/Tooling/CompilationDatabase.h"
37
+ #include " clang/Tooling/Execution.h"
37
38
#include " clang/Tooling/Tooling.h"
38
39
#include " llvm/Option/Arg.h"
39
40
#include " llvm/Option/ArgList.h"
42
43
#include " llvm/Support/CommandLine.h"
43
44
#include " llvm/Support/FileSystem.h"
44
45
#include " llvm/Support/GlobPattern.h"
46
+ #include " llvm/Support/InitLLVM.h"
45
47
#include " llvm/Support/MemoryBuffer.h"
46
48
#include " llvm/Support/Path.h"
47
49
#include " llvm/Support/ToolOutputFile.h"
@@ -56,108 +58,79 @@ using namespace clang;
56
58
using namespace clang ::tooling;
57
59
using namespace llvm ;
58
60
59
- // Options:
60
-
61
- // Collect the source files.
62
- static cl::list<std::string> SourcePaths (cl::Positional,
63
- cl::desc (" <source0> [... <sourceN>]" ),
64
- cl::OneOrMore);
61
+ static cl::OptionCategory Cat (" pp-trace options" );
65
62
66
63
static cl::opt<std::string> Callbacks (
67
64
" callbacks" , cl::init(" *" ),
68
65
cl::desc(" Comma-separated list of globs describing the list of callbacks "
69
66
" to output. Globs are processed in order of appearance. Globs "
70
67
" with the '-' prefix remove callbacks from the set. e.g. "
71
- " '*,-Macro*'." ));
68
+ " '*,-Macro*'." ),
69
+ cl::cat(Cat));
72
70
73
- // Option to specify the trace output file name.
74
71
static cl::opt<std::string> OutputFileName (
75
- " output" , cl::init(" " ),
76
- cl::desc(" Output trace to the given file name or '-' for stdout." ));
77
-
78
- // Collect all other arguments, which will be passed to the front end.
79
- static cl::list<std::string>
80
- CC1Arguments (cl::ConsumeAfter,
81
- cl::desc (" <arguments to be passed to front end>..." ));
72
+ " output" , cl::init(" -" ),
73
+ cl::desc(" Output trace to the given file name or '-' for stdout." ),
74
+ cl::cat(Cat));
82
75
83
- // Frontend action stuff:
76
+ LLVM_ATTRIBUTE_NORETURN static void error (Twine Message) {
77
+ WithColor::error () << Message << ' \n ' ;
78
+ exit (1 );
79
+ }
84
80
85
81
namespace {
86
- // Consumer is responsible for setting up the callbacks.
87
- class PPTraceConsumer : public ASTConsumer {
88
- public:
89
- PPTraceConsumer (const FilterType &Filters,
90
- std::vector<CallbackCall> &CallbackCalls, Preprocessor &PP) {
91
- // PP takes ownership.
92
- PP.addPPCallbacks (
93
- llvm::make_unique<PPCallbacksTracker>(Filters, CallbackCalls, PP));
94
- }
95
- };
96
82
97
- class PPTraceAction : public SyntaxOnlyAction {
83
+ class PPTraceAction : public ASTFrontendAction {
98
84
public:
99
- PPTraceAction (const FilterType &Filters,
100
- std::vector<CallbackCall> &CallbackCalls)
101
- : Filters(Filters), CallbackCalls(CallbackCalls) {}
85
+ PPTraceAction (const FilterType &Filters, raw_ostream &OS)
86
+ : Filters(Filters), OS(OS) {}
102
87
103
88
protected:
104
89
std::unique_ptr<clang::ASTConsumer>
105
90
CreateASTConsumer (CompilerInstance &CI, StringRef InFile) override {
106
- return llvm::make_unique<PPTraceConsumer>(Filters, CallbackCalls,
107
- CI.getPreprocessor ());
91
+ Preprocessor &PP = CI.getPreprocessor ();
92
+ PP.addPPCallbacks (
93
+ make_unique<PPCallbacksTracker>(Filters, CallbackCalls, PP));
94
+ return make_unique<ASTConsumer>();
95
+ }
96
+
97
+ void EndSourceFileAction () override {
98
+ OS << " ---\n " ;
99
+ for (const CallbackCall &Callback : CallbackCalls) {
100
+ OS << " - Callback: " << Callback.Name << " \n " ;
101
+ for (const Argument &Arg : Callback.Arguments )
102
+ OS << " " << Arg.Name << " : " << Arg.Value << " \n " ;
103
+ }
104
+ OS << " ...\n " ;
105
+
106
+ CallbackCalls.clear ();
108
107
}
109
108
110
109
private:
111
110
const FilterType &Filters;
112
- std::vector<CallbackCall> &CallbackCalls;
111
+ raw_ostream &OS;
112
+ std::vector<CallbackCall> CallbackCalls;
113
113
};
114
114
115
115
class PPTraceFrontendActionFactory : public FrontendActionFactory {
116
116
public:
117
- PPTraceFrontendActionFactory (const FilterType &Filters,
118
- std::vector<CallbackCall> &CallbackCalls)
119
- : Filters(Filters), CallbackCalls(CallbackCalls) {}
117
+ PPTraceFrontendActionFactory (const FilterType &Filters, raw_ostream &OS)
118
+ : Filters(Filters), OS(OS) {}
120
119
121
- PPTraceAction *create () override {
122
- return new PPTraceAction (Filters, CallbackCalls);
123
- }
120
+ PPTraceAction *create () override { return new PPTraceAction (Filters, OS); }
124
121
125
122
private:
126
123
const FilterType &Filters;
127
- std::vector<CallbackCall> &CallbackCalls ;
124
+ raw_ostream &OS ;
128
125
};
129
126
} // namespace
130
127
131
- // Output the trace given its data structure and a stream.
132
- static int outputPPTrace (std::vector<CallbackCall> &CallbackCalls,
133
- llvm::raw_ostream &OS) {
134
- // Mark start of document.
135
- OS << " ---\n " ;
136
-
137
- for (std::vector<CallbackCall>::const_iterator I = CallbackCalls.begin (),
138
- E = CallbackCalls.end ();
139
- I != E; ++I) {
140
- const CallbackCall &Callback = *I;
141
- OS << " - Callback: " << Callback.Name << " \n " ;
142
-
143
- for (auto AI = Callback.Arguments .begin (), AE = Callback.Arguments .end ();
144
- AI != AE; ++AI) {
145
- const Argument &Arg = *AI;
146
- OS << " " << Arg.Name << " : " << Arg.Value << " \n " ;
147
- }
148
- }
149
-
150
- // Mark end of document.
151
- OS << " ...\n " ;
152
-
153
- return 0 ;
154
- }
155
-
156
- // Program entry point.
157
- int main (int Argc, const char **Argv) {
128
+ int main (int argc, const char **argv) {
129
+ InitLLVM X (argc, argv);
158
130
159
- // Parse command line.
160
- cl::ParseCommandLineOptions (Argc, Argv, " pp-trace.\n " );
131
+ auto Exec = tooling::createExecutorFromCommandLineArgs (argc, argv, Cat);
132
+ if (!Exec)
133
+ error (toString (Exec.takeError ()));
161
134
162
135
// Parse the IgnoreCallbacks list into strings.
163
136
SmallVector<StringRef, 32 > Patterns;
@@ -169,51 +142,18 @@ int main(int Argc, const char **Argv) {
169
142
bool Enabled = !Pattern.consume_front (" -" );
170
143
if (Expected<GlobPattern> Pat = GlobPattern::create (Pattern))
171
144
Filters.emplace_back (std::move (*Pat), Enabled);
172
- else {
173
- WithColor::error (llvm::errs (), " pp-trace" )
174
- << toString (Pat.takeError ()) << ' \n ' ;
175
- return 1 ;
176
- }
145
+ else
146
+ error (toString (Pat.takeError ()));
177
147
}
178
148
179
- // Create the compilation database.
180
- SmallString<256 > PathBuf;
181
- sys::fs::current_path (PathBuf);
182
- std::unique_ptr<CompilationDatabase> Compilations;
183
- Compilations.reset (
184
- new FixedCompilationDatabase (Twine (PathBuf), CC1Arguments));
149
+ std::error_code EC;
150
+ llvm::ToolOutputFile Out (OutputFileName, EC, llvm::sys::fs::F_Text);
151
+ if (EC)
152
+ error (EC.message ());
185
153
186
- // Store the callback trace information here.
187
- std::vector<CallbackCall> CallbackCalls;
188
-
189
- // Create the tool and run the compilation.
190
- ClangTool Tool (*Compilations, SourcePaths);
191
- PPTraceFrontendActionFactory Factory (Filters, CallbackCalls);
192
- int HadErrors = Tool.run (&Factory);
193
-
194
- // If we had errors, exit early.
195
- if (HadErrors)
196
- return HadErrors;
197
-
198
- // Do the output.
199
- if (!OutputFileName.size ()) {
200
- HadErrors = outputPPTrace (CallbackCalls, llvm::outs ());
201
- } else {
202
- // Set up output file.
203
- std::error_code EC;
204
- llvm::ToolOutputFile Out (OutputFileName, EC, llvm::sys::fs::F_Text);
205
- if (EC) {
206
- llvm::errs () << " pp-trace: error creating " << OutputFileName << " :"
207
- << EC.message () << " \n " ;
208
- return 1 ;
209
- }
210
-
211
- HadErrors = outputPPTrace (CallbackCalls, Out.os ());
212
-
213
- // Tell ToolOutputFile that we want to keep the file.
214
- if (HadErrors == 0 )
215
- Out.keep ();
216
- }
217
-
218
- return HadErrors;
154
+ if (Error Err = Exec->get ()->execute (
155
+ make_unique<PPTraceFrontendActionFactory>(Filters, Out.os ())))
156
+ error (toString (std::move (Err)));
157
+ Out.keep ();
158
+ return 0 ;
219
159
}
0 commit comments