diff options
author | Richard Smith <richard-llvm@metafoo.co.uk> | 2011-09-18 11:14:50 +0000 |
---|---|---|
committer | Richard Smith <richard-llvm@metafoo.co.uk> | 2011-09-18 11:14:50 +0000 |
commit | c115f630b748b2f27cff34137fb6a50b10a471a5 (patch) | |
tree | 5df0cf9a2ef3f806d7973c657f0d28c7ba051cf7 | |
parent | 16ee8191ca6953e173c983e3175a8d7504b5baae (diff) |
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.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@139996 91177308-0d34-0410-b5e6-96231b3b80d8
-rw-r--r-- | lib/Sema/SemaDeclCXX.cpp | 16 | ||||
-rw-r--r-- | test/CXX/special/class.init/class.base.init/p8-0x.cpp | 30 |
2 files changed, 34 insertions, 12 deletions
diff --git a/lib/Sema/SemaDeclCXX.cpp b/lib/Sema/SemaDeclCXX.cpp index 0f7af4e30d..b408f7f088 100644 --- a/lib/Sema/SemaDeclCXX.cpp +++ b/lib/Sema/SemaDeclCXX.cpp @@ -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; diff --git a/test/CXX/special/class.init/class.base.init/p8-0x.cpp b/test/CXX/special/class.init/class.base.init/p8-0x.cpp index 8512a9f7bb..81ed9c2570 100644 --- a/test/CXX/special/class.init/class.base.init/p8-0x.cpp +++ b/test/CXX/special/class.init/class.base.init/p8-0x.cpp @@ -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) {} + }; +} |