From e6ac50ad71b7dd63e5b746fce9eb3a1e76debb01 Mon Sep 17 00:00:00 2001 From: Dmitri Gribenko Date: Thu, 5 Dec 2013 23:06:53 +0000 Subject: [PATCH] -Wassign-enum: compare unqualified types This commit changes -Wassign-enum to compare unqualified types. One could think that this does not matter much, because who wants a value of enum type that is const-qualified? But this breaks the intended pattern to silence this warning with an explicit cast: static const enum Foo z = (enum Foo) 42; In this case, source type is 'enum Foo', and destination type is 'const enum Foo', and if we compare qualified types, they don't match, so we used warn. llvm-svn: 196548 --- clang/lib/Sema/SemaStmt.cpp | 4 ++-- clang/test/Sema/warn-outof-range-assign-enum.c | 18 ++++++++++++++++++ 2 files changed, 20 insertions(+), 2 deletions(-) diff --git a/clang/lib/Sema/SemaStmt.cpp b/clang/lib/Sema/SemaStmt.cpp index 077282c1bf17..1d2ebad78810 100644 --- a/clang/lib/Sema/SemaStmt.cpp +++ b/clang/lib/Sema/SemaStmt.cpp @@ -1149,7 +1149,7 @@ Sema::DiagnoseAssignmentEnum(QualType DstType, QualType SrcType, return; if (const EnumType *ET = DstType->getAs()) - if (!Context.hasSameType(SrcType, DstType) && + if (!Context.hasSameUnqualifiedType(SrcType, DstType) && SrcType->isIntegerType()) { if (!SrcExpr->isTypeDependent() && !SrcExpr->isValueDependent() && SrcExpr->isIntegerConstantExpr(Context)) { @@ -1184,7 +1184,7 @@ Sema::DiagnoseAssignmentEnum(QualType DstType, QualType SrcType, EI++; if (EI == EIend || EI->first != RhsVal) { Diag(SrcExpr->getExprLoc(), diag::warn_not_in_enum_assignment) - << DstType; + << DstType.getUnqualifiedType(); } } } diff --git a/clang/test/Sema/warn-outof-range-assign-enum.c b/clang/test/Sema/warn-outof-range-assign-enum.c index 43eea0cef41f..edd4e3722927 100644 --- a/clang/test/Sema/warn-outof-range-assign-enum.c +++ b/clang/test/Sema/warn-outof-range-assign-enum.c @@ -11,6 +11,24 @@ typedef enum CCTestEnum CCTestEnum test = 50; // expected-warning {{integer constant not in range of enumerated type 'CCTestEnum'}} CCTestEnum test1 = -50; // expected-warning {{integer constant not in range of enumerated type 'CCTestEnum'}} +// Explicit cast should silence the warning. +static const CCTestEnum SilenceWithCast1 = 51; // expected-warning {{integer constant not in range of enumerated type 'CCTestEnum'}} +static const CCTestEnum SilenceWithCast2 = (CCTestEnum) 51; // no-warning +static const CCTestEnum SilenceWithCast3 = (const CCTestEnum) 51; // no-warning +static const CCTestEnum SilenceWithCast4 = (const volatile CCTestEnum) 51; // no-warning + +void SilenceWithCastLocalVar() { + CCTestEnum SilenceWithCast1 = 51; // expected-warning {{integer constant not in range of enumerated type 'CCTestEnum'}} + CCTestEnum SilenceWithCast2 = (CCTestEnum) 51; // no-warning + CCTestEnum SilenceWithCast3 = (const CCTestEnum) 51; // no-warning + CCTestEnum SilenceWithCast4 = (const volatile CCTestEnum) 51; // no-warning + + const CCTestEnum SilenceWithCast1c = 51; // expected-warning {{integer constant not in range of enumerated type 'CCTestEnum'}} + const CCTestEnum SilenceWithCast2c = (CCTestEnum) 51; // no-warning + const CCTestEnum SilenceWithCast3c = (const CCTestEnum) 51; // no-warning + const CCTestEnum SilenceWithCast4c = (const volatile CCTestEnum) 51; // no-warning +} + CCTestEnum foo(CCTestEnum r) { return 20; // expected-warning {{integer constant not in range of enumerated type 'CCTestEnum'}} }