diff --git a/clang-tools-extra/clang-move/ClangMove.cpp b/clang-tools-extra/clang-move/ClangMove.cpp index 3a44aa3e9da2..8397c956e5bb 100644 --- a/clang-tools-extra/clang-move/ClangMove.cpp +++ b/clang-tools-extra/clang-move/ClangMove.cpp @@ -694,22 +694,28 @@ void ClangMoveTool::addIncludes(llvm::StringRef IncludeHeader, bool IsAngled, const SourceManager &SM) { SmallVector HeaderWithSearchPath; llvm::sys::path::append(HeaderWithSearchPath, SearchPath, IncludeHeader); - std::string AbsoluteOldHeader = makeAbsolutePath(Context->Spec.OldHeader); - if (AbsoluteOldHeader == + std::string AbsoluteIncludeHeader = MakeAbsolutePath(SM, llvm::StringRef(HeaderWithSearchPath.data(), - HeaderWithSearchPath.size()))) { - OldHeaderIncludeRange = IncludeFilenameRange; - return; - } - + HeaderWithSearchPath.size())); std::string IncludeLine = IsAngled ? ("#include <" + IncludeHeader + ">\n").str() : ("#include \"" + IncludeHeader + "\"\n").str(); + std::string AbsoluteOldHeader = makeAbsolutePath(Context->Spec.OldHeader); std::string AbsoluteCurrentFile = MakeAbsolutePath(SM, FileName); if (AbsoluteOldHeader == AbsoluteCurrentFile) { + // Find old.h includes "old.h". + if (AbsoluteOldHeader == AbsoluteIncludeHeader) { + OldHeaderIncludeRangeInHeader = IncludeFilenameRange; + return; + } HeaderIncludes.push_back(IncludeLine); } else if (makeAbsolutePath(Context->Spec.OldCC) == AbsoluteCurrentFile) { + // Find old.cc includes "old.h". + if (AbsoluteOldHeader == AbsoluteIncludeHeader) { + OldHeaderIncludeRangeInCC = IncludeFilenameRange; + return; + } CCIncludes.push_back(IncludeLine); } } @@ -857,13 +863,18 @@ void ClangMoveTool::moveAll(SourceManager &SM, StringRef OldFile, if (!NewFile.empty()) { auto AllCode = clang::tooling::Replacements( clang::tooling::Replacement(NewFile, 0, 0, Code)); - // If we are moving from old.cc, an extra step is required: excluding - // the #include of "old.h", instead, we replace it with #include of "new.h". - if (Context->Spec.NewCC == NewFile && OldHeaderIncludeRange.isValid()) { - AllCode = AllCode.merge( - clang::tooling::Replacements(clang::tooling::Replacement( - SM, OldHeaderIncludeRange, '"' + Context->Spec.NewHeader + '"'))); - } + auto ReplaceOldInclude = [&](clang::CharSourceRange OldHeaderIncludeRange) { + AllCode = AllCode.merge(clang::tooling::Replacements( + clang::tooling::Replacement(SM, OldHeaderIncludeRange, + '"' + Context->Spec.NewHeader + '"'))); + }; + // Fix the case where old.h/old.cc includes "old.h", we replace the + // `#include "old.h"` with `#include "new.h"`. + if (Context->Spec.NewCC == NewFile && OldHeaderIncludeRangeInCC.isValid()) + ReplaceOldInclude(OldHeaderIncludeRangeInCC); + else if (Context->Spec.NewHeader == NewFile && + OldHeaderIncludeRangeInHeader.isValid()) + ReplaceOldInclude(OldHeaderIncludeRangeInHeader); Context->FileToReplacements[NewFile] = std::move(AllCode); } } diff --git a/clang-tools-extra/clang-move/ClangMove.h b/clang-tools-extra/clang-move/ClangMove.h index 2522f5390df2..945c6db0aaa9 100644 --- a/clang-tools-extra/clang-move/ClangMove.h +++ b/clang-tools-extra/clang-move/ClangMove.h @@ -176,7 +176,11 @@ private: /// The source range for the written file name in #include (i.e. "old.h" for /// #include "old.h") in old.cc, including the enclosing quotes or angle /// brackets. - clang::CharSourceRange OldHeaderIncludeRange; + clang::CharSourceRange OldHeaderIncludeRangeInCC; + /// The source range for the written file name in #include (i.e. "old.h" for + /// #include "old.h") in old.h, including the enclosing quotes or angle + /// brackets. + clang::CharSourceRange OldHeaderIncludeRangeInHeader; /// Mapping from FilePath to FileID, which can be used in post processes like /// cleanup around replacements. llvm::StringMap FilePathToFileID; diff --git a/clang-tools-extra/unittests/clang-move/ClangMoveTests.cpp b/clang-tools-extra/unittests/clang-move/ClangMoveTests.cpp index ad2a052f9548..5103d762aacd 100644 --- a/clang-tools-extra/unittests/clang-move/ClangMoveTests.cpp +++ b/clang-tools-extra/unittests/clang-move/ClangMoveTests.cpp @@ -293,6 +293,32 @@ TEST(ClangMove, MoveNonExistClass) { EXPECT_EQ(0u, Results.size()); } +TEST(ClangMove, HeaderIncludeSelf) { + move::MoveDefinitionSpec Spec; + Spec.Names = {std::string("Foo")}; + Spec.OldHeader = "foo.h"; + Spec.OldCC = "foo.cc"; + Spec.NewHeader = "new_foo.h"; + Spec.NewCC = "new_foo.cc"; + + const char TestHeader[] = "#ifndef FOO_H\n" + "#define FOO_H\n" + "#include \"foo.h\"\n" + "class Foo {};\n" + "#endif\n"; + const char TestCode[] = "#include \"foo.h\""; + const char ExpectedNewHeader[] = "#ifndef FOO_H\n" + "#define FOO_H\n" + "#include \"new_foo.h\"\n" + "class Foo {};\n" + "#endif\n"; + const char ExpectedNewCC[] = "#include \"new_foo.h\""; + auto Results = runClangMoveOnCode(Spec, TestHeader, TestCode); + EXPECT_EQ("", Results[Spec.OldHeader]); + EXPECT_EQ(ExpectedNewHeader, Results[Spec.NewHeader]); + EXPECT_EQ(ExpectedNewCC, Results[Spec.NewCC]); +} + TEST(ClangMove, MoveAll) { std::vector TestHeaders = { "class A {\npublic:\n int f();\n};",