PR10954: variant members should not be implicitly initialized in constructors if no

mem-initializer is specified for them, unless an in-class initializer is specified.

llvm-svn: 139996
This commit is contained in:
Richard Smith 2011-09-18 11:14:50 +00:00
parent 7ae11279e9
commit 12d5ed8850
2 changed files with 34 additions and 12 deletions

View File

@ -2291,6 +2291,11 @@ static bool CollectFieldInitializer(Sema &SemaRef, BaseAndFieldInfo &Info,
return false;
}
// Don't build an implicit initializer for union members if none was
// explicitly specified.
if (Field->getParent()->isUnion())
return false;
// Don't try to build an implicit initializer if there were semantic
// errors in any of the initializers (and therefore we might be
// missing some that the user actually wrote).
@ -2464,17 +2469,6 @@ bool Sema::SetCtorInitializers(CXXConstructorDecl *Constructor,
continue;
}
// If this field is somewhere within an anonymous union, we only
// initialize it if there's an explicit initializer.
if (isWithinAnonymousUnion(F)) {
if (CXXCtorInitializer *Init
= Info.AllBaseFields.lookup(F->getAnonField())) {
Info.AllToInit.push_back(Init);
}
continue;
}
// Initialize each field of an anonymous struct individually.
if (CollectFieldInitializer(*this, Info, F->getAnonField(), F))
HadError = true;

View File

@ -5,11 +5,15 @@ struct S {
int &a; // expected-note 2{{here}}
int &b = n;
union {
const int k = 42;
};
S() {} // expected-error {{constructor for 'S' must explicitly initialize the reference member 'a'}}
S(int) : a(n) {} // ok
S(char) : b(n) {} // expected-error {{constructor for 'S' must explicitly initialize the reference member 'a'}}
S(double) : a(n), b(n) {} // ok
};
} s(0);
union U {
int a = 0;
@ -21,3 +25,27 @@ union U {
U(char) : b('y') {} // desired-error {{at most one member of a union may be initialized}}
U(double) : a(1), b('y') {} // desired-error {{at most one member of a union may be initialized}}
};
// PR10954: variant members do not acquire an implicit initializer.
namespace VariantMembers {
struct NoDefaultCtor {
NoDefaultCtor(int);
};
union V {
NoDefaultCtor ndc;
int n;
V() {}
V(int n) : n(n) {}
V(int n, bool) : ndc(n) {}
};
struct K {
union {
NoDefaultCtor ndc;
int n;
};
K() {}
K(int n) : n(n) {}
K(int n, bool) : ndc(n) {}
};
}