Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
freebsd
GitHub Repository: freebsd/freebsd-src
Path: blob/main/contrib/llvm-project/clang/lib/ARCMigrate/TransformActions.cpp
35236 views
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
702