150
|
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).startswith(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 LLVM_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 }
|