Don't mark record decls invalid when one of its methods is invalid, PR18284.

Without this patch, record decls with invalid out-of-line method delcs would
sometimes be marked invalid, but not always.  With this patch, they are
consistently never marked invalid.

(The code to do this was added in
http://lists.cs.uiuc.edu/pipermail/cfe-commits/Week-of-Mon-20100809/033154.html
, but the test from that revision is still passing.)

As far as I can tell, this was the only place where a class was marked invalid
after its definition was complete.

llvm-svn: 197848
This commit is contained in:
Nico Weber 2013-12-21 00:49:51 +00:00
parent 2658ea9e90
commit 7607fce112
4 changed files with 29 additions and 11 deletions

View File

@ -7176,12 +7176,7 @@ Sema::ActOnFunctionDeclarator(Scope *S, Declarator &D, DeclContext *DC,
if (!NewFD->isInvalidDecl() && NewFD->isMSVCRTEntryPoint()) if (!NewFD->isInvalidDecl() && NewFD->isMSVCRTEntryPoint())
CheckMSVCRTEntryPoint(NewFD); CheckMSVCRTEntryPoint(NewFD);
if (NewFD->isInvalidDecl()) { if (!NewFD->isInvalidDecl())
// If this is a class member, mark the class invalid immediately.
// This avoids some consistency errors later.
if (CXXMethodDecl* methodDecl = dyn_cast<CXXMethodDecl>(NewFD))
methodDecl->getParent()->setInvalidDecl();
} else
D.setRedeclaration(CheckFunctionDeclaration(S, NewFD, Previous, D.setRedeclaration(CheckFunctionDeclaration(S, NewFD, Previous,
isExplicitSpecialization)); isExplicitSpecialization));
} }

View File

@ -232,15 +232,14 @@ namespace PR7402 {
// <rdar://problem/8308215>: don't crash. // <rdar://problem/8308215>: don't crash.
// Lots of questionable recovery here; errors can change. // Lots of questionable recovery here; errors can change.
namespace test3 { namespace test3 {
class A : public std::exception {}; // expected-error {{undeclared identifier}} expected-error {{expected class name}} expected-note 4 {{candidate}} class A : public std::exception {}; // expected-error {{undeclared identifier}} expected-error {{expected class name}} expected-note 2 {{candidate}}
class B : public A { class B : public A {
public: public:
B(const String& s, int e=0) // expected-error {{unknown type name}} B(const String& s, int e=0) // expected-error {{unknown type name}}
: A(e), m_String(s) , m_ErrorStr(__null) {} // expected-error {{no matching constructor}} expected-error {{does not name}} : A(e), m_String(s) , m_ErrorStr(__null) {} // expected-error {{no matching constructor}} expected-error {{does not name}}
B(const B& e) B(const B& e)
: A(e), m_String(e.m_String), m_ErrorStr(__null) { // expected-error {{does not name}} \ : A(e), m_String(e.m_String), m_ErrorStr(__null) { // expected-error {{does not name}} \
// expected-error {{no member named 'm_String' in 'test3::B'}} \ // expected-error {{no member named 'm_String' in 'test3::B'}}
// expected-error {{no matching}}
} }
}; };
} }

View File

@ -8,12 +8,12 @@ namespace stretch_v1 {
} }
namespace gatekeeper_v1 { namespace gatekeeper_v1 {
namespace gatekeeper_factory_v1 { namespace gatekeeper_factory_v1 {
struct closure_t { // expected-note {{'closure_t' declared here}} struct closure_t { // expected-note {{'closure_t' declared here}} expected-note {{'gatekeeper_factory_v1::closure_t' declared here}}
gatekeeper_v1::closure_t* create(); // expected-error {{no type named 'closure_t' in namespace 'gatekeeper_v1'; did you mean simply 'closure_t'?}} gatekeeper_v1::closure_t* create(); // expected-error {{no type named 'closure_t' in namespace 'gatekeeper_v1'; did you mean simply 'closure_t'?}}
}; };
} }
// FIXME: Typo correction should remove the 'gatekeeper_v1::' name specifier // FIXME: Typo correction should remove the 'gatekeeper_v1::' name specifier
gatekeeper_v1::closure_t *x; // expected-error-re {{no type named 'closure_t' in namespace 'gatekeeper_v1'{{$}}}} gatekeeper_v1::closure_t *x; // expected-error {{no type named 'closure_t' in namespace 'gatekeeper_v1'; did you mean 'gatekeeper_factory_v1::closure_t'}}
} }
namespace Foo { namespace Foo {

View File

@ -0,0 +1,24 @@
// RUN: %clang_cc1 -fsyntax-only -verify %s
// Don't crash (PR18284).
namespace n1 {
class A { };
class C { A a; };
A::RunTest() {} // expected-error {{C++ requires a type specifier for all declarations}}
void f() {
new C;
}
} // namespace n1
namespace n2 {
class A { };
class C : public A { };
A::RunTest() {} // expected-error {{C++ requires a type specifier for all declarations}}
void f() {
new C;
}
} // namespace n2