1 //===--- LeftRightQualifierAlignmentFixer.cpp -------------------*- C++--*-===//
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 /// \file
10 /// This file implements LeftRightQualifierAlignmentFixer, a TokenAnalyzer that
11 /// enforces either left or right const depending on the style.
12 ///
13 //===----------------------------------------------------------------------===//
14 
15 #include "QualifierAlignmentFixer.h"
16 #include "FormatToken.h"
17 #include "llvm/Support/Debug.h"
18 #include "llvm/Support/Regex.h"
19 
20 #include <algorithm>
21 
22 #define DEBUG_TYPE "format-qualifier-alignment-fixer"
23 
24 namespace clang {
25 namespace format {
26 
27 QualifierAlignmentFixer::QualifierAlignmentFixer(
28     const Environment &Env, const FormatStyle &Style, StringRef &Code,
29     ArrayRef<tooling::Range> Ranges, unsigned FirstStartColumn,
30     unsigned NextStartColumn, unsigned LastStartColumn, StringRef FileName)
31     : TokenAnalyzer(Env, Style), Code(Code), Ranges(Ranges),
32       FirstStartColumn(FirstStartColumn), NextStartColumn(NextStartColumn),
33       LastStartColumn(LastStartColumn), FileName(FileName) {
34   std::vector<std::string> LeftOrder;
35   std::vector<std::string> RightOrder;
36   std::vector<tok::TokenKind> ConfiguredQualifierTokens;
37   PrepareLeftRightOrdering(Style.QualifierOrder, LeftOrder, RightOrder,
38                            ConfiguredQualifierTokens);
39 
40   // Handle the left and right Alignment Seperately
41   for (const auto &Qualifier : LeftOrder) {
42     Passes.emplace_back(
43         [&, Qualifier, ConfiguredQualifierTokens](const Environment &Env) {
44           return LeftRightQualifierAlignmentFixer(Env, Style, Qualifier,
45                                                   ConfiguredQualifierTokens,
46                                                   /*RightAlign=*/false)
47               .process();
48         });
49   }
50   for (const auto &Qualifier : RightOrder) {
51     Passes.emplace_back(
52         [&, Qualifier, ConfiguredQualifierTokens](const Environment &Env) {
53           return LeftRightQualifierAlignmentFixer(Env, Style, Qualifier,
54                                                   ConfiguredQualifierTokens,
55                                                   /*RightAlign=*/true)
56               .process();
57         });
58   }
59 }
60 
61 std::pair<tooling::Replacements, unsigned> QualifierAlignmentFixer::analyze(
62     TokenAnnotator &Annotator, SmallVectorImpl<AnnotatedLine *> &AnnotatedLines,
63     FormatTokenLexer &Tokens) {
64 
65   auto Env =
66       std::make_unique<Environment>(Code, FileName, Ranges, FirstStartColumn,
67                                     NextStartColumn, LastStartColumn);
68   llvm::Optional<std::string> CurrentCode = None;
69   tooling::Replacements Fixes;
70   unsigned Penalty = 0;
71   for (size_t I = 0, E = Passes.size(); I < E; ++I) {
72     std::pair<tooling::Replacements, unsigned> PassFixes = Passes[I](*Env);
73     auto NewCode = applyAllReplacements(
74         CurrentCode ? StringRef(*CurrentCode) : Code, PassFixes.first);
75     if (NewCode) {
76       Fixes = Fixes.merge(PassFixes.first);
77       Penalty += PassFixes.second;
78       if (I + 1 < E) {
79         CurrentCode = std::move(*NewCode);
80         Env = std::make_unique<Environment>(
81             *CurrentCode, FileName,
82             tooling::calculateRangesAfterReplacements(Fixes, Ranges),
83             FirstStartColumn, NextStartColumn, LastStartColumn);
84       }
85     }
86   }
87   return {Fixes, Penalty};
88 }
89 
90 static void replaceToken(const SourceManager &SourceMgr,
91                          tooling::Replacements &Fixes,
92                          const CharSourceRange &Range, std::string NewText) {
93   auto Replacement = tooling::Replacement(SourceMgr, Range, NewText);
94   auto Err = Fixes.add(Replacement);
95 
96   if (Err)
97     llvm::errs() << "Error while rearranging Qualifier : "
98                  << llvm::toString(std::move(Err)) << "\n";
99 }
100 
101 static void removeToken(const SourceManager &SourceMgr,
102                         tooling::Replacements &Fixes,
103                         const FormatToken *First) {
104   auto Range = CharSourceRange::getCharRange(First->getStartOfNonWhitespace(),
105                                              First->Tok.getEndLoc());
106   replaceToken(SourceMgr, Fixes, Range, "");
107 }
108 
109 static void insertQualifierAfter(const SourceManager &SourceMgr,
110                                  tooling::Replacements &Fixes,
111                                  const FormatToken *First,
112                                  const std::string &Qualifier) {
113   FormatToken *Next = First->Next;
114   if (!Next)
115     return;
116   auto Range = CharSourceRange::getCharRange(Next->getStartOfNonWhitespace(),
117                                              Next->Tok.getEndLoc());
118 
119   std::string NewText = " " + Qualifier + " ";
120   NewText += Next->TokenText;
121   replaceToken(SourceMgr, Fixes, Range, NewText);
122 }
123 
124 static void insertQualifierBefore(const SourceManager &SourceMgr,
125                                   tooling::Replacements &Fixes,
126                                   const FormatToken *First,
127                                   const std::string &Qualifier) {
128   auto Range = CharSourceRange::getCharRange(First->getStartOfNonWhitespace(),
129                                              First->Tok.getEndLoc());
130 
131   std::string NewText = " " + Qualifier + " ";
132   NewText += First->TokenText;
133 
134   replaceToken(SourceMgr, Fixes, Range, NewText);
135 }
136 
137 static bool endsWithSpace(const std::string &s) {
138   if (s.empty()) {
139     return false;
140   }
141   return isspace(s.back());
142 }
143 
144 static bool startsWithSpace(const std::string &s) {
145   if (s.empty()) {
146     return false;
147   }
148   return isspace(s.front());
149 }
150 
151 static void rotateTokens(const SourceManager &SourceMgr,
152                          tooling::Replacements &Fixes, const FormatToken *First,
153                          const FormatToken *Last, bool Left) {
154   auto *End = Last;
155   auto *Begin = First;
156   if (!Left) {
157     End = Last->Next;
158     Begin = First->Next;
159   }
160 
161   std::string NewText;
162   // If we are rotating to the left we move the Last token to the front.
163   if (Left) {
164     NewText += Last->TokenText;
165     NewText += " ";
166   }
167 
168   // Then move through the other tokens.
169   auto *Tok = Begin;
170   while (Tok != End) {
171     if (!NewText.empty() && !endsWithSpace(NewText)) {
172       NewText += " ";
173     }
174 
175     NewText += Tok->TokenText;
176     Tok = Tok->Next;
177   }
178 
179   // If we are rotating to the right we move the first token to the back.
180   if (!Left) {
181     if (!NewText.empty() && !startsWithSpace(NewText)) {
182       NewText += " ";
183     }
184     NewText += First->TokenText;
185   }
186 
187   auto Range = CharSourceRange::getCharRange(First->getStartOfNonWhitespace(),
188                                              Last->Tok.getEndLoc());
189 
190   replaceToken(SourceMgr, Fixes, Range, NewText);
191 }
192 
193 FormatToken *LeftRightQualifierAlignmentFixer::analyzeRight(
194     const SourceManager &SourceMgr, const AdditionalKeywords &Keywords,
195     tooling::Replacements &Fixes, FormatToken *Tok,
196     const std::string &Qualifier, tok::TokenKind QualifierType) {
197   // We only need to think about streams that begin with a qualifier.
198   if (!Tok->is(QualifierType))
199     return Tok;
200   // Don't concern yourself if nothing follows the qualifier.
201   if (!Tok->Next)
202     return Tok;
203   if (LeftRightQualifierAlignmentFixer::isPossibleMacro(Tok->Next))
204     return Tok;
205 
206   FormatToken *Qual = Tok->Next;
207   FormatToken *LastQual = Qual;
208   while (Qual && isQualifierOrType(Qual, ConfiguredQualifierTokens)) {
209     LastQual = Qual;
210     Qual = Qual->Next;
211   }
212   if (LastQual && Qual != LastQual) {
213     rotateTokens(SourceMgr, Fixes, Tok, LastQual, /*Left=*/false);
214     Tok = LastQual;
215   } else if (Tok->startsSequence(QualifierType, tok::identifier,
216                                  TT_TemplateOpener)) {
217     // Read from the TemplateOpener to
218     // TemplateCloser as in const ArrayRef<int> a; const ArrayRef<int> &a;
219     FormatToken *EndTemplate = Tok->Next->Next->MatchingParen;
220     if (EndTemplate) {
221       // Move to the end of any template class members e.g.
222       // `Foo<int>::iterator`.
223       if (EndTemplate->startsSequence(TT_TemplateCloser, tok::coloncolon,
224                                       tok::identifier))
225         EndTemplate = EndTemplate->Next->Next;
226     }
227     if (EndTemplate && EndTemplate->Next &&
228         !EndTemplate->Next->isOneOf(tok::equal, tok::l_paren)) {
229       insertQualifierAfter(SourceMgr, Fixes, EndTemplate, Qualifier);
230       // Remove the qualifier.
231       removeToken(SourceMgr, Fixes, Tok);
232       return Tok;
233     }
234   } else if (Tok->startsSequence(QualifierType, tok::identifier)) {
235     FormatToken *Next = Tok->Next;
236     // The case  `const Foo` -> `Foo const`
237     // The case  `const Foo *` -> `Foo const *`
238     // The case  `const Foo &` -> `Foo const &`
239     // The case  `const Foo &&` -> `Foo const &&`
240     // The case  `const std::Foo &&` -> `std::Foo const &&`
241     // The case  `const std::Foo<T> &&` -> `std::Foo<T> const &&`
242     while (Next && Next->isOneOf(tok::identifier, tok::coloncolon)) {
243       Next = Next->Next;
244     }
245     if (Next && Next->is(TT_TemplateOpener)) {
246       Next = Next->MatchingParen;
247       // Move to the end of any template class members e.g.
248       // `Foo<int>::iterator`.
249       if (Next && Next->startsSequence(TT_TemplateCloser, tok::coloncolon,
250                                        tok::identifier)) {
251         Next = Next->Next->Next;
252         return Tok;
253       }
254       assert(Next && "Missing template opener");
255       Next = Next->Next;
256     }
257     if (Next && Next->isOneOf(tok::star, tok::amp, tok::ampamp) &&
258         !Tok->Next->isOneOf(Keywords.kw_override, Keywords.kw_final)) {
259       if (Next->Previous && !Next->Previous->is(QualifierType)) {
260         insertQualifierAfter(SourceMgr, Fixes, Next->Previous, Qualifier);
261         removeToken(SourceMgr, Fixes, Tok);
262       }
263       return Next;
264     }
265   }
266 
267   return Tok;
268 }
269 
270 FormatToken *LeftRightQualifierAlignmentFixer::analyzeLeft(
271     const SourceManager &SourceMgr, const AdditionalKeywords &Keywords,
272     tooling::Replacements &Fixes, FormatToken *Tok,
273     const std::string &Qualifier, tok::TokenKind QualifierType) {
274   // if Tok is an identifier and possibly a macro then don't convert.
275   if (LeftRightQualifierAlignmentFixer::isPossibleMacro(Tok))
276     return Tok;
277 
278   FormatToken *Qual = Tok;
279   FormatToken *LastQual = Qual;
280   while (Qual && isQualifierOrType(Qual, ConfiguredQualifierTokens)) {
281     LastQual = Qual;
282     Qual = Qual->Next;
283     if (Qual && Qual->is(QualifierType))
284       break;
285   }
286 
287   if (!Qual) {
288     return Tok;
289   }
290 
291   if (LastQual && Qual != LastQual && Qual->is(QualifierType)) {
292     rotateTokens(SourceMgr, Fixes, Tok, Qual, /*Left=*/true);
293     Tok = Qual->Next;
294   } else if (Tok->startsSequence(tok::identifier, QualifierType)) {
295     if (Tok->Next->Next && Tok->Next->Next->isOneOf(tok::identifier, tok::star,
296                                                     tok::amp, tok::ampamp)) {
297       // Don't swap `::iterator const` to `::const iterator`.
298       if (!Tok->Previous ||
299           (Tok->Previous && !Tok->Previous->is(tok::coloncolon))) {
300         rotateTokens(SourceMgr, Fixes, Tok, Tok->Next, /*Left=*/true);
301         Tok = Tok->Next;
302       }
303     }
304   }
305   if (Tok->is(TT_TemplateOpener) && Tok->Next &&
306       (Tok->Next->is(tok::identifier) || Tok->Next->isSimpleTypeSpecifier()) &&
307       Tok->Next->Next && Tok->Next->Next->is(QualifierType)) {
308     rotateTokens(SourceMgr, Fixes, Tok->Next, Tok->Next->Next, /*Left=*/true);
309   }
310   if (Tok->startsSequence(tok::identifier) && Tok->Next) {
311     if (Tok->Previous &&
312         Tok->Previous->isOneOf(tok::star, tok::ampamp, tok::amp)) {
313       return Tok;
314     }
315     FormatToken *Next = Tok->Next;
316     // The case  `std::Foo<T> const` -> `const std::Foo<T> &&`
317     while (Next && Next->isOneOf(tok::identifier, tok::coloncolon))
318       Next = Next->Next;
319     if (Next && Next->Previous &&
320         Next->Previous->startsSequence(tok::identifier, TT_TemplateOpener)) {
321       // Read from to the end of the TemplateOpener to
322       // TemplateCloser const ArrayRef<int> a; const ArrayRef<int> &a;
323       assert(Next->MatchingParen && "Missing template closer");
324       Next = Next->MatchingParen->Next;
325 
326       // Move to the end of any template class members e.g.
327       // `Foo<int>::iterator`.
328       if (Next && Next->startsSequence(tok::coloncolon, tok::identifier))
329         Next = Next->Next->Next;
330       if (Next && Next->is(QualifierType)) {
331         // Remove the const.
332         insertQualifierBefore(SourceMgr, Fixes, Tok, Qualifier);
333         removeToken(SourceMgr, Fixes, Next);
334         return Next;
335       }
336     }
337     if (Next && Next->Next &&
338         Next->Next->isOneOf(tok::amp, tok::ampamp, tok::star)) {
339       if (Next->is(QualifierType)) {
340         // Remove the qualifier.
341         insertQualifierBefore(SourceMgr, Fixes, Tok, Qualifier);
342         removeToken(SourceMgr, Fixes, Next);
343         return Next;
344       }
345     }
346   }
347   return Tok;
348 }
349 
350 tok::TokenKind LeftRightQualifierAlignmentFixer::getTokenFromQualifier(
351     const std::string &Qualifier) {
352   // don't let 'type' be an indentifier steal typeof token
353   return llvm::StringSwitch<tok::TokenKind>(Qualifier)
354       .Case("type", tok::kw_typeof)
355       .Case("const", tok::kw_const)
356       .Case("volatile", tok::kw_volatile)
357       .Case("static", tok::kw_static)
358       .Case("inline", tok::kw_inline)
359       .Case("constexpr", tok::kw_constexpr)
360       .Case("restrict", tok::kw_restrict)
361       .Default(tok::identifier);
362 }
363 
364 LeftRightQualifierAlignmentFixer::LeftRightQualifierAlignmentFixer(
365     const Environment &Env, const FormatStyle &Style,
366     const std::string &Qualifier,
367     const std::vector<tok::TokenKind> &QualifierTokens, bool RightAlign)
368     : TokenAnalyzer(Env, Style), Qualifier(Qualifier), RightAlign(RightAlign),
369       ConfiguredQualifierTokens(QualifierTokens) {}
370 
371 std::pair<tooling::Replacements, unsigned>
372 LeftRightQualifierAlignmentFixer::analyze(
373     TokenAnnotator &Annotator, SmallVectorImpl<AnnotatedLine *> &AnnotatedLines,
374     FormatTokenLexer &Tokens) {
375   tooling::Replacements Fixes;
376   const AdditionalKeywords &Keywords = Tokens.getKeywords();
377   const SourceManager &SourceMgr = Env.getSourceManager();
378   AffectedRangeMgr.computeAffectedLines(AnnotatedLines);
379 
380   tok::TokenKind QualifierToken = getTokenFromQualifier(Qualifier);
381   assert(QualifierToken != tok::identifier && "Unrecognised Qualifier");
382 
383   for (size_t I = 0, E = AnnotatedLines.size(); I != E; ++I) {
384     FormatToken *First = AnnotatedLines[I]->First;
385     const auto *Last = AnnotatedLines[I]->Last;
386 
387     for (auto *Tok = First; Tok && Tok != Last && Tok->Next; Tok = Tok->Next) {
388       if (Tok->is(tok::comment))
389         continue;
390       if (RightAlign)
391         Tok = analyzeRight(SourceMgr, Keywords, Fixes, Tok, Qualifier,
392                            QualifierToken);
393       else
394         Tok = analyzeLeft(SourceMgr, Keywords, Fixes, Tok, Qualifier,
395                           QualifierToken);
396     }
397   }
398   return {Fixes, 0};
399 }
400 
401 void QualifierAlignmentFixer::PrepareLeftRightOrdering(
402     const std::vector<std::string> &Order, std::vector<std::string> &LeftOrder,
403     std::vector<std::string> &RightOrder,
404     std::vector<tok::TokenKind> &Qualifiers) {
405 
406   // Depending on the position of type in the order you need
407   // To iterate forward or backward through the order list as qualifier
408   // can push through each other.
409   auto type = std::find(Order.begin(), Order.end(), "type");
410   // The Order list must define the position of "type" to signify
411   assert(type != Order.end() && "QualifierOrder must contain type");
412   // Split the Order list by type and reverse the left side.
413 
414   bool left = true;
415   for (const auto &s : Order) {
416     if (s == "type") {
417       left = false;
418       continue;
419     }
420 
421     tok::TokenKind QualifierToken =
422         LeftRightQualifierAlignmentFixer::getTokenFromQualifier(s);
423     if (QualifierToken != tok::kw_typeof && QualifierToken != tok::identifier) {
424       Qualifiers.push_back(QualifierToken);
425     }
426 
427     if (left)
428       // Reverse the order for left aligned items.
429       LeftOrder.insert(LeftOrder.begin(), s);
430     else
431       RightOrder.push_back(s);
432   }
433 }
434 
435 bool LeftRightQualifierAlignmentFixer::isQualifierOrType(
436     const FormatToken *Tok, const std::vector<tok::TokenKind> &specifiedTypes) {
437   return Tok && (Tok->isSimpleTypeSpecifier() || Tok->is(tok::kw_auto) ||
438                  (std::find(specifiedTypes.begin(), specifiedTypes.end(),
439                             Tok->Tok.getKind()) != specifiedTypes.end()));
440 }
441 
442 // If a token is an identifier and it's upper case, it could
443 // be a macro and hence we need to be able to ignore it.
444 bool LeftRightQualifierAlignmentFixer::isPossibleMacro(const FormatToken *Tok) {
445   if (!Tok)
446     return false;
447   if (!Tok->is(tok::identifier))
448     return false;
449   if (Tok->TokenText.upper() == Tok->TokenText.str())
450     return true;
451   return false;
452 }
453 
454 } // namespace format
455 } // namespace clang
456