Teach the uninitialized field warning about anonymous structs and union members.

Fixes PR14073!


git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@168031 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Nick Lewycky 2012-11-15 08:19:20 +00:00
Родитель 931c083381
Коммит 621ba4f0db
2 изменённых файлов: 29 добавлений и 9 удалений

Просмотреть файл

@ -1795,7 +1795,11 @@ namespace {
public:
typedef EvaluatedExprVisitor<UninitializedFieldVisitor> Inherited;
UninitializedFieldVisitor(Sema &S, ValueDecl *VD) : Inherited(S.Context),
S(S), VD(VD) {
S(S) {
if (IndirectFieldDecl *IFD = dyn_cast<IndirectFieldDecl>(VD))
this->VD = IFD->getAnonField();
else
this->VD = VD;
}
void HandleExpr(Expr *E) {
@ -1812,23 +1816,33 @@ namespace {
if (MemberExpr *ME = dyn_cast<MemberExpr>(E)) {
if (isa<EnumConstantDecl>(ME->getMemberDecl()))
return;
return;
// FieldME is the inner-most MemberExpr that is not an anonymous struct
// or union.
MemberExpr *FieldME = ME;
Expr *Base = E;
while (isa<MemberExpr>(Base)) {
ME = dyn_cast<MemberExpr>(Base);
if (VarDecl *VarD = dyn_cast<VarDecl>(ME->getMemberDecl()))
if (VarD->hasGlobalStorage())
return;
ME = cast<MemberExpr>(Base);
if (isa<VarDecl>(ME->getMemberDecl()))
return;
if (FieldDecl *FD = dyn_cast<FieldDecl>(ME->getMemberDecl()))
if (!FD->isAnonymousStructOrUnion())
FieldME = ME;
Base = ME->getBase();
}
if (VD == ME->getMemberDecl() && isa<CXXThisExpr>(Base)) {
if (VD == FieldME->getMemberDecl() && isa<CXXThisExpr>(Base)) {
unsigned diag = VD->getType()->isReferenceType()
? diag::warn_reference_field_is_uninit
: diag::warn_field_is_uninit;
S.Diag(ME->getExprLoc(), diag) << ME->getMemberNameInfo().getName();
return;
S.Diag(FieldME->getExprLoc(), diag) << VD;
}
return;
}
if (ConditionalOperator *CO = dyn_cast<ConditionalOperator>(E)) {

Просмотреть файл

@ -283,3 +283,9 @@ namespace PR12049 {
int member; // expected-error {{expected ')'}}
};
}
namespace PR14073 {
struct S1 { union { int n; }; S1() : n(n) {} }; // expected-warning {{field 'n' is uninitialized when used here}}
struct S2 { union { union { int n; }; char c; }; S2() : n(n) {} }; // expected-warning {{field 'n' is uninitialized when used here}}
struct S3 { struct { int n; }; S3() : n(n) {} }; // expected-warning {{field 'n' is uninitialized when used here}}
}