blob: 5bfe1860779d53c71c963fbe7706b8ca4d125b02 [file] [log] [blame]
John Thompson99794542013-10-31 12:23:32 +00001//===--- tools/pp-trace/PPTrace.cpp - Clang preprocessor tracer -----------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// This file implements pp-trace, a tool for displaying a textual trace
11// of the Clang preprocessor activity. It's based on a derivation of the
12// PPCallbacks class, that once registerd with Clang, receives callback calls
13// to its virtual members, and outputs the information passed to the callbacks
14// in a high-level YAML format.
15//
16// The pp-trace tool also serves as the basis for a test of the PPCallbacks
17// mechanism.
18//
19// The pp-trace tool supports the following general command line format:
20//
John Thompson17c4f7d2013-10-31 12:53:42 +000021// pp-trace [pp-trace options] (source file) [compiler options]
John Thompson99794542013-10-31 12:23:32 +000022//
23// Basically you put the pp-trace options first, then the source file or files,
24// and then any options you want to pass to the compiler.
25//
26// These are the pp-trace options:
27//
28// -ignore (callback list) Don't display output for a comma-separated
29// list of callbacks, i.e.:
30// -ignore "FileChanged,InclusionDirective"
31//
32// -output (file) Output trace to the given file in a YAML
33// format, e.g.:
34//
35// ---
36// - Callback: Name
37// Argument1: Value1
38// Argument2: Value2
39// (etc.)
40// ...
41//
42// Future Directions:
43//
44// 1. Add option opposite to "-ignore" that specifys a comma-separated option
45// list of callbacs. Perhaps "-only" or "-exclusive".
46//
47//===----------------------------------------------------------------------===//
48
49#include "PPCallbacksTracker.h"
50#include "clang/AST/ASTConsumer.h"
51#include "clang/AST/ASTContext.h"
52#include "clang/AST/RecursiveASTVisitor.h"
53#include "clang/Basic/SourceManager.h"
54#include "clang/Driver/Options.h"
55#include "clang/Frontend/CompilerInstance.h"
56#include "clang/Frontend/FrontendActions.h"
57#include "clang/Lex/Preprocessor.h"
58#include "clang/Tooling/CompilationDatabase.h"
59#include "clang/Tooling/Tooling.h"
John Thompson99794542013-10-31 12:23:32 +000060#include "llvm/Option/Arg.h"
61#include "llvm/Option/ArgList.h"
62#include "llvm/Option/OptTable.h"
63#include "llvm/Option/Option.h"
64#include "llvm/Support/CommandLine.h"
65#include "llvm/Support/FileSystem.h"
66#include "llvm/Support/MemoryBuffer.h"
67#include "llvm/Support/Path.h"
68#include "llvm/Support/ToolOutputFile.h"
69#include <algorithm>
70#include <fstream>
71#include <iterator>
72#include <string>
73#include <vector>
74
75using namespace clang;
76using namespace clang::driver;
77using namespace clang::driver::options;
78using namespace clang::tooling;
79using namespace llvm;
80using namespace llvm::opt;
81
Adrian Prantlb97f5c12015-02-25 02:46:37 +000082// Options:
83
84// Collect the source files.
Benjamin Kramere7103712015-03-23 12:49:15 +000085static cl::list<std::string> SourcePaths(cl::Positional,
86 cl::desc("<source0> [... <sourceN>]"),
87 cl::OneOrMore);
Adrian Prantlb97f5c12015-02-25 02:46:37 +000088
89// Option to specify a list or one or more callback names to ignore.
Benjamin Kramere7103712015-03-23 12:49:15 +000090static cl::opt<std::string> IgnoreCallbacks(
Adrian Prantlb97f5c12015-02-25 02:46:37 +000091 "ignore", cl::init(""),
92 cl::desc("Ignore callbacks, i.e. \"Callback1, Callback2...\"."));
93
94// Option to specify the trace output file name.
Benjamin Kramere7103712015-03-23 12:49:15 +000095static cl::opt<std::string> OutputFileName(
Adrian Prantlb97f5c12015-02-25 02:46:37 +000096 "output", cl::init(""),
97 cl::desc("Output trace to the given file name or '-' for stdout."));
98
99// Collect all other arguments, which will be passed to the front end.
Benjamin Kramere7103712015-03-23 12:49:15 +0000100static cl::list<std::string>
101 CC1Arguments(cl::ConsumeAfter,
102 cl::desc("<arguments to be passed to front end>..."));
Adrian Prantlb97f5c12015-02-25 02:46:37 +0000103
John Thompson99794542013-10-31 12:23:32 +0000104// Frontend action stuff:
105
Benjamin Kramere7103712015-03-23 12:49:15 +0000106namespace {
John Thompson99794542013-10-31 12:23:32 +0000107// Consumer is responsible for setting up the callbacks.
108class PPTraceConsumer : public ASTConsumer {
109public:
110 PPTraceConsumer(SmallSet<std::string, 4> &Ignore,
111 std::vector<CallbackCall> &CallbackCalls, Preprocessor &PP) {
112 // PP takes ownership.
Craig Topper775862a2014-09-10 05:07:57 +0000113 PP.addPPCallbacks(llvm::make_unique<PPCallbacksTracker>(Ignore,
114 CallbackCalls, PP));
John Thompson99794542013-10-31 12:23:32 +0000115 }
John Thompson99794542013-10-31 12:23:32 +0000116};
117
118class PPTraceAction : public SyntaxOnlyAction {
119public:
120 PPTraceAction(SmallSet<std::string, 4> &Ignore,
121 std::vector<CallbackCall> &CallbackCalls)
122 : Ignore(Ignore), CallbackCalls(CallbackCalls) {}
123
124protected:
David Blaikie680c4c82014-08-10 19:56:59 +0000125 std::unique_ptr<clang::ASTConsumer>
126 CreateASTConsumer(CompilerInstance &CI, StringRef InFile) override {
127 return llvm::make_unique<PPTraceConsumer>(Ignore, CallbackCalls,
128 CI.getPreprocessor());
John Thompson99794542013-10-31 12:23:32 +0000129 }
130
131private:
132 SmallSet<std::string, 4> &Ignore;
133 std::vector<CallbackCall> &CallbackCalls;
134};
135
136class PPTraceFrontendActionFactory : public FrontendActionFactory {
137public:
138 PPTraceFrontendActionFactory(SmallSet<std::string, 4> &Ignore,
139 std::vector<CallbackCall> &CallbackCalls)
140 : Ignore(Ignore), CallbackCalls(CallbackCalls) {}
141
142 virtual PPTraceAction *create() {
143 return new PPTraceAction(Ignore, CallbackCalls);
144 }
145
146private:
147 SmallSet<std::string, 4> &Ignore;
148 std::vector<CallbackCall> &CallbackCalls;
149};
Benjamin Kramere7103712015-03-23 12:49:15 +0000150} // namespace
John Thompson99794542013-10-31 12:23:32 +0000151
152// Output the trace given its data structure and a stream.
Benjamin Kramere7103712015-03-23 12:49:15 +0000153static int outputPPTrace(std::vector<CallbackCall> &CallbackCalls,
154 llvm::raw_ostream &OS) {
John Thompson99794542013-10-31 12:23:32 +0000155 // Mark start of document.
156 OS << "---\n";
157
158 for (std::vector<CallbackCall>::const_iterator I = CallbackCalls.begin(),
159 E = CallbackCalls.end();
160 I != E; ++I) {
161 const CallbackCall &Callback = *I;
162 OS << "- Callback: " << Callback.Name << "\n";
163
164 for (std::vector<Argument>::const_iterator AI = Callback.Arguments.begin(),
165 AE = Callback.Arguments.end();
166 AI != AE; ++AI) {
167 const Argument &Arg = *AI;
168 OS << " " << Arg.Name << ": " << Arg.Value << "\n";
169 }
170 }
171
172 // Mark end of document.
173 OS << "...\n";
174
175 return 0;
176}
177
178// Program entry point.
179int main(int Argc, const char **Argv) {
180
181 // Parse command line.
182 cl::ParseCommandLineOptions(Argc, Argv, "pp-trace.\n");
183
184 // Parse the IgnoreCallbacks list into strings.
185 SmallVector<StringRef, 32> IgnoreCallbacksStrings;
186 StringRef(IgnoreCallbacks).split(IgnoreCallbacksStrings, ",",
187 /*MaxSplit=*/ -1, /*KeepEmpty=*/false);
188 SmallSet<std::string, 4> Ignore;
189 for (SmallVector<StringRef, 32>::iterator I = IgnoreCallbacksStrings.begin(),
190 E = IgnoreCallbacksStrings.end();
191 I != E; ++I)
192 Ignore.insert(*I);
193
194 // Create the compilation database.
195 SmallString<256> PathBuf;
196 sys::fs::current_path(PathBuf);
Ahmed Charles6a2dc5c2014-03-09 09:24:40 +0000197 std::unique_ptr<CompilationDatabase> Compilations;
John Thompson99794542013-10-31 12:23:32 +0000198 Compilations.reset(
199 new FixedCompilationDatabase(Twine(PathBuf), CC1Arguments));
200
201 // Store the callback trace information here.
202 std::vector<CallbackCall> CallbackCalls;
203
204 // Create the tool and run the compilation.
205 ClangTool Tool(*Compilations, SourcePaths);
Benjamin Kramer6e914242014-07-24 10:23:33 +0000206 PPTraceFrontendActionFactory Factory(Ignore, CallbackCalls);
207 int HadErrors = Tool.run(&Factory);
John Thompson99794542013-10-31 12:23:32 +0000208
209 // If we had errors, exit early.
210 if (HadErrors)
211 return HadErrors;
212
213 // Do the output.
214 if (!OutputFileName.size()) {
215 HadErrors = outputPPTrace(CallbackCalls, llvm::outs());
216 } else {
217 // Set up output file.
Rafael Espindolab14bd532014-08-25 18:17:00 +0000218 std::error_code EC;
219 llvm::tool_output_file Out(OutputFileName, EC, llvm::sys::fs::F_Text);
220 if (EC) {
John Thompson99794542013-10-31 12:23:32 +0000221 llvm::errs() << "pp-trace: error creating " << OutputFileName << ":"
Rafael Espindolab14bd532014-08-25 18:17:00 +0000222 << EC.message() << "\n";
John Thompson99794542013-10-31 12:23:32 +0000223 return 1;
224 }
225
226 HadErrors = outputPPTrace(CallbackCalls, Out.os());
227
228 // Tell tool_output_file that we want to keep the file.
229 if (HadErrors == 0)
230 Out.keep();
231 }
232
233 return HadErrors;
234}