Index: clang/include/clang/Basic/SourceManager.h =================================================================== --- clang/include/clang/Basic/SourceManager.h +++ clang/include/clang/Basic/SourceManager.h @@ -1390,8 +1390,11 @@ /// returns zero if the column number isn't known. This may only be called /// on a file sloc, so you must choose a spelling or expansion location /// before calling this method. + /// When UseTabStopOption is true tabs will take the TabStop (-ftabstop=x) + /// option in account. Otherwise tabs are counted as 1. unsigned getColumnNumber(FileID FID, unsigned FilePos, - bool *Invalid = nullptr) const; + bool *Invalid = nullptr, + bool UseTabStopOption = false) const; unsigned getSpellingColumnNumber(SourceLocation Loc, bool *Invalid = nullptr) const; unsigned getExpansionColumnNumber(SourceLocation Loc, @@ -1399,6 +1402,11 @@ unsigned getPresumedColumnNumber(SourceLocation Loc, bool *Invalid = nullptr) const; + /// Return a number suitable to compare the indentation of source + /// locations. + unsigned getVisualColumnNumber(SourceLocation Loc, + bool *Invalid = nullptr) const; + /// Given a SourceLocation, return the spelling line number /// for the position indicated. /// Index: clang/lib/Basic/SourceManager.cpp =================================================================== --- clang/lib/Basic/SourceManager.cpp +++ clang/lib/Basic/SourceManager.cpp @@ -1149,7 +1149,8 @@ /// getColumnNumber - Return the column # for the specified file position. /// this is significantly cheaper to compute than the line number. unsigned SourceManager::getColumnNumber(FileID FID, unsigned FilePos, - bool *Invalid) const { + bool *Invalid, + bool UseTabStopOption) const { bool MyInvalid = false; const llvm::MemoryBuffer *MemBuf = getBuffer(FID, &MyInvalid); if (Invalid) @@ -1165,6 +1166,9 @@ return 1; } + unsigned LineStart; + bool HasCachedResult = false; + const char *Buf = MemBuf->getBufferStart(); // See if we just calculated the line number for this FilePos and can use // that to lookup the start of the line instead of searching for it. @@ -1172,9 +1176,10 @@ LastLineNoContentCache->SourceLineCache != nullptr && LastLineNoResult < LastLineNoContentCache->NumLines) { unsigned *SourceLineCache = LastLineNoContentCache->SourceLineCache; - unsigned LineStart = SourceLineCache[LastLineNoResult - 1]; + LineStart = SourceLineCache[LastLineNoResult - 1]; unsigned LineEnd = SourceLineCache[LastLineNoResult]; if (FilePos >= LineStart && FilePos < LineEnd) { + HasCachedResult = true; // LineEnd is the LineStart of the next line. // A line ends with separator LF or CR+LF on Windows. // FilePos might point to the last separator, @@ -1183,14 +1188,26 @@ if (Buf[FilePos - 1] == '\r' || Buf[FilePos - 1] == '\n') --FilePos; } - return FilePos - LineStart + 1; } } - - unsigned LineStart = FilePos; - while (LineStart && Buf[LineStart-1] != '\n' && Buf[LineStart-1] != '\r') - --LineStart; - return FilePos-LineStart+1; + if (!HasCachedResult) { + LineStart = FilePos; + while (LineStart && Buf[LineStart - 1] != '\n' && + Buf[LineStart - 1] != '\r') + --LineStart; + } + unsigned TabStop = + UseTabStopOption ? getDiagnostics().getDiagnosticOptions().TabStop : 1; + if (TabStop == 1) + return FilePos - LineStart + 1; + unsigned VirtualCharCount = 0; + while (LineStart < FilePos) { + VirtualCharCount = Buf[LineStart] == '\t' + ? llvm::alignTo(VirtualCharCount + 1, TabStop) + : VirtualCharCount + 1; + LineStart++; + } + return VirtualCharCount + 1; } // isInvalid - Return the result of calling loc.isInvalid(), and @@ -1210,6 +1227,14 @@ return getColumnNumber(LocInfo.first, LocInfo.second, Invalid); } +unsigned SourceManager::getVisualColumnNumber(SourceLocation Loc, + bool *Invalid) const { + if (isInvalid(Loc, Invalid)) + return 0; + std::pair LocInfo = getDecomposedSpellingLoc(Loc); + return getColumnNumber(LocInfo.first, LocInfo.second, Invalid, /*UseTabStopOption*/true); +} + unsigned SourceManager::getExpansionColumnNumber(SourceLocation Loc, bool *Invalid) const { if (isInvalid(Loc, Invalid)) return 0; Index: clang/lib/Parse/ParseStmt.cpp =================================================================== --- clang/lib/Parse/ParseStmt.cpp +++ clang/lib/Parse/ParseStmt.cpp @@ -1230,9 +1230,9 @@ P.MisleadingIndentationElseLoc = SourceLocation(); SourceManager &SM = P.getPreprocessor().getSourceManager(); - unsigned PrevColNum = SM.getSpellingColumnNumber(PrevLoc); - unsigned CurColNum = SM.getSpellingColumnNumber(Tok.getLocation()); - unsigned StmtColNum = SM.getSpellingColumnNumber(StmtLoc); + unsigned PrevColNum = SM.getVisualColumnNumber(PrevLoc); + unsigned CurColNum = SM.getVisualColumnNumber(Tok.getLocation()); + unsigned StmtColNum = SM.getVisualColumnNumber(StmtLoc); if (PrevColNum != 0 && CurColNum != 0 && StmtColNum != 0 && ((PrevColNum > StmtColNum && PrevColNum == CurColNum) || Index: clang/test/Parser/warn-misleading-indentation.cpp =================================================================== --- clang/test/Parser/warn-misleading-indentation.cpp +++ clang/test/Parser/warn-misleading-indentation.cpp @@ -1,7 +1,9 @@ // RUN: %clang_cc1 -x c -fsyntax-only -verify %s -// RUN: %clang_cc1 -x c -fsyntax-only -verify -Wmisleading-indentation -DWITH_WARN %s -// RUN: %clang_cc1 -std=c++17 -fsyntax-only -verify -Wall -Wno-unused -DWITH_WARN -DCXX17 %s // RUN: %clang_cc1 -std=c++17 -fsyntax-only -verify -Wall -Wno-unused -Wno-misleading-indentation -DCXX17 %s +// RUN: %clang_cc1 -x c -fsyntax-only -verify -Wmisleading-indentation -DWITH_WARN -ftabstop 8 -DTAB_SIZE=8 %s +// RUN: %clang_cc1 -std=c++17 -fsyntax-only -verify -Wall -Wno-unused -DWITH_WARN -ftabstop 4 -DTAB_SIZE=4 -DCXX17 %s +// RUN: %clang_cc1 -x c -fsyntax-only -verify -Wall -Wno-unused -DWITH_WARN -ftabstop 1 -DTAB_SIZE=1 %s +// RUN: %clang_cc1 -std=c++17 -fsyntax-only -verify -Wall -Wno-unused -Wmisleading-indentation -DCXX17 -DWITH_WARN -ftabstop 2 -DTAB_SIZE=2 %s #ifndef WITH_WARN // expected-no-diagnostics @@ -225,3 +227,80 @@ // expected-warning@-2 {{misleading indentation; statement is not part of the previous 'if'}} #endif } +int a4() +{ + if (0) + return 1; + return 0; +#if (TAB_SIZE == 1) +// expected-warning@-2 {{misleading indentation; statement is not part of the previous 'if'}} +// expected-note@-5 {{here}} +#endif +} + +int a5() +{ + if (0) + return 1; + return 0; +#if WITH_WARN +// expected-warning@-2 {{misleading indentation; statement is not part of the previous 'if'}} +// expected-note@-5 {{here}} +#endif +} + +int a6() +{ + if (0) + return 1; + return 0; +#if (TAB_SIZE == 8) +// expected-warning@-2 {{misleading indentation; statement is not part of the previous 'if'}} +// expected-note@-5 {{here}} +#endif +} + +#define FOO \ + goto fail + +int main(int argc, char* argv[]) { + if (5 != 0) + goto fail; + else + goto fail; + + if (1) { + if (1) + goto fail; + else if (1) + goto fail; + else if (1) + goto fail; + else + goto fail; + } else if (1) { + if (1) + goto fail; + } + + if (1) { + if (1) + goto fail; + } else if (1) + goto fail; + + + if (1) goto fail; goto fail; + + if (0) + goto fail; + + goto fail; + + if (0) + FOO; + + goto fail; + +fail:; +}