diff options
author | Chris Lattner <sabre@nondot.org> | 2009-02-19 23:45:49 +0000 |
---|---|---|
committer | Chris Lattner <sabre@nondot.org> | 2009-02-19 23:45:49 +0000 |
commit | d0344a4a6182ad704881cbbaa21cca14913d2296 (patch) | |
tree | 61b23aebb6f09877ef079f9fc32c30ff249da2f5 /lib/Sema/Sema.cpp | |
parent | 58e899b336c63fa25d4cc8986d97a40933cded9b (diff) |
Fix a long standard problem with clang retaining "too much" sugar
information about types. We often print diagnostics where we say
"foo_t" is bad, but the user doesn't know how foo_t is declared
(because it is a typedef). Fix this by expanding sugar when present
in a diagnostic (and not one of a few special cases, like vectors).
Before:
t.m:5:2: error: invalid operands to binary expression ('typeof(P)' and 'typeof(F)')
MAX(P, F);
^~~~~~~~~
t.m:1:78: note: instantiated from:
#define MAX(A,B) ({ __typeof__(A) __a = (A); __typeof__(B) __b = (B); __a < __b ? __b : __a; })
^
After:
t.m:5:2: error: invalid operands to binary expression ('typeof(P)' (aka 'struct mystruct') and 'typeof(F)' (aka 'float'))
MAX(P, F);
^~~~~~~~~
t.m:1:78: note: instantiated from:
#define MAX(A,B) ({ __typeof__(A) __a = (A); __typeof__(B) __b = (B); __a < __b ? __b : __a; })
^
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@65081 91177308-0d34-0410-b5e6-96231b3b80d8
Diffstat (limited to 'lib/Sema/Sema.cpp')
-rw-r--r-- | lib/Sema/Sema.cpp | 35 |
1 files changed, 29 insertions, 6 deletions
diff --git a/lib/Sema/Sema.cpp b/lib/Sema/Sema.cpp index 539457f8a6..1ffaf939ef 100644 --- a/lib/Sema/Sema.cpp +++ b/lib/Sema/Sema.cpp @@ -22,19 +22,39 @@ using namespace clang; /// ConvertQualTypeToStringFn - This function is used to pretty print the /// specified QualType as a string in diagnostics. static void ConvertArgToStringFn(Diagnostic::ArgumentKind Kind, intptr_t Val, - const char *Modifier, unsigned ModLen, - const char *Argument, unsigned ArgLen, - llvm::SmallVectorImpl<char> &Output) { + const char *Modifier, unsigned ModLen, + const char *Argument, unsigned ArgLen, + llvm::SmallVectorImpl<char> &Output) { std::string S; if (Kind == Diagnostic::ak_qualtype) { + assert(ModLen == 0 && ArgLen == 0 && + "Invalid modifier for QualType argument"); + QualType Ty(QualType::getFromOpaquePtr(reinterpret_cast<void*>(Val))); // FIXME: Playing with std::string is really slow. S = Ty.getAsString(); - - assert(ModLen == 0 && ArgLen == 0 && - "Invalid modifier for QualType argument"); + + // If this is a sugared type (like a typedef, typeof, etc), then unwrap one + // level of the sugar so that the type is more obvious to the user. + QualType DesugaredTy = Ty->getDesugaredType(); + DesugaredTy.setCVRQualifiers(DesugaredTy.getCVRQualifiers() | + Ty.getCVRQualifiers()); + + if (Ty != DesugaredTy && + // If the desugared type is a vector type, we don't want to expand it, + // it will turn into an attribute mess. People want their "vec4". + !isa<VectorType>(DesugaredTy) && + + // Don't desugar objc types. FIXME: THIS IS A HACK. + S != "id" && S != "Class") { + S = "'"+S+"' (aka '"; + S += DesugaredTy.getAsString(); + S += "')"; + Output.append(S.begin(), S.end()); + return; + } } else if (Kind == Diagnostic::ak_declarationname) { @@ -58,7 +78,10 @@ static void ConvertArgToStringFn(Diagnostic::ArgumentKind Kind, intptr_t Val, S = reinterpret_cast<NamedDecl*>(Val)->getNameAsString(); } } + + Output.push_back('\''); Output.append(S.begin(), S.end()); + Output.push_back('\''); } |