llvm-capstone/clang/test/Misc/diag-greatergreater.cpp
Richard Smith b5f8171a1b PR37189 Fix incorrect end source location and spelling for a split '>>' token.
When a '>>' token is split into two '>' tokens (in C++11 onwards), or (as an
extension) when we do the same for other tokens starting with a '>', we can't
just use a location pointing to the first '>' as the location of the split
token, because that would result in our miscomputing the length and spelling
for the token. As a consequence, for example, a refactoring replacing 'A<X>'
with something else would sometimes replace one character too many, and
similarly diagnostics highlighting a template-id source range would highlight
one character too many.

Fix this by creating an expansion range covering the first character of the
'>>' token, whose spelling is '>'. For this to work, we generalize the
expansion range of a macro FileID to be either a token range (the common case)
or a character range (used in this new case).

llvm-svn: 331155
2018-04-30 05:25:48 +00:00

44 lines
1.5 KiB
C++

// RUN: not %clang_cc1 %s -fdiagnostics-print-source-range-info 2>&1 | FileCheck %s --strict-whitespace
template<typename T> class C {};
template<int> class D {};
void g() {
// The range ending in the first > character does not extend to the second >
// character.
// CHECK: :{[[@LINE+3]]:5-[[@LINE+3]]:11}: error:
// CHECK-NEXT: D<C<int>> a;
// CHECK-NEXT: ^~~~~~{{$}}
D<C<int>> a;
// The range ending in the second > character does not extend to the third >
// character.
// CHECK: :{[[@LINE+3]]:5-[[@LINE+3]]:14}: error:
// CHECK-NEXT: D<C<C<int>>> b;
// CHECK-NEXT: ^~~~~~~~~{{$}}
D<C<C<int>>> b;
}
template<int> int V;
// Here, we split the >>= token into a > followed by a >=.
// Then we split the >= token into a > followed by an =,
// which we merge with the other = to form an ==.
// CHECK: error: a space is required
// CHECK-NEXT: int k = V<C<int>>==0;
// CHECK-NEXT: ^~{{$}}
// CHECK-NEXT: > >{{$}}
// CHECK: error: a space is required
// CHECK-NEXT: int k = V<C<int>>==0;
// CHECK-NEXT: ^~{{$}}
// CHECK-NEXT: > ={{$}}
// CHECK: :{[[@LINE+3]]:11-[[@LINE+3]]:17}: error:
// CHECK-NEXT: int k = V<C<int>>==0;
// CHECK-NEXT: ^~~~~~{{$}}
int k = V<C<int>>==0;
template<typename> int W;
// CHECK: :{[[@LINE+3]]:9-[[@LINE+3]]:18}{[[@LINE+3]]:20-[[@LINE+3]]:22}: error: comparison
// CHECK-NEXT: int l = W<C<int>>==&k;
// CHECK-NEXT: ~~~~~~~~~^ ~~{{$}}
int l = W<C<int>>==&k;