xref: /aosp_15_r20/external/llvm/lib/Support/LineIterator.cpp (revision 9880d6810fe72a1726cb53787c6711e909410d58)
1*9880d681SAndroid Build Coastguard Worker //===- LineIterator.cpp - Implementation of line iteration ----------------===//
2*9880d681SAndroid Build Coastguard Worker //
3*9880d681SAndroid Build Coastguard Worker //                     The LLVM Compiler Infrastructure
4*9880d681SAndroid Build Coastguard Worker //
5*9880d681SAndroid Build Coastguard Worker // This file is distributed under the University of Illinois Open Source
6*9880d681SAndroid Build Coastguard Worker // License. See LICENSE.TXT for details.
7*9880d681SAndroid Build Coastguard Worker //
8*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
9*9880d681SAndroid Build Coastguard Worker 
10*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/LineIterator.h"
11*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/MemoryBuffer.h"
12*9880d681SAndroid Build Coastguard Worker 
13*9880d681SAndroid Build Coastguard Worker using namespace llvm;
14*9880d681SAndroid Build Coastguard Worker 
isAtLineEnd(const char * P)15*9880d681SAndroid Build Coastguard Worker static bool isAtLineEnd(const char *P) {
16*9880d681SAndroid Build Coastguard Worker   if (*P == '\n')
17*9880d681SAndroid Build Coastguard Worker     return true;
18*9880d681SAndroid Build Coastguard Worker   if (*P == '\r' && *(P + 1) == '\n')
19*9880d681SAndroid Build Coastguard Worker     return true;
20*9880d681SAndroid Build Coastguard Worker   return false;
21*9880d681SAndroid Build Coastguard Worker }
22*9880d681SAndroid Build Coastguard Worker 
skipIfAtLineEnd(const char * & P)23*9880d681SAndroid Build Coastguard Worker static bool skipIfAtLineEnd(const char *&P) {
24*9880d681SAndroid Build Coastguard Worker   if (*P == '\n') {
25*9880d681SAndroid Build Coastguard Worker     ++P;
26*9880d681SAndroid Build Coastguard Worker     return true;
27*9880d681SAndroid Build Coastguard Worker   }
28*9880d681SAndroid Build Coastguard Worker   if (*P == '\r' && *(P + 1) == '\n') {
29*9880d681SAndroid Build Coastguard Worker     P += 2;
30*9880d681SAndroid Build Coastguard Worker     return true;
31*9880d681SAndroid Build Coastguard Worker   }
32*9880d681SAndroid Build Coastguard Worker   return false;
33*9880d681SAndroid Build Coastguard Worker }
34*9880d681SAndroid Build Coastguard Worker 
line_iterator(const MemoryBuffer & Buffer,bool SkipBlanks,char CommentMarker)35*9880d681SAndroid Build Coastguard Worker line_iterator::line_iterator(const MemoryBuffer &Buffer, bool SkipBlanks,
36*9880d681SAndroid Build Coastguard Worker                              char CommentMarker)
37*9880d681SAndroid Build Coastguard Worker     : Buffer(Buffer.getBufferSize() ? &Buffer : nullptr),
38*9880d681SAndroid Build Coastguard Worker       CommentMarker(CommentMarker), SkipBlanks(SkipBlanks), LineNumber(1),
39*9880d681SAndroid Build Coastguard Worker       CurrentLine(Buffer.getBufferSize() ? Buffer.getBufferStart() : nullptr,
40*9880d681SAndroid Build Coastguard Worker                   0) {
41*9880d681SAndroid Build Coastguard Worker   // Ensure that if we are constructed on a non-empty memory buffer that it is
42*9880d681SAndroid Build Coastguard Worker   // a null terminated buffer.
43*9880d681SAndroid Build Coastguard Worker   if (Buffer.getBufferSize()) {
44*9880d681SAndroid Build Coastguard Worker     assert(Buffer.getBufferEnd()[0] == '\0');
45*9880d681SAndroid Build Coastguard Worker     // Make sure we don't skip a leading newline if we're keeping blanks
46*9880d681SAndroid Build Coastguard Worker     if (SkipBlanks || !isAtLineEnd(Buffer.getBufferStart()))
47*9880d681SAndroid Build Coastguard Worker       advance();
48*9880d681SAndroid Build Coastguard Worker   }
49*9880d681SAndroid Build Coastguard Worker }
50*9880d681SAndroid Build Coastguard Worker 
advance()51*9880d681SAndroid Build Coastguard Worker void line_iterator::advance() {
52*9880d681SAndroid Build Coastguard Worker   assert(Buffer && "Cannot advance past the end!");
53*9880d681SAndroid Build Coastguard Worker 
54*9880d681SAndroid Build Coastguard Worker   const char *Pos = CurrentLine.end();
55*9880d681SAndroid Build Coastguard Worker   assert(Pos == Buffer->getBufferStart() || isAtLineEnd(Pos) || *Pos == '\0');
56*9880d681SAndroid Build Coastguard Worker 
57*9880d681SAndroid Build Coastguard Worker   if (skipIfAtLineEnd(Pos))
58*9880d681SAndroid Build Coastguard Worker     ++LineNumber;
59*9880d681SAndroid Build Coastguard Worker   if (!SkipBlanks && isAtLineEnd(Pos)) {
60*9880d681SAndroid Build Coastguard Worker     // Nothing to do for a blank line.
61*9880d681SAndroid Build Coastguard Worker   } else if (CommentMarker == '\0') {
62*9880d681SAndroid Build Coastguard Worker     // If we're not stripping comments, this is simpler.
63*9880d681SAndroid Build Coastguard Worker     while (skipIfAtLineEnd(Pos))
64*9880d681SAndroid Build Coastguard Worker       ++LineNumber;
65*9880d681SAndroid Build Coastguard Worker   } else {
66*9880d681SAndroid Build Coastguard Worker     // Skip comments and count line numbers, which is a bit more complex.
67*9880d681SAndroid Build Coastguard Worker     for (;;) {
68*9880d681SAndroid Build Coastguard Worker       if (isAtLineEnd(Pos) && !SkipBlanks)
69*9880d681SAndroid Build Coastguard Worker         break;
70*9880d681SAndroid Build Coastguard Worker       if (*Pos == CommentMarker)
71*9880d681SAndroid Build Coastguard Worker         do {
72*9880d681SAndroid Build Coastguard Worker           ++Pos;
73*9880d681SAndroid Build Coastguard Worker         } while (*Pos != '\0' && !isAtLineEnd(Pos));
74*9880d681SAndroid Build Coastguard Worker       if (!skipIfAtLineEnd(Pos))
75*9880d681SAndroid Build Coastguard Worker         break;
76*9880d681SAndroid Build Coastguard Worker       ++LineNumber;
77*9880d681SAndroid Build Coastguard Worker     }
78*9880d681SAndroid Build Coastguard Worker   }
79*9880d681SAndroid Build Coastguard Worker 
80*9880d681SAndroid Build Coastguard Worker   if (*Pos == '\0') {
81*9880d681SAndroid Build Coastguard Worker     // We've hit the end of the buffer, reset ourselves to the end state.
82*9880d681SAndroid Build Coastguard Worker     Buffer = nullptr;
83*9880d681SAndroid Build Coastguard Worker     CurrentLine = StringRef();
84*9880d681SAndroid Build Coastguard Worker     return;
85*9880d681SAndroid Build Coastguard Worker   }
86*9880d681SAndroid Build Coastguard Worker 
87*9880d681SAndroid Build Coastguard Worker   // Measure the line.
88*9880d681SAndroid Build Coastguard Worker   size_t Length = 0;
89*9880d681SAndroid Build Coastguard Worker   while (Pos[Length] != '\0' && !isAtLineEnd(&Pos[Length])) {
90*9880d681SAndroid Build Coastguard Worker     ++Length;
91*9880d681SAndroid Build Coastguard Worker   }
92*9880d681SAndroid Build Coastguard Worker 
93*9880d681SAndroid Build Coastguard Worker   CurrentLine = StringRef(Pos, Length);
94*9880d681SAndroid Build Coastguard Worker }
95