<div dir="ltr"><div class="gmail_quote"><div dir="ltr">On Mon, Feb 29, 2016 at 5:39 PM Chandler Carruth <<a href="mailto:chandlerc@gmail.com">chandlerc@gmail.com</a>> wrote:<br></div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><div dir="ltr"><div class="gmail_quote"><div dir="ltr">On Mon, Feb 29, 2016 at 11:32 AM Manuel Klimek via cfe-commits <<a href="mailto:cfe-commits@lists.llvm.org" target="_blank">cfe-commits@lists.llvm.org</a>> wrote:<br></div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">Author: klimek<br>
Date: Mon Feb 29 10:27:41 2016<br>
New Revision: 262232<br>
<br>
URL: <a href="http://llvm.org/viewvc/llvm-project?rev=262232&view=rev" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project?rev=262232&view=rev</a><br>
Log:<br>
Implement new interfaces for code-formatting when applying replacements.<br></blockquote><div><br></div></div></div><div dir="ltr"><div class="gmail_quote"><div>Random request: provide more context in commit logs. =D</div><div><br></div><div>When I randomly end up staring at patches for some reason, its really useful to know what the background is, or at least a bit of a clue.</div><div><br></div><div>For example, here, why does applying replacements need new interfaces?</div><div><br></div><div>Specifically context for those not 100% actively following along with the detailed discussion are super useful in the commit log where others end up reading about changes.</div></div></div></blockquote><div><br></div><div>Thanks. Btw, this particular change has been rolled back because it violates layering (that I completely missed).</div><div> </div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><div dir="ltr"><div class="gmail_quote"><div> </div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">
<br>
Patch by Eric Liu.<br>
<br>
Modified:<br>
cfe/trunk/include/clang/Tooling/Core/Replacement.h<br>
cfe/trunk/lib/Tooling/Core/Replacement.cpp<br>
cfe/trunk/unittests/Tooling/CMakeLists.txt<br>
cfe/trunk/unittests/Tooling/RefactoringTest.cpp<br>
<br>
Modified: cfe/trunk/include/clang/Tooling/Core/Replacement.h<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/include/clang/Tooling/Core/Replacement.h?rev=262232&r1=262231&r2=262232&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/cfe/trunk/include/clang/Tooling/Core/Replacement.h?rev=262232&r1=262231&r2=262232&view=diff</a><br>
==============================================================================<br>
--- cfe/trunk/include/clang/Tooling/Core/Replacement.h (original)<br>
+++ cfe/trunk/include/clang/Tooling/Core/Replacement.h Mon Feb 29 10:27:41 2016<br>
@@ -30,6 +30,10 @@ namespace clang {<br>
<br>
class Rewriter;<br>
<br>
+namespace format {<br>
+struct FormatStyle;<br>
+} // namespace format<br>
+<br>
namespace tooling {<br>
<br>
/// \brief A source range independent of the \c SourceManager.<br>
@@ -220,6 +224,41 @@ bool applyAllReplacements(const std::vec<br>
/// replacements cannot be applied, this returns an empty \c string.<br>
std::string applyAllReplacements(StringRef Code, const Replacements &Replaces);<br>
<br>
+/// \brief Applies all replacements in \p Replaces to \p Code.<br>
+///<br>
+/// This completely ignores the path stored in each replacement. If one or more<br>
+/// replacements cannot be applied, this returns an empty \c string.<br>
+std::string applyAllReplacements(StringRef Code,<br>
+ const std::vector<Replacements> &Replaces);<br>
+<br>
+/// \brief Calculate the ranges in a single file that are affected by the<br>
+/// Replacements.<br>
+///<br>
+/// \pre Replacements must be for the same file.<br>
+std::vector<tooling::Range><br>
+calculateChangedRangesInFile(const tooling::Replacements &Replaces);<br>
+<br>
+/// \brief Return replacements that are merged from orginal replacements<br>
+/// and the replacements for formatting the code after applying the orginal<br>
+/// replacements.<br>
+tooling::Replacements formatReplacements(StringRef Code,<br>
+ const tooling::Replacements &Replaces,<br>
+ const format::FormatStyle &Style);<br>
+<br>
+/// \brief In addition to applying replacements as in `applyAllReplacements`,<br>
+/// this function also reformats the changed code after applying replacements.<br>
+///<br>
+/// \pre Replacements must be for the same file and conflict-free.<br>
+///<br>
+/// Replacement applications happen independently of the success of<br>
+/// other applications.<br>
+///<br>
+/// \returns the changed code if all replacements apply and code is fixed.<br>
+/// empty string otherwise.<br>
+std::string applyAllReplacementsAndFormat(StringRef Code,<br>
+ const Replacements &Replaces,<br>
+ const format::FormatStyle &Style);<br>
+<br>
/// \brief Merges two sets of replacements with the second set referring to the<br>
/// code after applying the first set. Within both 'First' and 'Second',<br>
/// replacements must not overlap.<br>
<br>
Modified: cfe/trunk/lib/Tooling/Core/Replacement.cpp<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/lib/Tooling/Core/Replacement.cpp?rev=262232&r1=262231&r2=262232&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/cfe/trunk/lib/Tooling/Core/Replacement.cpp?rev=262232&r1=262231&r2=262232&view=diff</a><br>
==============================================================================<br>
--- cfe/trunk/lib/Tooling/Core/Replacement.cpp (original)<br>
+++ cfe/trunk/lib/Tooling/Core/Replacement.cpp Mon Feb 29 10:27:41 2016<br>
@@ -11,17 +11,20 @@<br>
//<br>
//===----------------------------------------------------------------------===//<br>
<br>
+#include "clang/Tooling/Core/Replacement.h"<br>
+<br>
#include "clang/Basic/Diagnostic.h"<br>
#include "clang/Basic/DiagnosticIDs.h"<br>
#include "clang/Basic/DiagnosticOptions.h"<br>
#include "clang/Basic/FileManager.h"<br>
#include "clang/Basic/SourceManager.h"<br>
+#include "clang/Format/Format.h"<br>
#include "clang/Lex/Lexer.h"<br>
#include "clang/Rewrite/Core/Rewriter.h"<br>
-#include "clang/Tooling/Core/Replacement.h"<br>
#include "llvm/Support/FileSystem.h"<br>
#include "llvm/Support/Path.h"<br>
#include "llvm/Support/raw_os_ostream.h"<br>
+#include <iterator><br>
<br>
namespace clang {<br>
namespace tooling {<br>
@@ -281,6 +284,44 @@ std::string applyAllReplacements(StringR<br>
return Result;<br>
}<br>
<br>
+tooling::Replacements formatReplacements(StringRef Code,<br>
+ const tooling::Replacements &Replaces,<br>
+ const format::FormatStyle &Style) {<br>
+ if (Replaces.empty()) return Replacements();<br>
+<br>
+ std::string NewCode = applyAllReplacements(Code, Replaces);<br>
+ std::vector<tooling::Range> ChangedRanges =<br>
+ calculateChangedRangesInFile(Replaces);<br>
+ StringRef FileName = Replaces.begin()->getFilePath();<br>
+ tooling::Replacements FormatReplaces =<br>
+ format::reformat(Style, NewCode, ChangedRanges, FileName);<br>
+<br>
+ tooling::Replacements MergedReplacements =<br>
+ mergeReplacements(Replaces, FormatReplaces);<br>
+ return MergedReplacements;<br>
+}<br>
+<br>
+std::vector<Range> calculateChangedRangesInFile(const Replacements &Replaces) {<br>
+ std::vector<Range> ChangedRanges;<br>
+ int Shift = 0;<br>
+ for (const tooling::Replacement &R : Replaces) {<br>
+ unsigned Offset = R.getOffset() + Shift;<br>
+ unsigned Length = R.getReplacementText().size();<br>
+ Shift += Length - R.getLength();<br>
+ ChangedRanges.push_back(tooling::Range(Offset, Length));<br>
+ }<br>
+ return ChangedRanges;<br>
+}<br>
+<br>
+std::string applyAllReplacementsAndFormat(StringRef Code,<br>
+ const Replacements &Replaces,<br>
+ const format::FormatStyle &Style) {<br>
+ Replacements NewReplacements = formatReplacements(Code, Replaces, Style);<br>
+ if (NewReplacements.empty())<br>
+ return Code; // Exit early to avoid overhead in `applyAllReplacements`.<br>
+ return applyAllReplacements(Code, NewReplacements);<br>
+}<br>
+<br>
namespace {<br>
// Represents a merged replacement, i.e. a replacement consisting of multiple<br>
// overlapping replacements from 'First' and 'Second' in mergeReplacements.<br>
@@ -314,7 +355,7 @@ public:<br>
<br>
// Merges the next element 'R' into this merged element. As we always merge<br>
// from 'First' into 'Second' or vice versa, the MergedReplacement knows what<br>
- // set the next element is coming from.<br>
+ // set the next element is coming from.<br>
void merge(const Replacement &R) {<br>
if (MergeSecond) {<br>
unsigned REnd = R.getOffset() + Delta + R.getLength();<br>
<br>
Modified: cfe/trunk/unittests/Tooling/CMakeLists.txt<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/unittests/Tooling/CMakeLists.txt?rev=262232&r1=262231&r2=262232&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/cfe/trunk/unittests/Tooling/CMakeLists.txt?rev=262232&r1=262231&r2=262232&view=diff</a><br>
==============================================================================<br>
--- cfe/trunk/unittests/Tooling/CMakeLists.txt (original)<br>
+++ cfe/trunk/unittests/Tooling/CMakeLists.txt Mon Feb 29 10:27:41 2016<br>
@@ -24,6 +24,7 @@ target_link_libraries(ToolingTests<br>
clangAST<br>
clangASTMatchers<br>
clangBasic<br>
+ clangFormat<br>
clangFrontend<br>
clangLex<br>
clangRewrite<br>
<br>
Modified: cfe/trunk/unittests/Tooling/RefactoringTest.cpp<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/unittests/Tooling/RefactoringTest.cpp?rev=262232&r1=262231&r2=262232&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/cfe/trunk/unittests/Tooling/RefactoringTest.cpp?rev=262232&r1=262231&r2=262232&view=diff</a><br>
==============================================================================<br>
--- cfe/trunk/unittests/Tooling/RefactoringTest.cpp (original)<br>
+++ cfe/trunk/unittests/Tooling/RefactoringTest.cpp Mon Feb 29 10:27:41 2016<br>
@@ -18,6 +18,7 @@<br>
#include "clang/Basic/FileManager.h"<br>
#include "clang/Basic/LangOptions.h"<br>
#include "clang/Basic/SourceManager.h"<br>
+#include "clang/Format/Format.h"<br>
#include "clang/Frontend/CompilerInstance.h"<br>
#include "clang/Frontend/FrontendAction.h"<br>
#include "clang/Frontend/TextDiagnosticPrinter.h"<br>
@@ -166,6 +167,35 @@ TEST_F(ReplacementTest, ApplyAllFailsIfO<br>
EXPECT_EQ("z", Context.getRewrittenText(IDz));<br>
}<br>
<br>
+TEST_F(ReplacementTest, FormatCodeAfterReplacements) {<br>
+ // Column limit is 20.<br>
+ std::string Code = "Type *a =\n"<br>
+ " new Type();\n"<br>
+ "g(iiiii, 0, jjjjj,\n"<br>
+ " 0, kkkkk, 0, mm);\n"<br>
+ "int bad = format ;";<br>
+ std::string Expected = "auto a = new Type();\n"<br>
+ "g(iiiii, nullptr,\n"<br>
+ " jjjjj, nullptr,\n"<br>
+ " kkkkk, nullptr,\n"<br>
+ " mm);\n"<br>
+ "int bad = format ;";<br>
+ FileID ID = Context.createInMemoryFile("format.cpp", Code);<br>
+ Replacements Replaces;<br>
+ Replaces.insert(<br>
+ Replacement(Context.Sources, Context.getLocation(ID, 1, 1), 6, "auto "));<br>
+ Replaces.insert(Replacement(Context.Sources, Context.getLocation(ID, 3, 10),<br>
+ 1, "nullptr"));<br>
+ Replaces.insert(Replacement(Context.Sources, Context.getLocation(ID, 4, 3), 1,<br>
+ "nullptr"));<br>
+ Replaces.insert(Replacement(Context.Sources, Context.getLocation(ID, 4, 13),<br>
+ 1, "nullptr"));<br>
+<br>
+ format::FormatStyle Style = format::getLLVMStyle();<br>
+ Style.ColumnLimit = 20; // Set column limit to 20 to increase readibility.<br>
+ EXPECT_EQ(Expected, applyAllReplacementsAndFormat(Code, Replaces, Style));<br>
+}<br>
+<br>
TEST(ShiftedCodePositionTest, FindsNewCodePosition) {<br>
Replacements Replaces;<br>
Replaces.insert(Replacement("", 0, 1, ""));<br>
@@ -418,6 +448,25 @@ TEST(Range, contains) {<br>
EXPECT_FALSE(Range(0, 10).contains(Range(0, 11)));<br>
}<br>
<br>
+TEST(Range, CalculateRangesOfReplacements) {<br>
+ // Before: aaaabbbbbbz<br>
+ // After : bbbbbbzzzzzzoooooooooooooooo<br>
+ Replacements Replaces;<br>
+ Replaces.insert(Replacement("foo", 0, 4, ""));<br>
+ Replaces.insert(Replacement("foo", 10, 1, "zzzzzz"));<br>
+ Replaces.insert(Replacement("foo", 11, 0, "oooooooooooooooo"));<br>
+<br>
+ std::vector<Range> Ranges = calculateChangedRangesInFile(Replaces);<br>
+<br>
+ EXPECT_EQ(3ul, Ranges.size());<br>
+ EXPECT_TRUE(Ranges[0].getOffset() == 0);<br>
+ EXPECT_TRUE(Ranges[0].getLength() == 0);<br>
+ EXPECT_TRUE(Ranges[1].getOffset() == 6);<br>
+ EXPECT_TRUE(Ranges[1].getLength() == 6);<br>
+ EXPECT_TRUE(Ranges[2].getOffset() == 12);<br>
+ EXPECT_TRUE(Ranges[2].getLength() == 16);<br>
+}<br>
+<br>
TEST(DeduplicateTest, removesDuplicates) {<br>
std::vector<Replacement> Input;<br>
Input.push_back(Replacement("fileA", 50, 0, " foo "));<br>
<br>
<br>
_______________________________________________<br>
cfe-commits mailing list<br>
<a href="mailto:cfe-commits@lists.llvm.org" target="_blank">cfe-commits@lists.llvm.org</a><br>
<a href="http://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits" rel="noreferrer" target="_blank">http://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits</a><br>
</blockquote></div></div></blockquote></div></div>