blob: 001a043d803a9f2034c06c0092638026b011cfb1 [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
27// FIXME: This is terrible, we need this for ::close.
28#if !defined(_MSC_VER)
Douglas Gregor24466d82012-01-26 03:20:14 +000029#include <unistd.h>
NAKAMURA Takumi02770392012-01-26 03:47:18 +000030#else
31#include <io.h>
32#endif
Douglas Gregor24466d82012-01-26 03:20:14 +000033
Daniel Dunbar9b414d32010-06-15 17:48:49 +000034using namespace clang;
35
36//===----------------------------------------------------------------------===//
37// AST Consumer Actions
38//===----------------------------------------------------------------------===//
39
40ASTConsumer *HTMLPrintAction::CreateASTConsumer(CompilerInstance &CI,
Chris Lattner5f9e2722011-07-23 10:55:15 +000041 StringRef InFile) {
42 if (raw_ostream *OS = CI.createDefaultOutputFile(false, InFile))
Daniel Dunbar9b414d32010-06-15 17:48:49 +000043 return CreateHTMLPrinter(OS, CI.getPreprocessor());
44 return 0;
45}
46
47FixItAction::FixItAction() {}
48FixItAction::~FixItAction() {}
49
50ASTConsumer *FixItAction::CreateASTConsumer(CompilerInstance &CI,
Chris Lattner5f9e2722011-07-23 10:55:15 +000051 StringRef InFile) {
Daniel Dunbar9b414d32010-06-15 17:48:49 +000052 return new ASTConsumer();
53}
54
Benjamin Kramer79ba2a62010-10-22 16:48:22 +000055namespace {
Nick Lewycky96872c42010-08-15 16:47:39 +000056class FixItRewriteInPlace : public FixItOptions {
57public:
Argyrios Kyrtzidisc8af9102012-01-26 04:19:04 +000058 std::string RewriteFilename(const std::string &Filename, int &fd) {
59 fd = -1;
60 return Filename;
61 }
Nick Lewycky96872c42010-08-15 16:47:39 +000062};
63
Nick Lewycky1450f262010-08-13 17:31:00 +000064class FixItActionSuffixInserter : public FixItOptions {
Daniel Dunbar9b414d32010-06-15 17:48:49 +000065 std::string NewSuffix;
66
67public:
Nick Lewycky1450f262010-08-13 17:31:00 +000068 FixItActionSuffixInserter(std::string NewSuffix, bool FixWhatYouCan)
69 : NewSuffix(NewSuffix) {
70 this->FixWhatYouCan = FixWhatYouCan;
71 }
Daniel Dunbar9b414d32010-06-15 17:48:49 +000072
Argyrios Kyrtzidisc8af9102012-01-26 04:19:04 +000073 std::string RewriteFilename(const std::string &Filename, int &fd) {
74 fd = -1;
Michael J. Spencerd5b08be2010-12-18 04:13:32 +000075 llvm::SmallString<128> Path(Filename);
76 llvm::sys::path::replace_extension(Path,
77 NewSuffix + llvm::sys::path::extension(Path));
78 return Path.str();
Daniel Dunbar9b414d32010-06-15 17:48:49 +000079 }
80};
Argyrios Kyrtzidis61d679a2012-01-26 02:40:48 +000081
82class FixItRewriteToTemp : public FixItOptions {
83public:
Argyrios Kyrtzidisc8af9102012-01-26 04:19:04 +000084 std::string RewriteFilename(const std::string &Filename, int &fd) {
Argyrios Kyrtzidis61d679a2012-01-26 02:40:48 +000085 llvm::SmallString<128> Path;
86 Path = llvm::sys::path::filename(Filename);
87 Path += "-%%%%%%%%";
88 Path += llvm::sys::path::extension(Filename);
Argyrios Kyrtzidis61d679a2012-01-26 02:40:48 +000089 llvm::SmallString<128> NewPath;
Argyrios Kyrtzidisc8af9102012-01-26 04:19:04 +000090 llvm::sys::fs::unique_file(Path.str(), fd, NewPath);
Argyrios Kyrtzidis61d679a2012-01-26 02:40:48 +000091 return NewPath.str();
92 }
93};
Benjamin Kramer79ba2a62010-10-22 16:48:22 +000094} // end anonymous namespace
Daniel Dunbar9b414d32010-06-15 17:48:49 +000095
96bool FixItAction::BeginSourceFileAction(CompilerInstance &CI,
Chris Lattner5f9e2722011-07-23 10:55:15 +000097 StringRef Filename) {
Daniel Dunbar9b414d32010-06-15 17:48:49 +000098 const FrontendOptions &FEOpts = getCompilerInstance().getFrontendOpts();
99 if (!FEOpts.FixItSuffix.empty()) {
Nick Lewycky1450f262010-08-13 17:31:00 +0000100 FixItOpts.reset(new FixItActionSuffixInserter(FEOpts.FixItSuffix,
101 FEOpts.FixWhatYouCan));
Daniel Dunbar9b414d32010-06-15 17:48:49 +0000102 } else {
Nick Lewycky96872c42010-08-15 16:47:39 +0000103 FixItOpts.reset(new FixItRewriteInPlace);
104 FixItOpts->FixWhatYouCan = FEOpts.FixWhatYouCan;
Daniel Dunbar9b414d32010-06-15 17:48:49 +0000105 }
106 Rewriter.reset(new FixItRewriter(CI.getDiagnostics(), CI.getSourceManager(),
Nick Lewycky1450f262010-08-13 17:31:00 +0000107 CI.getLangOpts(), FixItOpts.get()));
Daniel Dunbar9b414d32010-06-15 17:48:49 +0000108 return true;
109}
110
111void FixItAction::EndSourceFileAction() {
112 // Otherwise rewrite all files.
113 Rewriter->WriteFixedFiles();
114}
115
Argyrios Kyrtzidis61d679a2012-01-26 02:40:48 +0000116bool FixItRecompile::BeginInvocation(CompilerInstance &CI) {
117
118 std::vector<std::pair<std::string, std::string> > RewrittenFiles;
119 bool err = false;
120 {
121 const FrontendOptions &FEOpts = CI.getFrontendOpts();
122 llvm::OwningPtr<FrontendAction> FixAction(new SyntaxOnlyAction());
123 FixAction->BeginSourceFile(CI, FEOpts.Inputs[0]);
124
125 llvm::OwningPtr<FixItOptions> FixItOpts;
126 if (FEOpts.FixToTemporaries)
127 FixItOpts.reset(new FixItRewriteToTemp());
128 else
129 FixItOpts.reset(new FixItRewriteInPlace());
130 FixItOpts->Silent = true;
131 FixItOpts->FixWhatYouCan = FEOpts.FixWhatYouCan;
132 FixItOpts->FixOnlyWarnings = FEOpts.FixOnlyWarnings;
133 FixItRewriter Rewriter(CI.getDiagnostics(), CI.getSourceManager(),
134 CI.getLangOpts(), FixItOpts.get());
135 FixAction->Execute();
136
137 err = Rewriter.WriteFixedFiles(&RewrittenFiles);
138
139 FixAction->EndSourceFile();
140 CI.setSourceManager(0);
141 CI.setFileManager(0);
142 }
143 if (err)
144 return false;
145 CI.getDiagnosticClient().clear();
146
147 PreprocessorOptions &PPOpts = CI.getPreprocessorOpts();
148 PPOpts.RemappedFiles.insert(PPOpts.RemappedFiles.end(),
149 RewrittenFiles.begin(), RewrittenFiles.end());
150 PPOpts.RemappedFilesKeepOriginalName = false;
151
152 return true;
153}
154
Daniel Dunbar9b414d32010-06-15 17:48:49 +0000155//===----------------------------------------------------------------------===//
156// Preprocessor Actions
157//===----------------------------------------------------------------------===//
158
159ASTConsumer *RewriteObjCAction::CreateASTConsumer(CompilerInstance &CI,
Chris Lattner5f9e2722011-07-23 10:55:15 +0000160 StringRef InFile) {
161 if (raw_ostream *OS = CI.createDefaultOutputFile(false, InFile, "cpp"))
Daniel Dunbar9b414d32010-06-15 17:48:49 +0000162 return CreateObjCRewriter(InFile, OS,
163 CI.getDiagnostics(), CI.getLangOpts(),
164 CI.getDiagnosticOpts().NoRewriteMacros);
165 return 0;
166}
167
168void RewriteMacrosAction::ExecuteAction() {
169 CompilerInstance &CI = getCompilerInstance();
Chris Lattner5f9e2722011-07-23 10:55:15 +0000170 raw_ostream *OS = CI.createDefaultOutputFile(true, getCurrentFile());
Daniel Dunbar9b414d32010-06-15 17:48:49 +0000171 if (!OS) return;
172
173 RewriteMacrosInInput(CI.getPreprocessor(), OS);
174}
175
176void RewriteTestAction::ExecuteAction() {
177 CompilerInstance &CI = getCompilerInstance();
Chris Lattner5f9e2722011-07-23 10:55:15 +0000178 raw_ostream *OS = CI.createDefaultOutputFile(false, getCurrentFile());
Daniel Dunbar9b414d32010-06-15 17:48:49 +0000179 if (!OS) return;
180
181 DoRewriteTest(CI.getPreprocessor(), OS);
182}