blob: 9bc218e994fe83bf0994d8bfea7e9429eef53296 [file] [log] [blame]
Daniel Dunbar9b414d32010-06-15 17:48:49 +00001//===--- FrontendActions.cpp ----------------------------------------------===//
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#include "clang/Rewrite/FrontendActions.h"
11#include "clang/AST/ASTConsumer.h"
12#include "clang/Lex/Preprocessor.h"
13#include "clang/Parse/Parser.h"
14#include "clang/Basic/FileManager.h"
Argyrios Kyrtzidis61d679a2012-01-26 02:40:48 +000015#include "clang/Frontend/FrontendActions.h"
Daniel Dunbar9b414d32010-06-15 17:48:49 +000016#include "clang/Frontend/CompilerInstance.h"
17#include "clang/Frontend/FrontendDiagnostic.h"
18#include "clang/Frontend/Utils.h"
19#include "clang/Rewrite/ASTConsumers.h"
20#include "clang/Rewrite/FixItRewriter.h"
21#include "clang/Rewrite/Rewriters.h"
22#include "llvm/ADT/OwningPtr.h"
23#include "llvm/Support/raw_ostream.h"
Michael J. Spencer03013fa2010-11-29 18:12:39 +000024#include "llvm/Support/Path.h"
Argyrios Kyrtzidis61d679a2012-01-26 02:40:48 +000025#include "llvm/Support/FileSystem.h"
NAKAMURA Takumi02770392012-01-26 03:47:18 +000026
Daniel Dunbar9b414d32010-06-15 17:48:49 +000027using namespace clang;
28
29//===----------------------------------------------------------------------===//
30// AST Consumer Actions
31//===----------------------------------------------------------------------===//
32
33ASTConsumer *HTMLPrintAction::CreateASTConsumer(CompilerInstance &CI,
Chris Lattner5f9e2722011-07-23 10:55:15 +000034 StringRef InFile) {
35 if (raw_ostream *OS = CI.createDefaultOutputFile(false, InFile))
Daniel Dunbar9b414d32010-06-15 17:48:49 +000036 return CreateHTMLPrinter(OS, CI.getPreprocessor());
37 return 0;
38}
39
40FixItAction::FixItAction() {}
41FixItAction::~FixItAction() {}
42
43ASTConsumer *FixItAction::CreateASTConsumer(CompilerInstance &CI,
Chris Lattner5f9e2722011-07-23 10:55:15 +000044 StringRef InFile) {
Daniel Dunbar9b414d32010-06-15 17:48:49 +000045 return new ASTConsumer();
46}
47
Benjamin Kramer79ba2a62010-10-22 16:48:22 +000048namespace {
Nick Lewycky96872c42010-08-15 16:47:39 +000049class FixItRewriteInPlace : public FixItOptions {
50public:
Argyrios Kyrtzidisc8af9102012-01-26 04:19:04 +000051 std::string RewriteFilename(const std::string &Filename, int &fd) {
52 fd = -1;
53 return Filename;
54 }
Nick Lewycky96872c42010-08-15 16:47:39 +000055};
56
Nick Lewycky1450f262010-08-13 17:31:00 +000057class FixItActionSuffixInserter : public FixItOptions {
Daniel Dunbar9b414d32010-06-15 17:48:49 +000058 std::string NewSuffix;
59
60public:
Nick Lewycky1450f262010-08-13 17:31:00 +000061 FixItActionSuffixInserter(std::string NewSuffix, bool FixWhatYouCan)
62 : NewSuffix(NewSuffix) {
63 this->FixWhatYouCan = FixWhatYouCan;
64 }
Daniel Dunbar9b414d32010-06-15 17:48:49 +000065
Argyrios Kyrtzidisc8af9102012-01-26 04:19:04 +000066 std::string RewriteFilename(const std::string &Filename, int &fd) {
67 fd = -1;
Dylan Noblesmithf7ccbad2012-02-05 02:13:05 +000068 SmallString<128> Path(Filename);
Michael J. Spencerd5b08be2010-12-18 04:13:32 +000069 llvm::sys::path::replace_extension(Path,
70 NewSuffix + llvm::sys::path::extension(Path));
71 return Path.str();
Daniel Dunbar9b414d32010-06-15 17:48:49 +000072 }
73};
Argyrios Kyrtzidis61d679a2012-01-26 02:40:48 +000074
75class FixItRewriteToTemp : public FixItOptions {
76public:
Argyrios Kyrtzidisc8af9102012-01-26 04:19:04 +000077 std::string RewriteFilename(const std::string &Filename, int &fd) {
Dylan Noblesmithf7ccbad2012-02-05 02:13:05 +000078 SmallString<128> Path;
Argyrios Kyrtzidis61d679a2012-01-26 02:40:48 +000079 Path = llvm::sys::path::filename(Filename);
80 Path += "-%%%%%%%%";
81 Path += llvm::sys::path::extension(Filename);
Dylan Noblesmithf7ccbad2012-02-05 02:13:05 +000082 SmallString<128> NewPath;
Argyrios Kyrtzidisc8af9102012-01-26 04:19:04 +000083 llvm::sys::fs::unique_file(Path.str(), fd, NewPath);
Argyrios Kyrtzidis61d679a2012-01-26 02:40:48 +000084 return NewPath.str();
85 }
86};
Benjamin Kramer79ba2a62010-10-22 16:48:22 +000087} // end anonymous namespace
Daniel Dunbar9b414d32010-06-15 17:48:49 +000088
89bool FixItAction::BeginSourceFileAction(CompilerInstance &CI,
Chris Lattner5f9e2722011-07-23 10:55:15 +000090 StringRef Filename) {
Daniel Dunbar9b414d32010-06-15 17:48:49 +000091 const FrontendOptions &FEOpts = getCompilerInstance().getFrontendOpts();
92 if (!FEOpts.FixItSuffix.empty()) {
Nick Lewycky1450f262010-08-13 17:31:00 +000093 FixItOpts.reset(new FixItActionSuffixInserter(FEOpts.FixItSuffix,
94 FEOpts.FixWhatYouCan));
Daniel Dunbar9b414d32010-06-15 17:48:49 +000095 } else {
Nick Lewycky96872c42010-08-15 16:47:39 +000096 FixItOpts.reset(new FixItRewriteInPlace);
97 FixItOpts->FixWhatYouCan = FEOpts.FixWhatYouCan;
Daniel Dunbar9b414d32010-06-15 17:48:49 +000098 }
99 Rewriter.reset(new FixItRewriter(CI.getDiagnostics(), CI.getSourceManager(),
Nick Lewycky1450f262010-08-13 17:31:00 +0000100 CI.getLangOpts(), FixItOpts.get()));
Daniel Dunbar9b414d32010-06-15 17:48:49 +0000101 return true;
102}
103
104void FixItAction::EndSourceFileAction() {
105 // Otherwise rewrite all files.
106 Rewriter->WriteFixedFiles();
107}
108
Argyrios Kyrtzidis61d679a2012-01-26 02:40:48 +0000109bool FixItRecompile::BeginInvocation(CompilerInstance &CI) {
110
111 std::vector<std::pair<std::string, std::string> > RewrittenFiles;
112 bool err = false;
113 {
114 const FrontendOptions &FEOpts = CI.getFrontendOpts();
Dylan Noblesmith6f42b622012-02-05 02:12:40 +0000115 OwningPtr<FrontendAction> FixAction(new SyntaxOnlyAction());
Argyrios Kyrtzidis30ae84c2012-01-27 01:00:47 +0000116 if (FixAction->BeginSourceFile(CI, FEOpts.Inputs[0])) {
Dylan Noblesmith6f42b622012-02-05 02:12:40 +0000117 OwningPtr<FixItOptions> FixItOpts;
Argyrios Kyrtzidis30ae84c2012-01-27 01:00:47 +0000118 if (FEOpts.FixToTemporaries)
119 FixItOpts.reset(new FixItRewriteToTemp());
120 else
121 FixItOpts.reset(new FixItRewriteInPlace());
122 FixItOpts->Silent = true;
123 FixItOpts->FixWhatYouCan = FEOpts.FixWhatYouCan;
124 FixItOpts->FixOnlyWarnings = FEOpts.FixOnlyWarnings;
125 FixItRewriter Rewriter(CI.getDiagnostics(), CI.getSourceManager(),
126 CI.getLangOpts(), FixItOpts.get());
127 FixAction->Execute();
Argyrios Kyrtzidis61d679a2012-01-26 02:40:48 +0000128
Argyrios Kyrtzidis30ae84c2012-01-27 01:00:47 +0000129 err = Rewriter.WriteFixedFiles(&RewrittenFiles);
130
131 FixAction->EndSourceFile();
132 CI.setSourceManager(0);
133 CI.setFileManager(0);
134 } else {
135 err = true;
136 }
Argyrios Kyrtzidis61d679a2012-01-26 02:40:48 +0000137 }
138 if (err)
139 return false;
140 CI.getDiagnosticClient().clear();
Argyrios Kyrtzidis97a9cf32012-01-27 06:15:37 +0000141 CI.getDiagnostics().Reset();
Argyrios Kyrtzidis61d679a2012-01-26 02:40:48 +0000142
143 PreprocessorOptions &PPOpts = CI.getPreprocessorOpts();
144 PPOpts.RemappedFiles.insert(PPOpts.RemappedFiles.end(),
145 RewrittenFiles.begin(), RewrittenFiles.end());
146 PPOpts.RemappedFilesKeepOriginalName = false;
147
148 return true;
149}
150
Daniel Dunbar9b414d32010-06-15 17:48:49 +0000151//===----------------------------------------------------------------------===//
152// Preprocessor Actions
153//===----------------------------------------------------------------------===//
154
155ASTConsumer *RewriteObjCAction::CreateASTConsumer(CompilerInstance &CI,
Chris Lattner5f9e2722011-07-23 10:55:15 +0000156 StringRef InFile) {
Fariborz Jahanian64cb63a2012-02-07 17:11:38 +0000157 if (raw_ostream *OS = CI.createDefaultOutputFile(false, InFile, "cpp")) {
John McCall260611a2012-06-20 06:18:46 +0000158 if (CI.getLangOpts().ObjCRuntime.isNonFragile())
Fariborz Jahanian64cb63a2012-02-07 17:11:38 +0000159 return CreateModernObjCRewriter(InFile, OS,
160 CI.getDiagnostics(), CI.getLangOpts(),
161 CI.getDiagnosticOpts().NoRewriteMacros);
Daniel Dunbar9b414d32010-06-15 17:48:49 +0000162 return CreateObjCRewriter(InFile, OS,
163 CI.getDiagnostics(), CI.getLangOpts(),
164 CI.getDiagnosticOpts().NoRewriteMacros);
Fariborz Jahanian64cb63a2012-02-07 17:11:38 +0000165 }
Daniel Dunbar9b414d32010-06-15 17:48:49 +0000166 return 0;
167}
168
169void RewriteMacrosAction::ExecuteAction() {
170 CompilerInstance &CI = getCompilerInstance();
Chris Lattner5f9e2722011-07-23 10:55:15 +0000171 raw_ostream *OS = CI.createDefaultOutputFile(true, getCurrentFile());
Daniel Dunbar9b414d32010-06-15 17:48:49 +0000172 if (!OS) return;
173
174 RewriteMacrosInInput(CI.getPreprocessor(), OS);
175}
176
177void RewriteTestAction::ExecuteAction() {
178 CompilerInstance &CI = getCompilerInstance();
Chris Lattner5f9e2722011-07-23 10:55:15 +0000179 raw_ostream *OS = CI.createDefaultOutputFile(false, getCurrentFile());
Daniel Dunbar9b414d32010-06-15 17:48:49 +0000180 if (!OS) return;
181
182 DoRewriteTest(CI.getPreprocessor(), OS);
183}
David Blaikie8c0b3782012-06-06 18:52:13 +0000184
185void RewriteIncludesAction::ExecuteAction() {
186 CompilerInstance &CI = getCompilerInstance();
187 raw_ostream *OS = CI.createDefaultOutputFile(true, getCurrentFile());
188 if (!OS) return;
189
190 RewriteIncludesInInput(CI.getPreprocessor(), OS,
191 CI.getPreprocessorOutputOpts());
192}