blob: 180e8c38292125da678cf4b12bd670f56c418c7b [file] [log] [blame]
Eric Liu495b2112016-09-19 17:40:32 +00001//===-- ClangIncludeFixer.cpp - Standalone change namespace ---------------===//
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// This tool can be used to change the surrounding namespaces of class/function
10// definitions.
11//
12// Example: test.cc
13// namespace na {
14// class X {};
15// namespace nb {
16// class Y { X x; };
17// } // namespace nb
18// } // namespace na
19// To move the definition of class Y from namespace "na::nb" to "x::y", run:
20// clang-change-namespace --old_namespace "na::nb" \
21// --new_namespace "x::y" --file_pattern "test.cc" test.cc --
22// Output:
23// namespace na {
24// class X {};
25// } // namespace na
26// namespace x {
27// namespace y {
28// class Y { na::X x; };
29// } // namespace y
30// } // namespace x
31
32#include "ChangeNamespace.h"
Eric Liu77b0c1a2016-09-19 20:41:39 +000033#include "clang/ASTMatchers/ASTMatchFinder.h"
Eric Liu495b2112016-09-19 17:40:32 +000034#include "clang/Frontend/FrontendActions.h"
35#include "clang/Frontend/TextDiagnosticPrinter.h"
36#include "clang/Rewrite/Core/Rewriter.h"
37#include "clang/Tooling/CommonOptionsParser.h"
38#include "clang/Tooling/Refactoring.h"
39#include "clang/Tooling/Tooling.h"
40#include "llvm/Support/CommandLine.h"
Eric Liu3db05212016-10-13 18:56:14 +000041#include "llvm/Support/Signals.h"
Eric Liua13c4192017-02-13 17:24:14 +000042#include "llvm/Support/YAMLTraits.h"
Eric Liu495b2112016-09-19 17:40:32 +000043
44using namespace clang;
45using namespace llvm;
46
47namespace {
48
49cl::OptionCategory ChangeNamespaceCategory("Change namespace.");
50
51cl::opt<std::string> OldNamespace("old_namespace", cl::Required,
52 cl::desc("Old namespace."),
53 cl::cat(ChangeNamespaceCategory));
54
55cl::opt<std::string> NewNamespace("new_namespace", cl::Required,
56 cl::desc("New namespace."),
57 cl::cat(ChangeNamespaceCategory));
58
59cl::opt<std::string> FilePattern(
60 "file_pattern", cl::Required,
61 cl::desc("Only rename namespaces in files that match the given pattern."),
62 cl::cat(ChangeNamespaceCategory));
63
64cl::opt<bool> Inplace("i", cl::desc("Inplace edit <file>s, if specified."),
65 cl::cat(ChangeNamespaceCategory));
66
Eric Liua13c4192017-02-13 17:24:14 +000067cl::opt<bool>
68 DumpYAML("dump_result",
69 cl::desc("Dump new file contents in YAML, if specified."),
70 cl::cat(ChangeNamespaceCategory));
71
Eric Liu495b2112016-09-19 17:40:32 +000072cl::opt<std::string> Style("style",
73 cl::desc("The style name used for reformatting."),
74 cl::init("LLVM"), cl::cat(ChangeNamespaceCategory));
75
Eric Liu7fccc992017-02-24 11:54:45 +000076cl::opt<std::string> WhiteListFile(
77 "whitelist_file",
78 cl::desc("A file containing regexes of symbol names that are not expected "
79 "to be updated when changing namespaces around them."),
80 cl::init(""), cl::cat(ChangeNamespaceCategory));
81
82llvm::ErrorOr<std::vector<std::string>> GetWhiteListedSymbolPatterns() {
Eric Liu3a7e1312017-02-28 10:00:15 +000083 std::vector<std::string> Patterns;
Eric Liu4c8767f2017-02-24 12:56:51 +000084 if (WhiteListFile.empty())
Eric Liu3a7e1312017-02-28 10:00:15 +000085 return Patterns;
Eric Liu4c8767f2017-02-24 12:56:51 +000086
Eric Liu7fccc992017-02-24 11:54:45 +000087 llvm::SmallVector<StringRef, 8> Lines;
Eric Liu4c8767f2017-02-24 12:56:51 +000088 llvm::ErrorOr<std::unique_ptr<llvm::MemoryBuffer>> File =
89 llvm::MemoryBuffer::getFile(WhiteListFile);
90 if (!File)
91 return File.getError();
92 llvm::StringRef Content = File.get()->getBuffer();
93 Content.split(Lines, '\n', /*MaxSplit=*/-1, /*KeepEmpty=*/false);
Eric Liu3a7e1312017-02-28 10:00:15 +000094 for (auto Line : Lines)
95 Patterns.push_back(Line.trim());
96 return Patterns;
Eric Liu7fccc992017-02-24 11:54:45 +000097}
98
Eric Liu495b2112016-09-19 17:40:32 +000099} // anonymous namespace
100
101int main(int argc, const char **argv) {
Eric Liu3db05212016-10-13 18:56:14 +0000102 llvm::sys::PrintStackTraceOnErrorSignal(argv[0]);
Eric Liu495b2112016-09-19 17:40:32 +0000103 tooling::CommonOptionsParser OptionsParser(argc, argv,
104 ChangeNamespaceCategory);
105 const auto &Files = OptionsParser.getSourcePathList();
106 tooling::RefactoringTool Tool(OptionsParser.getCompilations(), Files);
Eric Liu7fccc992017-02-24 11:54:45 +0000107 llvm::ErrorOr<std::vector<std::string>> WhiteListPatterns =
108 GetWhiteListedSymbolPatterns();
109 if (!WhiteListPatterns) {
110 llvm::errs() << "Failed to open whitelist file " << WhiteListFile << ". "
111 << WhiteListPatterns.getError().message() << "\n";
112 return 1;
113 }
Eric Liu495b2112016-09-19 17:40:32 +0000114 change_namespace::ChangeNamespaceTool NamespaceTool(
Eric Liu7fccc992017-02-24 11:54:45 +0000115 OldNamespace, NewNamespace, FilePattern, *WhiteListPatterns,
116 &Tool.getReplacements(), Style);
Eric Liu495b2112016-09-19 17:40:32 +0000117 ast_matchers::MatchFinder Finder;
118 NamespaceTool.registerMatchers(&Finder);
119 std::unique_ptr<tooling::FrontendActionFactory> Factory =
120 tooling::newFrontendActionFactory(&Finder);
121
122 if (int Result = Tool.run(Factory.get()))
123 return Result;
124 LangOptions DefaultLangOptions;
125 IntrusiveRefCntPtr<DiagnosticOptions> DiagOpts = new DiagnosticOptions();
126 clang::TextDiagnosticPrinter DiagnosticPrinter(errs(), &*DiagOpts);
127 DiagnosticsEngine Diagnostics(
128 IntrusiveRefCntPtr<DiagnosticIDs>(new DiagnosticIDs()), &*DiagOpts,
129 &DiagnosticPrinter, false);
130 auto &FileMgr = Tool.getFiles();
131 SourceManager Sources(Diagnostics, FileMgr);
132 Rewriter Rewrite(Sources, DefaultLangOptions);
133
134 if (!formatAndApplyAllReplacements(Tool.getReplacements(), Rewrite, Style)) {
135 llvm::errs() << "Failed applying all replacements.\n";
136 return 1;
137 }
138 if (Inplace)
139 return Rewrite.overwriteChangedFiles();
140
Eric Liua13c4192017-02-13 17:24:14 +0000141 std::set<llvm::StringRef> ChangedFiles;
142 for (const auto &it : Tool.getReplacements())
143 ChangedFiles.insert(it.first);
144
145 if (DumpYAML) {
146 auto WriteToYAML = [&](llvm::raw_ostream &OS) {
147 OS << "[\n";
148 for (auto I = ChangedFiles.begin(), E = ChangedFiles.end(); I != E; ++I) {
149 OS << " {\n";
150 OS << " \"FilePath\": \"" << *I << "\",\n";
151 const auto *Entry = FileMgr.getFile(*I);
152 auto ID = Sources.getOrCreateFileID(Entry, SrcMgr::C_User);
153 std::string Content;
154 llvm::raw_string_ostream ContentStream(Content);
155 Rewrite.getEditBuffer(ID).write(ContentStream);
156 OS << " \"SourceText\": \""
157 << llvm::yaml::escape(ContentStream.str()) << "\"\n";
158 OS << " }";
159 if (I != std::prev(E))
160 OS << ",\n";
161 }
162 OS << "\n]\n";
163 };
164 WriteToYAML(llvm::outs());
165 return 0;
166 }
167
168 for (const auto &File : ChangedFiles) {
Eric Liu495b2112016-09-19 17:40:32 +0000169 const auto *Entry = FileMgr.getFile(File);
170
171 auto ID = Sources.getOrCreateFileID(Entry, SrcMgr::C_User);
Eric Liu495b2112016-09-19 17:40:32 +0000172 outs() << "============== " << File << " ==============\n";
173 Rewrite.getEditBuffer(ID).write(llvm::outs());
174 outs() << "\n============================================\n";
175 }
Eric Liua13c4192017-02-13 17:24:14 +0000176
Eric Liu495b2112016-09-19 17:40:32 +0000177 return 0;
178}