forked from OSchip/llvm-project
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:
parent
2658ea9e90
commit
7607fce112
|
@ -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));
|
||||||
}
|
}
|
||||||
|
|
|
@ -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}}
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
@ -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 {
|
||||||
|
|
|
@ -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
|
Loading…
Reference in New Issue