xref: /freebsd/contrib/llvm-project/clang/lib/ARCMigrate/TransformActions.cpp (revision 35c0a8c449fd2b7f75029ebed5e10852240f0865)
1 //===-- TransformActions.cpp - Migration to ARC mode ----------------------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 
9 #include "Internals.h"
10 #include "clang/AST/ASTContext.h"
11 #include "clang/AST/Expr.h"
12 #include "clang/Basic/SourceManager.h"
13 #include "clang/Lex/Preprocessor.h"
14 #include "llvm/ADT/DenseSet.h"
15 #include <map>
16 using namespace clang;
17 using namespace arcmt;
18 
19 namespace {
20 
21 /// Collects transformations and merges them before applying them with
22 /// with applyRewrites(). E.g. if the same source range
23 /// is requested to be removed twice, only one rewriter remove will be invoked.
24 /// Rewrites happen in "transactions"; if one rewrite in the transaction cannot
25 /// be done (e.g. it resides in a macro) all rewrites in the transaction are
26 /// aborted.
27 /// FIXME: "Transactional" rewrites support should be baked in the Rewriter.
28 class TransformActionsImpl {
29   CapturedDiagList &CapturedDiags;
30   ASTContext &Ctx;
31   Preprocessor &PP;
32 
33   bool IsInTransaction;
34 
35   enum ActionKind {
36     Act_Insert, Act_InsertAfterToken,
37     Act_Remove, Act_RemoveStmt,
38     Act_Replace, Act_ReplaceText,
39     Act_IncreaseIndentation,
40     Act_ClearDiagnostic
41   };
42 
43   struct ActionData {
44     ActionKind Kind;
45     SourceLocation Loc;
46     SourceRange R1, R2;
47     StringRef Text1, Text2;
48     Stmt *S;
49     SmallVector<unsigned, 2> DiagIDs;
50   };
51 
52   std::vector<ActionData> CachedActions;
53 
54   enum RangeComparison {
55     Range_Before,
56     Range_After,
57     Range_Contains,
58     Range_Contained,
59     Range_ExtendsBegin,
60     Range_ExtendsEnd
61   };
62 
63   /// A range to remove. It is a character range.
64   struct CharRange {
65     FullSourceLoc Begin, End;
66 
67     CharRange(CharSourceRange range, SourceManager &srcMgr, Preprocessor &PP) {
68       SourceLocation beginLoc = range.getBegin(), endLoc = range.getEnd();
69       assert(beginLoc.isValid() && endLoc.isValid());
70       if (range.isTokenRange()) {
71         Begin = FullSourceLoc(srcMgr.getExpansionLoc(beginLoc), srcMgr);
72         End = FullSourceLoc(getLocForEndOfToken(endLoc, srcMgr, PP), srcMgr);
73       } else {
74         Begin = FullSourceLoc(srcMgr.getExpansionLoc(beginLoc), srcMgr);
75         End = FullSourceLoc(srcMgr.getExpansionLoc(endLoc), srcMgr);
76       }
77       assert(Begin.isValid() && End.isValid());
78     }
79 
80     RangeComparison compareWith(const CharRange &RHS) const {
81       if (End.isBeforeInTranslationUnitThan(RHS.Begin))
82         return Range_Before;
83       if (RHS.End.isBeforeInTranslationUnitThan(Begin))
84         return Range_After;
85       if (!Begin.isBeforeInTranslationUnitThan(RHS.Begin) &&
86           !RHS.End.isBeforeInTranslationUnitThan(End))
87         return Range_Contained;
88       if (Begin.isBeforeInTranslationUnitThan(RHS.Begin) &&
89           RHS.End.isBeforeInTranslationUnitThan(End))
90         return Range_Contains;
91       if (Begin.isBeforeInTranslationUnitThan(RHS.Begin))
92         return Range_ExtendsBegin;
93       else
94         return Range_ExtendsEnd;
95     }
96 
97     static RangeComparison compare(SourceRange LHS, SourceRange RHS,
98                                    SourceManager &SrcMgr, Preprocessor &PP) {
99       return CharRange(CharSourceRange::getTokenRange(LHS), SrcMgr, PP)
100                   .compareWith(CharRange(CharSourceRange::getTokenRange(RHS),
101                                             SrcMgr, PP));
102     }
103   };
104 
105   typedef SmallVector<StringRef, 2> TextsVec;
106   typedef std::map<FullSourceLoc, TextsVec, FullSourceLoc::BeforeThanCompare>
107       InsertsMap;
108   InsertsMap Inserts;
109   /// A list of ranges to remove. They are always sorted and they never
110   /// intersect with each other.
111   std::list<CharRange> Removals;
112 
113   llvm::DenseSet<Stmt *> StmtRemovals;
114 
115   std::vector<std::pair<CharRange, SourceLocation> > IndentationRanges;
116 
117   /// Keeps text passed to transformation methods.
118   llvm::StringMap<bool> UniqueText;
119 
120 public:
121   TransformActionsImpl(CapturedDiagList &capturedDiags,
122                        ASTContext &ctx, Preprocessor &PP)
123     : CapturedDiags(capturedDiags), Ctx(ctx), PP(PP), IsInTransaction(false) { }
124 
125   ASTContext &getASTContext() { return Ctx; }
126 
127   void startTransaction();
128   bool commitTransaction();
129   void abortTransaction();
130 
131   bool isInTransaction() const { return IsInTransaction; }
132 
133   void insert(SourceLocation loc, StringRef text);
134   void insertAfterToken(SourceLocation loc, StringRef text);
135   void remove(SourceRange range);
136   void removeStmt(Stmt *S);
137   void replace(SourceRange range, StringRef text);
138   void replace(SourceRange range, SourceRange replacementRange);
139   void replaceStmt(Stmt *S, StringRef text);
140   void replaceText(SourceLocation loc, StringRef text,
141                    StringRef replacementText);
142   void increaseIndentation(SourceRange range,
143                            SourceLocation parentIndent);
144 
145   bool clearDiagnostic(ArrayRef<unsigned> IDs, SourceRange range);
146 
147   void applyRewrites(TransformActions::RewriteReceiver &receiver);
148 
149 private:
150   bool canInsert(SourceLocation loc);
151   bool canInsertAfterToken(SourceLocation loc);
152   bool canRemoveRange(SourceRange range);
153   bool canReplaceRange(SourceRange range, SourceRange replacementRange);
154   bool canReplaceText(SourceLocation loc, StringRef text);
155 
156   void commitInsert(SourceLocation loc, StringRef text);
157   void commitInsertAfterToken(SourceLocation loc, StringRef text);
158   void commitRemove(SourceRange range);
159   void commitRemoveStmt(Stmt *S);
160   void commitReplace(SourceRange range, SourceRange replacementRange);
161   void commitReplaceText(SourceLocation loc, StringRef text,
162                          StringRef replacementText);
163   void commitIncreaseIndentation(SourceRange range,SourceLocation parentIndent);
164   void commitClearDiagnostic(ArrayRef<unsigned> IDs, SourceRange range);
165 
166   void addRemoval(CharSourceRange range);
167   void addInsertion(SourceLocation loc, StringRef text);
168 
169   /// Stores text passed to the transformation methods to keep the string
170   /// "alive". Since the vast majority of text will be the same, we also unique
171   /// the strings using a StringMap.
172   StringRef getUniqueText(StringRef text);
173 
174   /// Computes the source location just past the end of the token at
175   /// the given source location. If the location points at a macro, the whole
176   /// macro expansion is skipped.
177   static SourceLocation getLocForEndOfToken(SourceLocation loc,
178                                             SourceManager &SM,Preprocessor &PP);
179 };
180 
181 } // anonymous namespace
182 
183 void TransformActionsImpl::startTransaction() {
184   assert(!IsInTransaction &&
185          "Cannot start a transaction in the middle of another one");
186   IsInTransaction = true;
187 }
188 
189 bool TransformActionsImpl::commitTransaction() {
190   assert(IsInTransaction && "No transaction started");
191 
192   if (CachedActions.empty()) {
193     IsInTransaction = false;
194     return false;
195   }
196 
197   // Verify that all actions are possible otherwise abort the whole transaction.
198   bool AllActionsPossible = true;
199   for (unsigned i = 0, e = CachedActions.size(); i != e; ++i) {
200     ActionData &act = CachedActions[i];
201     switch (act.Kind) {
202     case Act_Insert:
203       if (!canInsert(act.Loc))
204         AllActionsPossible = false;
205       break;
206     case Act_InsertAfterToken:
207       if (!canInsertAfterToken(act.Loc))
208         AllActionsPossible = false;
209       break;
210     case Act_Remove:
211       if (!canRemoveRange(act.R1))
212         AllActionsPossible = false;
213       break;
214     case Act_RemoveStmt:
215       assert(act.S);
216       if (!canRemoveRange(act.S->getSourceRange()))
217         AllActionsPossible = false;
218       break;
219     case Act_Replace:
220       if (!canReplaceRange(act.R1, act.R2))
221         AllActionsPossible = false;
222       break;
223     case Act_ReplaceText:
224       if (!canReplaceText(act.Loc, act.Text1))
225         AllActionsPossible = false;
226       break;
227     case Act_IncreaseIndentation:
228       // This is not important, we don't care if it will fail.
229       break;
230     case Act_ClearDiagnostic:
231       // We are just checking source rewrites.
232       break;
233     }
234     if (!AllActionsPossible)
235       break;
236   }
237 
238   if (!AllActionsPossible) {
239     abortTransaction();
240     return true;
241   }
242 
243   for (unsigned i = 0, e = CachedActions.size(); i != e; ++i) {
244     ActionData &act = CachedActions[i];
245     switch (act.Kind) {
246     case Act_Insert:
247       commitInsert(act.Loc, act.Text1);
248       break;
249     case Act_InsertAfterToken:
250       commitInsertAfterToken(act.Loc, act.Text1);
251       break;
252     case Act_Remove:
253       commitRemove(act.R1);
254       break;
255     case Act_RemoveStmt:
256       commitRemoveStmt(act.S);
257       break;
258     case Act_Replace:
259       commitReplace(act.R1, act.R2);
260       break;
261     case Act_ReplaceText:
262       commitReplaceText(act.Loc, act.Text1, act.Text2);
263       break;
264     case Act_IncreaseIndentation:
265       commitIncreaseIndentation(act.R1, act.Loc);
266       break;
267     case Act_ClearDiagnostic:
268       commitClearDiagnostic(act.DiagIDs, act.R1);
269       break;
270     }
271   }
272 
273   CachedActions.clear();
274   IsInTransaction = false;
275   return false;
276 }
277 
278 void TransformActionsImpl::abortTransaction() {
279   assert(IsInTransaction && "No transaction started");
280   CachedActions.clear();
281   IsInTransaction = false;
282 }
283 
284 void TransformActionsImpl::insert(SourceLocation loc, StringRef text) {
285   assert(IsInTransaction && "Actions only allowed during a transaction");
286   text = getUniqueText(text);
287   ActionData data;
288   data.Kind = Act_Insert;
289   data.Loc = loc;
290   data.Text1 = text;
291   CachedActions.push_back(data);
292 }
293 
294 void TransformActionsImpl::insertAfterToken(SourceLocation loc, StringRef text) {
295   assert(IsInTransaction && "Actions only allowed during a transaction");
296   text = getUniqueText(text);
297   ActionData data;
298   data.Kind = Act_InsertAfterToken;
299   data.Loc = loc;
300   data.Text1 = text;
301   CachedActions.push_back(data);
302 }
303 
304 void TransformActionsImpl::remove(SourceRange range) {
305   assert(IsInTransaction && "Actions only allowed during a transaction");
306   ActionData data;
307   data.Kind = Act_Remove;
308   data.R1 = range;
309   CachedActions.push_back(data);
310 }
311 
312 void TransformActionsImpl::removeStmt(Stmt *S) {
313   assert(IsInTransaction && "Actions only allowed during a transaction");
314   ActionData data;
315   data.Kind = Act_RemoveStmt;
316   if (auto *E = dyn_cast<Expr>(S))
317     S = E->IgnoreImplicit(); // important for uniquing
318   data.S = S;
319   CachedActions.push_back(data);
320 }
321 
322 void TransformActionsImpl::replace(SourceRange range, StringRef text) {
323   assert(IsInTransaction && "Actions only allowed during a transaction");
324   text = getUniqueText(text);
325   remove(range);
326   insert(range.getBegin(), text);
327 }
328 
329 void TransformActionsImpl::replace(SourceRange range,
330                                    SourceRange replacementRange) {
331   assert(IsInTransaction && "Actions only allowed during a transaction");
332   ActionData data;
333   data.Kind = Act_Replace;
334   data.R1 = range;
335   data.R2 = replacementRange;
336   CachedActions.push_back(data);
337 }
338 
339 void TransformActionsImpl::replaceText(SourceLocation loc, StringRef text,
340                                        StringRef replacementText) {
341   text = getUniqueText(text);
342   replacementText = getUniqueText(replacementText);
343   ActionData data;
344   data.Kind = Act_ReplaceText;
345   data.Loc = loc;
346   data.Text1 = text;
347   data.Text2 = replacementText;
348   CachedActions.push_back(data);
349 }
350 
351 void TransformActionsImpl::replaceStmt(Stmt *S, StringRef text) {
352   assert(IsInTransaction && "Actions only allowed during a transaction");
353   text = getUniqueText(text);
354   insert(S->getBeginLoc(), text);
355   removeStmt(S);
356 }
357 
358 void TransformActionsImpl::increaseIndentation(SourceRange range,
359                                                SourceLocation parentIndent) {
360   if (range.isInvalid()) return;
361   assert(IsInTransaction && "Actions only allowed during a transaction");
362   ActionData data;
363   data.Kind = Act_IncreaseIndentation;
364   data.R1 = range;
365   data.Loc = parentIndent;
366   CachedActions.push_back(data);
367 }
368 
369 bool TransformActionsImpl::clearDiagnostic(ArrayRef<unsigned> IDs,
370                                            SourceRange range) {
371   assert(IsInTransaction && "Actions only allowed during a transaction");
372   if (!CapturedDiags.hasDiagnostic(IDs, range))
373     return false;
374 
375   ActionData data;
376   data.Kind = Act_ClearDiagnostic;
377   data.R1 = range;
378   data.DiagIDs.append(IDs.begin(), IDs.end());
379   CachedActions.push_back(data);
380   return true;
381 }
382 
383 bool TransformActionsImpl::canInsert(SourceLocation loc) {
384   if (loc.isInvalid())
385     return false;
386 
387   SourceManager &SM = Ctx.getSourceManager();
388   if (SM.isInSystemHeader(SM.getExpansionLoc(loc)))
389     return false;
390 
391   if (loc.isFileID())
392     return true;
393   return PP.isAtStartOfMacroExpansion(loc);
394 }
395 
396 bool TransformActionsImpl::canInsertAfterToken(SourceLocation loc) {
397   if (loc.isInvalid())
398     return false;
399 
400   SourceManager &SM = Ctx.getSourceManager();
401   if (SM.isInSystemHeader(SM.getExpansionLoc(loc)))
402     return false;
403 
404   if (loc.isFileID())
405     return true;
406   return PP.isAtEndOfMacroExpansion(loc);
407 }
408 
409 bool TransformActionsImpl::canRemoveRange(SourceRange range) {
410   return canInsert(range.getBegin()) && canInsertAfterToken(range.getEnd());
411 }
412 
413 bool TransformActionsImpl::canReplaceRange(SourceRange range,
414                                            SourceRange replacementRange) {
415   return canRemoveRange(range) && canRemoveRange(replacementRange);
416 }
417 
418 bool TransformActionsImpl::canReplaceText(SourceLocation loc, StringRef text) {
419   if (!canInsert(loc))
420     return false;
421 
422   SourceManager &SM = Ctx.getSourceManager();
423   loc = SM.getExpansionLoc(loc);
424 
425   // Break down the source location.
426   std::pair<FileID, unsigned> locInfo = SM.getDecomposedLoc(loc);
427 
428   // Try to load the file buffer.
429   bool invalidTemp = false;
430   StringRef file = SM.getBufferData(locInfo.first, &invalidTemp);
431   if (invalidTemp)
432     return false;
433 
434   return file.substr(locInfo.second).starts_with(text);
435 }
436 
437 void TransformActionsImpl::commitInsert(SourceLocation loc, StringRef text) {
438   addInsertion(loc, text);
439 }
440 
441 void TransformActionsImpl::commitInsertAfterToken(SourceLocation loc,
442                                                   StringRef text) {
443   addInsertion(getLocForEndOfToken(loc, Ctx.getSourceManager(), PP), text);
444 }
445 
446 void TransformActionsImpl::commitRemove(SourceRange range) {
447   addRemoval(CharSourceRange::getTokenRange(range));
448 }
449 
450 void TransformActionsImpl::commitRemoveStmt(Stmt *S) {
451   assert(S);
452   if (StmtRemovals.count(S))
453     return; // already removed.
454 
455   if (Expr *E = dyn_cast<Expr>(S)) {
456     commitRemove(E->getSourceRange());
457     commitInsert(E->getSourceRange().getBegin(), getARCMTMacroName());
458   } else
459     commitRemove(S->getSourceRange());
460 
461   StmtRemovals.insert(S);
462 }
463 
464 void TransformActionsImpl::commitReplace(SourceRange range,
465                                          SourceRange replacementRange) {
466   RangeComparison comp = CharRange::compare(replacementRange, range,
467                                                Ctx.getSourceManager(), PP);
468   assert(comp == Range_Contained);
469   if (comp != Range_Contained)
470     return; // Although we asserted, be extra safe for release build.
471   if (range.getBegin() != replacementRange.getBegin())
472     addRemoval(CharSourceRange::getCharRange(range.getBegin(),
473                                              replacementRange.getBegin()));
474   if (replacementRange.getEnd() != range.getEnd())
475     addRemoval(CharSourceRange::getTokenRange(
476                                   getLocForEndOfToken(replacementRange.getEnd(),
477                                                       Ctx.getSourceManager(), PP),
478                                   range.getEnd()));
479 }
480 void TransformActionsImpl::commitReplaceText(SourceLocation loc,
481                                              StringRef text,
482                                              StringRef replacementText) {
483   SourceManager &SM = Ctx.getSourceManager();
484   loc = SM.getExpansionLoc(loc);
485   // canReplaceText already checked if loc points at text.
486   SourceLocation afterText = loc.getLocWithOffset(text.size());
487 
488   addRemoval(CharSourceRange::getCharRange(loc, afterText));
489   commitInsert(loc, replacementText);
490 }
491 
492 void TransformActionsImpl::commitIncreaseIndentation(SourceRange range,
493                                                   SourceLocation parentIndent) {
494   SourceManager &SM = Ctx.getSourceManager();
495   IndentationRanges.push_back(
496                  std::make_pair(CharRange(CharSourceRange::getTokenRange(range),
497                                           SM, PP),
498                                 SM.getExpansionLoc(parentIndent)));
499 }
500 
501 void TransformActionsImpl::commitClearDiagnostic(ArrayRef<unsigned> IDs,
502                                                  SourceRange range) {
503   CapturedDiags.clearDiagnostic(IDs, range);
504 }
505 
506 void TransformActionsImpl::addInsertion(SourceLocation loc, StringRef text) {
507   SourceManager &SM = Ctx.getSourceManager();
508   loc = SM.getExpansionLoc(loc);
509   for (const CharRange &I : llvm::reverse(Removals)) {
510     if (!SM.isBeforeInTranslationUnit(loc, I.End))
511       break;
512     if (I.Begin.isBeforeInTranslationUnitThan(loc))
513       return;
514   }
515 
516   Inserts[FullSourceLoc(loc, SM)].push_back(text);
517 }
518 
519 void TransformActionsImpl::addRemoval(CharSourceRange range) {
520   CharRange newRange(range, Ctx.getSourceManager(), PP);
521   if (newRange.Begin == newRange.End)
522     return;
523 
524   Inserts.erase(Inserts.upper_bound(newRange.Begin),
525                 Inserts.lower_bound(newRange.End));
526 
527   std::list<CharRange>::iterator I = Removals.end();
528   while (I != Removals.begin()) {
529     std::list<CharRange>::iterator RI = I;
530     --RI;
531     RangeComparison comp = newRange.compareWith(*RI);
532     switch (comp) {
533     case Range_Before:
534       --I;
535       break;
536     case Range_After:
537       Removals.insert(I, newRange);
538       return;
539     case Range_Contained:
540       return;
541     case Range_Contains:
542       RI->End = newRange.End;
543       [[fallthrough]];
544     case Range_ExtendsBegin:
545       newRange.End = RI->End;
546       Removals.erase(RI);
547       break;
548     case Range_ExtendsEnd:
549       RI->End = newRange.End;
550       return;
551     }
552   }
553 
554   Removals.insert(Removals.begin(), newRange);
555 }
556 
557 void TransformActionsImpl::applyRewrites(
558                                   TransformActions::RewriteReceiver &receiver) {
559   for (InsertsMap::iterator I = Inserts.begin(), E = Inserts.end(); I!=E; ++I) {
560     SourceLocation loc = I->first;
561     for (TextsVec::iterator
562            TI = I->second.begin(), TE = I->second.end(); TI != TE; ++TI) {
563       receiver.insert(loc, *TI);
564     }
565   }
566 
567   for (std::vector<std::pair<CharRange, SourceLocation> >::iterator
568        I = IndentationRanges.begin(), E = IndentationRanges.end(); I!=E; ++I) {
569     CharSourceRange range = CharSourceRange::getCharRange(I->first.Begin,
570                                                           I->first.End);
571     receiver.increaseIndentation(range, I->second);
572   }
573 
574   for (std::list<CharRange>::iterator
575          I = Removals.begin(), E = Removals.end(); I != E; ++I) {
576     CharSourceRange range = CharSourceRange::getCharRange(I->Begin, I->End);
577     receiver.remove(range);
578   }
579 }
580 
581 /// Stores text passed to the transformation methods to keep the string
582 /// "alive". Since the vast majority of text will be the same, we also unique
583 /// the strings using a StringMap.
584 StringRef TransformActionsImpl::getUniqueText(StringRef text) {
585   return UniqueText.insert(std::make_pair(text, false)).first->first();
586 }
587 
588 /// Computes the source location just past the end of the token at
589 /// the given source location. If the location points at a macro, the whole
590 /// macro expansion is skipped.
591 SourceLocation TransformActionsImpl::getLocForEndOfToken(SourceLocation loc,
592                                                          SourceManager &SM,
593                                                          Preprocessor &PP) {
594   if (loc.isMacroID()) {
595     CharSourceRange Exp = SM.getExpansionRange(loc);
596     if (Exp.isCharRange())
597       return Exp.getEnd();
598     loc = Exp.getEnd();
599   }
600   return PP.getLocForEndOfToken(loc);
601 }
602 
603 TransformActions::RewriteReceiver::~RewriteReceiver() { }
604 
605 TransformActions::TransformActions(DiagnosticsEngine &diag,
606                                    CapturedDiagList &capturedDiags,
607                                    ASTContext &ctx, Preprocessor &PP)
608     : Diags(diag), CapturedDiags(capturedDiags) {
609   Impl = new TransformActionsImpl(capturedDiags, ctx, PP);
610 }
611 
612 TransformActions::~TransformActions() {
613   delete static_cast<TransformActionsImpl*>(Impl);
614 }
615 
616 void TransformActions::startTransaction() {
617   static_cast<TransformActionsImpl*>(Impl)->startTransaction();
618 }
619 
620 bool TransformActions::commitTransaction() {
621   return static_cast<TransformActionsImpl*>(Impl)->commitTransaction();
622 }
623 
624 void TransformActions::abortTransaction() {
625   static_cast<TransformActionsImpl*>(Impl)->abortTransaction();
626 }
627 
628 
629 void TransformActions::insert(SourceLocation loc, StringRef text) {
630   static_cast<TransformActionsImpl*>(Impl)->insert(loc, text);
631 }
632 
633 void TransformActions::insertAfterToken(SourceLocation loc,
634                                         StringRef text) {
635   static_cast<TransformActionsImpl*>(Impl)->insertAfterToken(loc, text);
636 }
637 
638 void TransformActions::remove(SourceRange range) {
639   static_cast<TransformActionsImpl*>(Impl)->remove(range);
640 }
641 
642 void TransformActions::removeStmt(Stmt *S) {
643   static_cast<TransformActionsImpl*>(Impl)->removeStmt(S);
644 }
645 
646 void TransformActions::replace(SourceRange range, StringRef text) {
647   static_cast<TransformActionsImpl*>(Impl)->replace(range, text);
648 }
649 
650 void TransformActions::replace(SourceRange range,
651                                SourceRange replacementRange) {
652   static_cast<TransformActionsImpl*>(Impl)->replace(range, replacementRange);
653 }
654 
655 void TransformActions::replaceStmt(Stmt *S, StringRef text) {
656   static_cast<TransformActionsImpl*>(Impl)->replaceStmt(S, text);
657 }
658 
659 void TransformActions::replaceText(SourceLocation loc, StringRef text,
660                                    StringRef replacementText) {
661   static_cast<TransformActionsImpl*>(Impl)->replaceText(loc, text,
662                                                         replacementText);
663 }
664 
665 void TransformActions::increaseIndentation(SourceRange range,
666                                            SourceLocation parentIndent) {
667   static_cast<TransformActionsImpl*>(Impl)->increaseIndentation(range,
668                                                                 parentIndent);
669 }
670 
671 bool TransformActions::clearDiagnostic(ArrayRef<unsigned> IDs,
672                                        SourceRange range) {
673   return static_cast<TransformActionsImpl*>(Impl)->clearDiagnostic(IDs, range);
674 }
675 
676 void TransformActions::applyRewrites(RewriteReceiver &receiver) {
677   static_cast<TransformActionsImpl*>(Impl)->applyRewrites(receiver);
678 }
679 
680 DiagnosticBuilder TransformActions::report(SourceLocation loc, unsigned diagId,
681                                            SourceRange range) {
682   assert(!static_cast<TransformActionsImpl *>(Impl)->isInTransaction() &&
683          "Errors should be emitted out of a transaction");
684   return Diags.Report(loc, diagId) << range;
685 }
686 
687 void TransformActions::reportError(StringRef message, SourceLocation loc,
688                                    SourceRange range) {
689   report(loc, diag::err_mt_message, range) << message;
690 }
691 
692 void TransformActions::reportWarning(StringRef message, SourceLocation loc,
693                                      SourceRange range) {
694   report(loc, diag::warn_mt_message, range) << message;
695 }
696 
697 void TransformActions::reportNote(StringRef message, SourceLocation loc,
698                                   SourceRange range) {
699   report(loc, diag::note_mt_message, range) << message;
700 }
701