[clang][modules] Don't prevent translation of FW_Private includes when explicitly...
[llvm-project.git] / clang / lib / Edit / Commit.cpp
blob7c5aea6e5069a504b5c9c0f30ef9ae866932b530
1 //===- Commit.cpp - A unit of edits ---------------------------------------===//
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 //===----------------------------------------------------------------------===//
9 #include "clang/Edit/Commit.h"
10 #include "clang/Basic/LLVM.h"
11 #include "clang/Basic/SourceLocation.h"
12 #include "clang/Basic/SourceManager.h"
13 #include "clang/Edit/EditedSource.h"
14 #include "clang/Edit/FileOffset.h"
15 #include "clang/Lex/Lexer.h"
16 #include "clang/Lex/PPConditionalDirectiveRecord.h"
17 #include "llvm/ADT/StringRef.h"
18 #include <cassert>
19 #include <utility>
21 using namespace clang;
22 using namespace edit;
24 SourceLocation Commit::Edit::getFileLocation(SourceManager &SM) const {
25 SourceLocation Loc = SM.getLocForStartOfFile(Offset.getFID());
26 Loc = Loc.getLocWithOffset(Offset.getOffset());
27 assert(Loc.isFileID());
28 return Loc;
31 CharSourceRange Commit::Edit::getFileRange(SourceManager &SM) const {
32 SourceLocation Loc = getFileLocation(SM);
33 return CharSourceRange::getCharRange(Loc, Loc.getLocWithOffset(Length));
36 CharSourceRange Commit::Edit::getInsertFromRange(SourceManager &SM) const {
37 SourceLocation Loc = SM.getLocForStartOfFile(InsertFromRangeOffs.getFID());
38 Loc = Loc.getLocWithOffset(InsertFromRangeOffs.getOffset());
39 assert(Loc.isFileID());
40 return CharSourceRange::getCharRange(Loc, Loc.getLocWithOffset(Length));
43 Commit::Commit(EditedSource &Editor)
44 : SourceMgr(Editor.getSourceManager()), LangOpts(Editor.getLangOpts()),
45 PPRec(Editor.getPPCondDirectiveRecord()),
46 Editor(&Editor) {}
48 bool Commit::insert(SourceLocation loc, StringRef text,
49 bool afterToken, bool beforePreviousInsertions) {
50 if (text.empty())
51 return true;
53 FileOffset Offs;
54 if ((!afterToken && !canInsert(loc, Offs)) ||
55 ( afterToken && !canInsertAfterToken(loc, Offs, loc))) {
56 IsCommitable = false;
57 return false;
60 addInsert(loc, Offs, text, beforePreviousInsertions);
61 return true;
64 bool Commit::insertFromRange(SourceLocation loc,
65 CharSourceRange range,
66 bool afterToken, bool beforePreviousInsertions) {
67 FileOffset RangeOffs;
68 unsigned RangeLen;
69 if (!canRemoveRange(range, RangeOffs, RangeLen)) {
70 IsCommitable = false;
71 return false;
74 FileOffset Offs;
75 if ((!afterToken && !canInsert(loc, Offs)) ||
76 ( afterToken && !canInsertAfterToken(loc, Offs, loc))) {
77 IsCommitable = false;
78 return false;
81 if (PPRec &&
82 PPRec->areInDifferentConditionalDirectiveRegion(loc, range.getBegin())) {
83 IsCommitable = false;
84 return false;
87 addInsertFromRange(loc, Offs, RangeOffs, RangeLen, beforePreviousInsertions);
88 return true;
91 bool Commit::remove(CharSourceRange range) {
92 FileOffset Offs;
93 unsigned Len;
94 if (!canRemoveRange(range, Offs, Len)) {
95 IsCommitable = false;
96 return false;
99 addRemove(range.getBegin(), Offs, Len);
100 return true;
103 bool Commit::insertWrap(StringRef before, CharSourceRange range,
104 StringRef after) {
105 bool commitableBefore = insert(range.getBegin(), before, /*afterToken=*/false,
106 /*beforePreviousInsertions=*/true);
107 bool commitableAfter;
108 if (range.isTokenRange())
109 commitableAfter = insertAfterToken(range.getEnd(), after);
110 else
111 commitableAfter = insert(range.getEnd(), after);
113 return commitableBefore && commitableAfter;
116 bool Commit::replace(CharSourceRange range, StringRef text) {
117 if (text.empty())
118 return remove(range);
120 FileOffset Offs;
121 unsigned Len;
122 if (!canInsert(range.getBegin(), Offs) || !canRemoveRange(range, Offs, Len)) {
123 IsCommitable = false;
124 return false;
127 addRemove(range.getBegin(), Offs, Len);
128 addInsert(range.getBegin(), Offs, text, false);
129 return true;
132 bool Commit::replaceWithInner(CharSourceRange range,
133 CharSourceRange replacementRange) {
134 FileOffset OuterBegin;
135 unsigned OuterLen;
136 if (!canRemoveRange(range, OuterBegin, OuterLen)) {
137 IsCommitable = false;
138 return false;
141 FileOffset InnerBegin;
142 unsigned InnerLen;
143 if (!canRemoveRange(replacementRange, InnerBegin, InnerLen)) {
144 IsCommitable = false;
145 return false;
148 FileOffset OuterEnd = OuterBegin.getWithOffset(OuterLen);
149 FileOffset InnerEnd = InnerBegin.getWithOffset(InnerLen);
150 if (OuterBegin.getFID() != InnerBegin.getFID() ||
151 InnerBegin < OuterBegin ||
152 InnerBegin > OuterEnd ||
153 InnerEnd > OuterEnd) {
154 IsCommitable = false;
155 return false;
158 addRemove(range.getBegin(),
159 OuterBegin, InnerBegin.getOffset() - OuterBegin.getOffset());
160 addRemove(replacementRange.getEnd(),
161 InnerEnd, OuterEnd.getOffset() - InnerEnd.getOffset());
162 return true;
165 bool Commit::replaceText(SourceLocation loc, StringRef text,
166 StringRef replacementText) {
167 if (text.empty() || replacementText.empty())
168 return true;
170 FileOffset Offs;
171 unsigned Len;
172 if (!canReplaceText(loc, replacementText, Offs, Len)) {
173 IsCommitable = false;
174 return false;
177 addRemove(loc, Offs, Len);
178 addInsert(loc, Offs, text, false);
179 return true;
182 void Commit::addInsert(SourceLocation OrigLoc, FileOffset Offs, StringRef text,
183 bool beforePreviousInsertions) {
184 if (text.empty())
185 return;
187 Edit data;
188 data.Kind = Act_Insert;
189 data.OrigLoc = OrigLoc;
190 data.Offset = Offs;
191 data.Text = text.copy(StrAlloc);
192 data.BeforePrev = beforePreviousInsertions;
193 CachedEdits.push_back(data);
196 void Commit::addInsertFromRange(SourceLocation OrigLoc, FileOffset Offs,
197 FileOffset RangeOffs, unsigned RangeLen,
198 bool beforePreviousInsertions) {
199 if (RangeLen == 0)
200 return;
202 Edit data;
203 data.Kind = Act_InsertFromRange;
204 data.OrigLoc = OrigLoc;
205 data.Offset = Offs;
206 data.InsertFromRangeOffs = RangeOffs;
207 data.Length = RangeLen;
208 data.BeforePrev = beforePreviousInsertions;
209 CachedEdits.push_back(data);
212 void Commit::addRemove(SourceLocation OrigLoc,
213 FileOffset Offs, unsigned Len) {
214 if (Len == 0)
215 return;
217 Edit data;
218 data.Kind = Act_Remove;
219 data.OrigLoc = OrigLoc;
220 data.Offset = Offs;
221 data.Length = Len;
222 CachedEdits.push_back(data);
225 bool Commit::canInsert(SourceLocation loc, FileOffset &offs) {
226 if (loc.isInvalid())
227 return false;
229 if (loc.isMacroID())
230 isAtStartOfMacroExpansion(loc, &loc);
232 const SourceManager &SM = SourceMgr;
233 loc = SM.getTopMacroCallerLoc(loc);
235 if (loc.isMacroID())
236 if (!isAtStartOfMacroExpansion(loc, &loc))
237 return false;
239 if (SM.isInSystemHeader(loc))
240 return false;
242 std::pair<FileID, unsigned> locInfo = SM.getDecomposedLoc(loc);
243 if (locInfo.first.isInvalid())
244 return false;
245 offs = FileOffset(locInfo.first, locInfo.second);
246 return canInsertInOffset(loc, offs);
249 bool Commit::canInsertAfterToken(SourceLocation loc, FileOffset &offs,
250 SourceLocation &AfterLoc) {
251 if (loc.isInvalid())
253 return false;
255 SourceLocation spellLoc = SourceMgr.getSpellingLoc(loc);
256 unsigned tokLen = Lexer::MeasureTokenLength(spellLoc, SourceMgr, LangOpts);
257 AfterLoc = loc.getLocWithOffset(tokLen);
259 if (loc.isMacroID())
260 isAtEndOfMacroExpansion(loc, &loc);
262 const SourceManager &SM = SourceMgr;
263 loc = SM.getTopMacroCallerLoc(loc);
265 if (loc.isMacroID())
266 if (!isAtEndOfMacroExpansion(loc, &loc))
267 return false;
269 if (SM.isInSystemHeader(loc))
270 return false;
272 loc = Lexer::getLocForEndOfToken(loc, 0, SourceMgr, LangOpts);
273 if (loc.isInvalid())
274 return false;
276 std::pair<FileID, unsigned> locInfo = SM.getDecomposedLoc(loc);
277 if (locInfo.first.isInvalid())
278 return false;
279 offs = FileOffset(locInfo.first, locInfo.second);
280 return canInsertInOffset(loc, offs);
283 bool Commit::canInsertInOffset(SourceLocation OrigLoc, FileOffset Offs) {
284 for (const auto &act : CachedEdits)
285 if (act.Kind == Act_Remove) {
286 if (act.Offset.getFID() == Offs.getFID() &&
287 Offs > act.Offset && Offs < act.Offset.getWithOffset(act.Length))
288 return false; // position has been removed.
291 if (!Editor)
292 return true;
293 return Editor->canInsertInOffset(OrigLoc, Offs);
296 bool Commit::canRemoveRange(CharSourceRange range,
297 FileOffset &Offs, unsigned &Len) {
298 const SourceManager &SM = SourceMgr;
299 range = Lexer::makeFileCharRange(range, SM, LangOpts);
300 if (range.isInvalid())
301 return false;
303 if (range.getBegin().isMacroID() || range.getEnd().isMacroID())
304 return false;
305 if (SM.isInSystemHeader(range.getBegin()) ||
306 SM.isInSystemHeader(range.getEnd()))
307 return false;
309 if (PPRec && PPRec->rangeIntersectsConditionalDirective(range.getAsRange()))
310 return false;
312 std::pair<FileID, unsigned> beginInfo = SM.getDecomposedLoc(range.getBegin());
313 std::pair<FileID, unsigned> endInfo = SM.getDecomposedLoc(range.getEnd());
314 if (beginInfo.first != endInfo.first ||
315 beginInfo.second > endInfo.second)
316 return false;
318 Offs = FileOffset(beginInfo.first, beginInfo.second);
319 Len = endInfo.second - beginInfo.second;
320 return true;
323 bool Commit::canReplaceText(SourceLocation loc, StringRef text,
324 FileOffset &Offs, unsigned &Len) {
325 assert(!text.empty());
327 if (!canInsert(loc, Offs))
328 return false;
330 // Try to load the file buffer.
331 bool invalidTemp = false;
332 StringRef file = SourceMgr.getBufferData(Offs.getFID(), &invalidTemp);
333 if (invalidTemp)
334 return false;
336 Len = text.size();
337 return file.substr(Offs.getOffset()).startswith(text);
340 bool Commit::isAtStartOfMacroExpansion(SourceLocation loc,
341 SourceLocation *MacroBegin) const {
342 return Lexer::isAtStartOfMacroExpansion(loc, SourceMgr, LangOpts, MacroBegin);
345 bool Commit::isAtEndOfMacroExpansion(SourceLocation loc,
346 SourceLocation *MacroEnd) const {
347 return Lexer::isAtEndOfMacroExpansion(loc, SourceMgr, LangOpts, MacroEnd);