summaryrefslogtreecommitdiffstats
path: root/clang
diff options
context:
space:
mode:
authorDouglas Gregor <dgregor@apple.com>2012-02-24 21:25:53 +0000
committerDouglas Gregor <dgregor@apple.com>2012-02-24 21:25:53 +0000
commit232ee49c7b705dce2bfe5cc3a8691916b8a1c51a (patch)
tree1900cadab5ee2133c791163e5df055c8d5df528b /clang
parent2120e2bd732b0e0612e97cf253480807d76174ed (diff)
downloadbcm5719-llvm-232ee49c7b705dce2bfe5cc3a8691916b8a1c51a.tar.gz
bcm5719-llvm-232ee49c7b705dce2bfe5cc3a8691916b8a1c51a.zip
C++11 [class.ctor]p5 says that
A defaulted default constructor for a class X is defined as deleted if [...] - X is a union and all of its variant members are of const-qualified type. A pedantic reading therefore says that union X { }; has a deleted default constructor, which is both silly and almost certainly unintended. Pretend as if this this read - X is a union with one or more variant members, and all of its variant members are of const-qualified type. llvm-svn: 151394
Diffstat (limited to 'clang')
-rw-r--r--clang/lib/Sema/SemaDeclCXX.cpp8
-rw-r--r--clang/test/SemaCXX/cxx0x-deleted-default-ctor.cpp8
2 files changed, 10 insertions, 6 deletions
diff --git a/clang/lib/Sema/SemaDeclCXX.cpp b/clang/lib/Sema/SemaDeclCXX.cpp
index 6512929c2f3..6d2f3b252b1 100644
--- a/clang/lib/Sema/SemaDeclCXX.cpp
+++ b/clang/lib/Sema/SemaDeclCXX.cpp
@@ -4531,7 +4531,8 @@ bool SpecialMemberDeletionInfo::shouldDeleteForField(FieldDecl *FD) {
}
// At least one member in each anonymous union must be non-const
- if (CSM == Sema::CXXDefaultConstructor && AllVariantFieldsAreConst)
+ if (CSM == Sema::CXXDefaultConstructor && AllVariantFieldsAreConst &&
+ FieldRecord->field_begin() != FieldRecord->field_end())
return true;
// Don't try to initialize the anonymous union
@@ -4611,7 +4612,10 @@ bool SpecialMemberDeletionInfo::shouldDeleteForField(FieldDecl *FD) {
/// A defaulted default constructor for a class X is defined as deleted if
/// X is a union and all of its variant members are of const-qualified type.
bool SpecialMemberDeletionInfo::shouldDeleteForAllConstMembers() {
- return CSM == Sema::CXXDefaultConstructor && inUnion() && AllFieldsAreConst;
+ // This is a silly definition, because it gives an empty union a deleted
+ // default constructor. Don't do that.
+ return CSM == Sema::CXXDefaultConstructor && inUnion() && AllFieldsAreConst &&
+ (MD->getParent()->field_begin() != MD->getParent()->field_end());
}
/// Determine whether a defaulted special member function should be defined as
diff --git a/clang/test/SemaCXX/cxx0x-deleted-default-ctor.cpp b/clang/test/SemaCXX/cxx0x-deleted-default-ctor.cpp
index 0f0863515c6..f6e5593d850 100644
--- a/clang/test/SemaCXX/cxx0x-deleted-default-ctor.cpp
+++ b/clang/test/SemaCXX/cxx0x-deleted-default-ctor.cpp
@@ -121,11 +121,11 @@ late_delete::late_delete() = default; // expected-error {{would delete it}}
// See also rdar://problem/8125400.
namespace empty {
- static union {}; // expected-error {{implicitly-deleted default constructor}} expected-note {{here}}
- static union { union {}; }; // expected-error {{implicitly-deleted default constructor}} expected-note {{here}}
+ static union {};
+ static union { union {}; };
static union { struct {}; };
- static union { union { union {}; }; }; // expected-error {{implicitly-deleted default constructor}} expected-note {{here}}
+ static union { union { union {}; }; };
static union { union { struct {}; }; };
- static union { struct { union {}; }; }; // expected-error {{implicitly-deleted default constructor}} expected-note {{here}}
+ static union { struct { union {}; }; };
static union { struct { struct {}; }; };
}
OpenPOWER on IntegriCloud