Skip to content

Commit 8dbc6b2

Browse files
committed
Make diagnostic for use of default member initializer before enclosing class is
complete a little more general; it is produced in other cases than the one that it previously talked about. llvm-svn: 287713
1 parent 1e6ca44 commit 8dbc6b2

File tree

7 files changed

+49
-43
lines changed

7 files changed

+49
-43
lines changed

clang/include/clang/Basic/DiagnosticSemaKinds.td

Lines changed: 5 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -7099,12 +7099,11 @@ def err_in_class_initializer_literal_type : Error<
70997099
"'constexpr' specifier">;
71007100
def err_in_class_initializer_non_constant : Error<
71017101
"in-class initializer for static data member is not a constant expression">;
7102-
def err_in_class_initializer_not_yet_parsed
7103-
: Error<"cannot use defaulted default constructor of %0 within the class "
7104-
"outside of member functions because %1 has an initializer">;
7105-
def err_in_class_initializer_not_yet_parsed_outer_class
7106-
: Error<"cannot use defaulted default constructor of %0 within "
7107-
"%1 outside of member functions because %2 has an initializer">;
7102+
def err_in_class_initializer_not_yet_parsed : Error<
7103+
"default member initializer for %1 needed within definition of enclosing "
7104+
"class %0 outside of member functions">;
7105+
def note_in_class_initializer_not_yet_parsed : Note<
7106+
"default member initializer declared here">;
71087107
def err_in_class_initializer_cycle
71097108
: Error<"default member initializer for %0 uses itself">;
71107109
def err_exception_spec_cycle

clang/lib/Sema/SemaDeclCXX.cpp

Lines changed: 3 additions & 8 deletions
Original file line numberDiff line numberDiff line change
@@ -12378,14 +12378,9 @@ ExprResult Sema::BuildCXXDefaultInitExpr(SourceLocation Loc, FieldDecl *Field) {
1237812378
// constructor before the initializer is lexically complete will ultimately
1237912379
// come here at which point we can diagnose it.
1238012380
RecordDecl *OutermostClass = ParentRD->getOuterLexicalRecordContext();
12381-
if (OutermostClass == ParentRD) {
12382-
Diag(Field->getLocEnd(), diag::err_in_class_initializer_not_yet_parsed)
12383-
<< ParentRD << Field;
12384-
} else {
12385-
Diag(Field->getLocEnd(),
12386-
diag::err_in_class_initializer_not_yet_parsed_outer_class)
12387-
<< ParentRD << OutermostClass << Field;
12388-
}
12381+
Diag(Loc, diag::err_in_class_initializer_not_yet_parsed)
12382+
<< OutermostClass << Field;
12383+
Diag(Field->getLocEnd(), diag::note_in_class_initializer_not_yet_parsed);
1238912384

1239012385
return ExprError();
1239112386
}

clang/lib/Sema/SemaTemplateInstantiate.cpp

