Skip to content

Commit

Permalink
[ObjC++] Don't enter a C++ declarator scope when the current context is
Browse files Browse the repository at this point in the history
an Objective-C declaration

This commit ensures that Sema won't enter a C++ declarator scope when the
current context is an Objective-C declaration. This prevents an assertion
failure in EnterDeclaratorContext that's used to ensure that current context
will be restored correctly after exiting the declarator context.

rdar://20560175

Differential Revision: https://reviews.llvm.org/D26922

llvm-svn: 288893
  • Loading branch information
hyp committed Dec 7, 2016
1 parent 59336a0 commit e151f01
Show file tree
Hide file tree
Showing 3 changed files with 48 additions and 0 deletions.
8 changes: 8 additions & 0 deletions clang/lib/Parse/ParseDecl.cpp
Expand Up @@ -5264,6 +5264,14 @@ void Parser::ParseDirectDeclarator(Declarator &D) {
// Change the declaration context for name lookup, until this function
// is exited (and the declarator has been parsed).
DeclScopeObj.EnterDeclaratorScope();
else if (getObjCDeclContext()) {
// Ensure that we don't interpret the next token as an identifier when
// dealing with declarations in an Objective-C container.
D.SetIdentifier(nullptr, Tok.getLocation());
D.setInvalidType(true);
ConsumeToken();
goto PastIdentifier;
}
}

// C++0x [dcl.fct]p14:
Expand Down
5 changes: 5 additions & 0 deletions clang/lib/Sema/SemaCXXScopeSpec.cpp
Expand Up @@ -1001,6 +1001,11 @@ void Sema::RestoreNestedNameSpecifierAnnotation(void *AnnotationPtr,
bool Sema::ShouldEnterDeclaratorScope(Scope *S, const CXXScopeSpec &SS) {
assert(SS.isSet() && "Parser passed invalid CXXScopeSpec.");

// Don't enter a declarator context when the current context is an Objective-C
// declaration.
if (isa<ObjCContainerDecl>(CurContext) || isa<ObjCMethodDecl>(CurContext))
return false;

NestedNameSpecifier *Qualifier = SS.getScopeRep();

// There are only two places a well-formed program may qualify a
Expand Down
35 changes: 35 additions & 0 deletions clang/test/SemaObjCXX/crash.mm
Expand Up @@ -25,3 +25,38 @@ @implementation Test
// expected-warning@-2 {{variadic templates are a C++11 extension}}
#endif
@end

// rdar://20560175

struct OuterType {
typedef int InnerType;
};

namespace ns {
typedef int InnerType;
};

@protocol InvalidProperties

@property (nonatomic) (OuterType::InnerType) invalidTypeParens;
// expected-error@-1 {{type name requires a specifier or qualifier}}
// expected-error@-2 {{property requires fields to be named}}
// expected-error@-3 {{expected ';' at end of declaration list}}
// expected-error@-4 {{C++ requires a type specifier for all declarations}}
// expected-error@-5 {{cannot declare variable inside @interface or @protocol}}

@property (nonatomic) (ns::InnerType) invalidTypeParens2;
// expected-error@-1 {{type name requires a specifier or qualifier}}
// expected-error@-2 {{property requires fields to be named}}
// expected-error@-3 {{expected ';' at end of declaration list}}
// expected-error@-4 {{C++ requires a type specifier for all declarations}}
// expected-error@-5 {{cannot declare variable inside @interface or @protocol}}

@property (nonatomic) int OuterType::InnerType; // expected-error {{property requires fields to be named}}

@property (nonatomic) int OuterType::InnerType foo; // expected-error {{property requires fields to be named}}
// expected-error@-1 {{expected ';' at end of declaration list}}
// expected-error@-2 {{C++ requires a type specifier for all declarations}}
// expected-error@-3 {{cannot declare variable inside @interface or @protocol}}

@end

0 comments on commit e151f01

Please sign in to comment.