Lines changed: 4 additions & 8 deletions
Original file line numberDiff line numberDiff line change
@@ -2198,14 +2198,10 @@ bool Sema::InstantiateInClassInitializer(
21982198
if (!OldInit) {
21992199
RecordDecl *PatternRD = Pattern->getParent();
22002200
RecordDecl *OutermostClass = PatternRD->getOuterLexicalRecordContext();
2201-
if (OutermostClass == PatternRD) {
2202-
Diag(Pattern->getLocEnd(), diag::err_in_class_initializer_not_yet_parsed)
2203-
<< PatternRD << Pattern;
2204-
} else {
2205-
Diag(Pattern->getLocEnd(),
2206-
diag::err_in_class_initializer_not_yet_parsed_outer_class)
2207-
<< PatternRD << OutermostClass << Pattern;
2208-
}
2201+
Diag(PointOfInstantiation,
2202+
diag::err_in_class_initializer_not_yet_parsed)
2203+
<< OutermostClass << Pattern;
2204+
Diag(Pattern->getLocEnd(), diag::note_in_class_initializer_not_yet_parsed);
22092205
Instantiation->setInvalidDecl();
22102206
return true;
22112207
}

clang/test/SemaCXX/constant-expression-cxx11.cpp

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -1862,8 +1862,8 @@ namespace ZeroSizeTypes {
18621862
namespace BadDefaultInit {
18631863
template<int N> struct X { static const int n = N; };
18641864

1865-
struct A {
1866-
int k = // expected-error {{cannot use defaulted default constructor of 'A' within the class outside of member functions because 'k' has an initializer}}
1865+
struct A { // expected-error {{default member initializer for 'k' needed within definition of enclosing class}}
1866+
int k = // expected-note {{default member initializer declared here}}
18671867
X<A().k>::n; // expected-error {{not a constant expression}} expected-note {{implicit default constructor for 'BadDefaultInit::A' first required here}}
18681868
};
18691869

clang/test/SemaCXX/cxx1y-initializer-aggregates.cpp

Lines changed: 16 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -61,3 +61,19 @@ namespace nested_aggregate_init {
6161
};
6262
static_assert(B(6).f() == 18, "");
6363
}
64+
65+
namespace use_self {
66+
struct FibTree {
67+
int n;
68+
FibTree *l = // expected-note {{declared here}}
69+
n > 1 ? new FibTree{n-1} : &fib0; // expected-error {{default member initializer for 'l' needed}}
70+
FibTree *r = // expected-note {{declared here}}
71+
n > 2 ? new FibTree{n-2} : &fib0; // expected-error {{default member initializer for 'r' needed}}
72+
int v = l->v + r->v;
73+
74+
static FibTree fib0;
75+
};
76+
FibTree FibTree::fib0{0, nullptr, nullptr, 1};
77+
78+
int fib(int n) { return FibTree{n}.v; }
79+
}

clang/test/SemaCXX/implicit-exception-spec.cpp

Lines changed: 9 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -16,31 +16,31 @@ namespace InClassInitializers {
1616
// Noexcept::Noexcept is not declared constexpr, therefore noexcept(Noexcept())
1717
// is false.
1818
bool ThrowSomething() noexcept(false);
19-
struct ConstExpr {
20-
bool b = noexcept(ConstExpr()) && ThrowSomething(); // expected-error {{cannot use defaulted default constructor of 'ConstExpr' within the class outside of member functions}}
19+
struct ConstExpr { // expected-error {{default member initializer for 'b' needed}}
20+
bool b = noexcept(ConstExpr()) && ThrowSomething(); // expected-note {{declared here}}
2121
// expected-note@-1 {{implicit default constructor for 'InClassInitializers::ConstExpr' first required here}}
2222
};
2323

2424
// Much more obviously broken: we can't parse the initializer without already
2525
// knowing whether it produces a noexcept expression.
26-
struct TemplateArg {
27-
int n = ExceptionIf<noexcept(TemplateArg())>::f(); // expected-error {{cannot use defaulted default constructor of 'TemplateArg' within the class outside of member functions}}
26+
struct TemplateArg { // expected-error {{default member initializer for 'n' needed}}
27+
int n = ExceptionIf<noexcept(TemplateArg())>::f(); // expected-note {{declared here}}
2828
// expected-note@-1 {{implicit default constructor for 'InClassInitializers::TemplateArg' first required here}}
2929
};
3030

3131
// And within a nested class.
3232
struct Nested { // expected-note {{implicit default constructor for 'InClassInitializers::Nested::Inner' first required here}}
33-
struct Inner {
34-
// expected-error@+1 {{cannot use defaulted default constructor of 'Inner' within 'Nested' outside of member functions}}
35-
int n = ExceptionIf<noexcept(Nested())>::f(); // expected-note {{implicit default constructor for 'InClassInitializers::Nested' first required here}}
33+
struct Inner { // expected-error {{default member initializer for 'n' needed}}
34+
int n = // expected-note {{declared here}}
35+
ExceptionIf<noexcept(Nested())>::f(); // expected-note {{implicit default constructor for 'InClassInitializers::Nested' first required here}}
3636
} inner;
3737
};
3838

3939
struct Nested2 { // expected-error {{implicit default constructor for 'InClassInitializers::Nested2' must explicitly initialize the member 'inner' which does not have a default constructor}}
4040
struct Inner;
4141
int n = Inner().n; // expected-note {{implicit default constructor for 'InClassInitializers::Nested2::Inner' first required here}}
42-
struct Inner { // expected-note {{declared here}}
43-
// expected-error@+1 {{cannot use defaulted default constructor of 'Inner' within 'Nested2' outside of member functions}}
42+
struct Inner { // expected-error {{initializer for 'n' needed}} expected-note {{declared here}}
43+
// expected-note@+1 {{declared here}}
4444
int n = ExceptionIf<noexcept(Nested2())>::f();
4545
// expected-note@-1 {{implicit default constructor for 'InClassInitializers::Nested2' first required here}}
4646
} inner; // expected-note {{member is declared here}}

clang/test/SemaCXX/member-init.cpp

Lines changed: 10 additions & 10 deletions
Original file line numberDiff line numberDiff line change
@@ -13,8 +13,8 @@ class NoWarning {
1313

1414
bool b();
1515
int k;
16-
struct Recurse {
17-
int &n = // expected-error {{cannot use defaulted default constructor of 'Recurse' within the class outside of member functions because 'n' has an initializer}}
16+
struct Recurse { // expected-error {{initializer for 'n' needed}}
17+
int &n = // expected-note {{declared here}}
1818
b() ?
1919
Recurse().n : // expected-note {{implicit default constructor for 'Recurse' first required here}}
2020
k;
@@ -128,8 +128,8 @@ A::A() {}
128128
namespace template_default_ctor {
129129
struct A {
130130
template <typename T>
131-
struct B {
132-
int m1 = 0; // expected-error {{cannot use defaulted default constructor of 'B' within 'A' outside of member functions because 'm1' has an initializer}}
131+
struct B { // expected-error {{initializer for 'm1' needed}}
132+
int m1 = 0; // expected-note {{declared here}}
133133
};
134134
// expected-note@+1 {{implicit default constructor for 'template_default_ctor::A::B<int>' first required here}}
135135
enum { NOE = noexcept(B<int>()) };
@@ -138,8 +138,8 @@ struct A {
138138

139139
namespace default_ctor {
140140
struct A {
141-
struct B {
142-
int m1 = 0; // expected-error {{cannot use defaulted default constructor of 'B' within 'A' outside of member functions because 'm1' has an initializer}}
141+
struct B { // expected-error {{initializer for 'm1' needed}}
142+
int m1 = 0; // expected-note {{declared here}}
143143
};
144144
// expected-note@+1 {{implicit default constructor for 'default_ctor::A::B' first required here}}
145145
enum { NOE = noexcept(B()) };
@@ -150,12 +150,12 @@ namespace member_template {
150150
struct A {
151151
template <typename T>
152152
struct B {
153-
struct C {
154-
int m1 = 0; // expected-error {{cannot use defaulted default constructor of 'C' within 'A' outside of member functions because 'm1' has an initializer}}
153+
struct C { // expected-error {{initializer for 'm1' needed}}
154+
int m1 = 0; // expected-note {{declared here}}
155155
};
156156
template <typename U>
157-
struct D {
158-
int m1 = 0; // expected-error {{cannot use defaulted default constructor of 'D' within 'A' outside of member functions because 'm1' has an initializer}}
157+
struct D { // expected-error {{initializer for 'm1' needed}}
158+
int m1 = 0; // expected-note {{declared here}}
159159
};
160160
};
161161
enum {

0 commit comments

Comments
 (0)