2008-01-08 03:49:32 +08:00
|
|
|
//===--- ParseObjC.cpp - Objective C Parsing ------------------------------===//
|
2006-11-05 10:08:13 +08:00
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
2007-12-30 03:59:25 +08:00
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
2006-11-05 10:08:13 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file implements the Objective-C portions of the Parser interface.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2009-01-29 13:15:15 +08:00
|
|
|
#include "clang/Parse/ParseDiagnostic.h"
|
2010-08-21 02:27:03 +08:00
|
|
|
#include "clang/Parse/Parser.h"
|
2010-09-15 22:51:05 +08:00
|
|
|
#include "RAIIObjectsForParser.h"
|
2010-08-21 02:27:03 +08:00
|
|
|
#include "clang/Sema/DeclSpec.h"
|
2010-08-27 07:41:50 +08:00
|
|
|
#include "clang/Sema/PrettyDeclStackTrace.h"
|
2010-08-21 02:27:03 +08:00
|
|
|
#include "clang/Sema/Scope.h"
|
2006-11-05 10:08:13 +08:00
|
|
|
#include "llvm/ADT/SmallVector.h"
|
|
|
|
using namespace clang;
|
|
|
|
|
|
|
|
|
2008-12-09 05:53:24 +08:00
|
|
|
/// ParseObjCAtDirectives - Handle parts of the external-declaration production:
|
2006-11-05 10:08:13 +08:00
|
|
|
/// external-declaration: [C99 6.9]
|
|
|
|
/// [OBJC] objc-class-definition
|
2007-10-30 05:39:29 +08:00
|
|
|
/// [OBJC] objc-class-declaration
|
|
|
|
/// [OBJC] objc-alias-declaration
|
|
|
|
/// [OBJC] objc-protocol-definition
|
|
|
|
/// [OBJC] objc-method-definition
|
|
|
|
/// [OBJC] '@' 'end'
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *Parser::ParseObjCAtDirectives() {
|
2006-11-05 10:08:13 +08:00
|
|
|
SourceLocation AtLoc = ConsumeToken(); // the "@"
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-12-07 17:27:33 +08:00
|
|
|
if (Tok.is(tok::code_completion)) {
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.CodeCompleteObjCAtDirective(getCurScope(), ObjCImpDecl, false);
|
2010-05-25 13:58:43 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
2009-12-07 17:27:33 +08:00
|
|
|
}
|
|
|
|
|
2007-08-24 02:16:40 +08:00
|
|
|
switch (Tok.getObjCKeywordID()) {
|
2008-08-23 10:02:23 +08:00
|
|
|
case tok::objc_class:
|
|
|
|
return ParseObjCAtClassDeclaration(AtLoc);
|
2010-12-24 10:08:15 +08:00
|
|
|
case tok::objc_interface: {
|
|
|
|
ParsedAttributes attrs;
|
|
|
|
return ParseObjCAtInterfaceDeclaration(AtLoc, attrs);
|
|
|
|
}
|
|
|
|
case tok::objc_protocol: {
|
|
|
|
ParsedAttributes attrs;
|
|
|
|
return ParseObjCAtProtocolDeclaration(AtLoc, attrs);
|
|
|
|
}
|
2008-08-23 10:02:23 +08:00
|
|
|
case tok::objc_implementation:
|
|
|
|
return ParseObjCAtImplementationDeclaration(AtLoc);
|
|
|
|
case tok::objc_end:
|
|
|
|
return ParseObjCAtEndDeclaration(AtLoc);
|
|
|
|
case tok::objc_compatibility_alias:
|
|
|
|
return ParseObjCAtAliasDeclaration(AtLoc);
|
|
|
|
case tok::objc_synthesize:
|
|
|
|
return ParseObjCPropertySynthesize(AtLoc);
|
|
|
|
case tok::objc_dynamic:
|
|
|
|
return ParseObjCPropertyDynamic(AtLoc);
|
|
|
|
default:
|
|
|
|
Diag(AtLoc, diag::err_unexpected_at);
|
|
|
|
SkipUntil(tok::semi);
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2006-11-05 10:08:13 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
///
|
2009-09-09 23:08:12 +08:00
|
|
|
/// objc-class-declaration:
|
2006-11-05 10:08:13 +08:00
|
|
|
/// '@' 'class' identifier-list ';'
|
2009-09-09 23:08:12 +08:00
|
|
|
///
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *Parser::ParseObjCAtClassDeclaration(SourceLocation atLoc) {
|
2006-11-05 10:08:13 +08:00
|
|
|
ConsumeToken(); // the identifier "class"
|
2007-06-16 07:05:46 +08:00
|
|
|
llvm::SmallVector<IdentifierInfo *, 8> ClassNames;
|
2009-11-18 07:12:20 +08:00
|
|
|
llvm::SmallVector<SourceLocation, 8> ClassLocs;
|
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2006-11-05 10:08:13 +08:00
|
|
|
while (1) {
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::identifier)) {
|
2006-11-10 13:19:25 +08:00
|
|
|
Diag(Tok, diag::err_expected_ident);
|
2006-11-05 10:08:13 +08:00
|
|
|
SkipUntil(tok::semi);
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2006-11-05 10:08:13 +08:00
|
|
|
}
|
|
|
|
ClassNames.push_back(Tok.getIdentifierInfo());
|
2009-11-18 07:12:20 +08:00
|
|
|
ClassLocs.push_back(Tok.getLocation());
|
2006-11-05 10:08:13 +08:00
|
|
|
ConsumeToken();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::comma))
|
2006-11-05 10:08:13 +08:00
|
|
|
break;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2006-11-05 10:08:13 +08:00
|
|
|
ConsumeToken();
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2006-11-05 10:08:13 +08:00
|
|
|
// Consume the ';'.
|
|
|
|
if (ExpectAndConsume(tok::semi, diag::err_expected_semi_after, "@class"))
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-11-18 07:12:20 +08:00
|
|
|
return Actions.ActOnForwardClassDeclaration(atLoc, ClassNames.data(),
|
|
|
|
ClassLocs.data(),
|
|
|
|
ClassNames.size());
|
2006-11-05 10:08:13 +08:00
|
|
|
}
|
|
|
|
|
2007-08-21 05:31:48 +08:00
|
|
|
///
|
|
|
|
/// objc-interface:
|
|
|
|
/// objc-class-interface-attributes[opt] objc-class-interface
|
|
|
|
/// objc-category-interface
|
|
|
|
///
|
|
|
|
/// objc-class-interface:
|
2009-09-09 23:08:12 +08:00
|
|
|
/// '@' 'interface' identifier objc-superclass[opt]
|
2007-08-21 05:31:48 +08:00
|
|
|
/// objc-protocol-refs[opt]
|
2009-09-09 23:08:12 +08:00
|
|
|
/// objc-class-instance-variables[opt]
|
2007-08-21 05:31:48 +08:00
|
|
|
/// objc-interface-decl-list
|
|
|
|
/// @end
|
|
|
|
///
|
|
|
|
/// objc-category-interface:
|
2009-09-09 23:08:12 +08:00
|
|
|
/// '@' 'interface' identifier '(' identifier[opt] ')'
|
2007-08-21 05:31:48 +08:00
|
|
|
/// objc-protocol-refs[opt]
|
|
|
|
/// objc-interface-decl-list
|
|
|
|
/// @end
|
|
|
|
///
|
|
|
|
/// objc-superclass:
|
|
|
|
/// ':' identifier
|
|
|
|
///
|
|
|
|
/// objc-class-interface-attributes:
|
|
|
|
/// __attribute__((visibility("default")))
|
|
|
|
/// __attribute__((visibility("hidden")))
|
|
|
|
/// __attribute__((deprecated))
|
|
|
|
/// __attribute__((unavailable))
|
|
|
|
/// __attribute__((objc_exception)) - used by NSException on 64-bit
|
|
|
|
///
|
2010-12-24 10:08:15 +08:00
|
|
|
Decl *Parser::ParseObjCAtInterfaceDeclaration(SourceLocation atLoc,
|
|
|
|
ParsedAttributes &attrs) {
|
2007-08-24 02:16:40 +08:00
|
|
|
assert(Tok.isObjCAtKeyword(tok::objc_interface) &&
|
2007-08-21 05:31:48 +08:00
|
|
|
"ParseObjCAtInterfaceDeclaration(): Expected @interface");
|
|
|
|
ConsumeToken(); // the "interface" identifier
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-11-19 00:26:39 +08:00
|
|
|
// Code completion after '@interface'.
|
|
|
|
if (Tok.is(tok::code_completion)) {
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.CodeCompleteObjCInterfaceDecl(getCurScope());
|
2010-05-25 13:58:43 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
2009-11-19 00:26:39 +08:00
|
|
|
}
|
|
|
|
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::identifier)) {
|
2007-08-21 05:31:48 +08:00
|
|
|
Diag(Tok, diag::err_expected_ident); // missing class or category name.
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2007-08-21 05:31:48 +08:00
|
|
|
}
|
2009-11-17 02:57:01 +08:00
|
|
|
|
2007-08-21 05:31:48 +08:00
|
|
|
// We have a class or category name - consume it.
|
2007-08-23 06:17:26 +08:00
|
|
|
IdentifierInfo *nameId = Tok.getIdentifierInfo();
|
2007-08-21 05:31:48 +08:00
|
|
|
SourceLocation nameLoc = ConsumeToken();
|
2010-04-27 05:18:08 +08:00
|
|
|
if (Tok.is(tok::l_paren) &&
|
|
|
|
!isKnownToBeTypeSpecifier(GetLookAheadToken(1))) { // we have a category.
|
2011-01-18 10:00:16 +08:00
|
|
|
// TODO(dgregor): Use the return value from the next line to provide better
|
|
|
|
// recovery.
|
|
|
|
ConsumeParen();
|
2007-08-21 05:31:48 +08:00
|
|
|
SourceLocation categoryLoc, rparenLoc;
|
|
|
|
IdentifierInfo *categoryId = 0;
|
2009-11-19 03:08:43 +08:00
|
|
|
if (Tok.is(tok::code_completion)) {
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.CodeCompleteObjCInterfaceCategory(getCurScope(), nameId, nameLoc);
|
2010-05-25 13:58:43 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
2009-11-19 03:08:43 +08:00
|
|
|
}
|
|
|
|
|
2007-08-24 03:56:30 +08:00
|
|
|
// For ObjC2, the category name is optional (not an error).
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.is(tok::identifier)) {
|
2007-08-21 05:31:48 +08:00
|
|
|
categoryId = Tok.getIdentifierInfo();
|
|
|
|
categoryLoc = ConsumeToken();
|
2010-04-03 07:15:40 +08:00
|
|
|
}
|
|
|
|
else if (!getLang().ObjC2) {
|
2007-08-24 03:56:30 +08:00
|
|
|
Diag(Tok, diag::err_expected_ident); // missing category name.
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2007-08-21 05:31:48 +08:00
|
|
|
}
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::r_paren)) {
|
2007-08-21 05:31:48 +08:00
|
|
|
Diag(Tok, diag::err_expected_rparen);
|
|
|
|
SkipUntil(tok::r_paren, false); // don't stop at ';'
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2007-08-21 05:31:48 +08:00
|
|
|
}
|
|
|
|
rparenLoc = ConsumeParen();
|
2010-04-27 05:18:08 +08:00
|
|
|
// Next, we need to check for any protocol references.
|
|
|
|
SourceLocation LAngleLoc, EndProtoLoc;
|
2010-08-21 17:40:31 +08:00
|
|
|
llvm::SmallVector<Decl *, 8> ProtocolRefs;
|
2010-04-27 05:18:08 +08:00
|
|
|
llvm::SmallVector<SourceLocation, 8> ProtocolLocs;
|
|
|
|
if (Tok.is(tok::less) &&
|
|
|
|
ParseObjCProtocolReferences(ProtocolRefs, ProtocolLocs, true,
|
2009-09-30 03:41:44 +08:00
|
|
|
LAngleLoc, EndProtoLoc))
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2010-04-03 07:15:40 +08:00
|
|
|
|
2010-12-24 10:08:15 +08:00
|
|
|
if (!attrs.empty()) // categories don't support attributes.
|
2010-04-27 05:18:08 +08:00
|
|
|
Diag(Tok, diag::err_objc_no_attributes_on_category);
|
|
|
|
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *CategoryType =
|
2010-04-27 05:18:08 +08:00
|
|
|
Actions.ActOnStartCategoryInterface(atLoc,
|
|
|
|
nameId, nameLoc,
|
|
|
|
categoryId, categoryLoc,
|
|
|
|
ProtocolRefs.data(),
|
|
|
|
ProtocolRefs.size(),
|
|
|
|
ProtocolLocs.data(),
|
|
|
|
EndProtoLoc);
|
|
|
|
if (Tok.is(tok::l_brace))
|
2010-02-23 07:04:20 +08:00
|
|
|
ParseObjCClassInstanceVariables(CategoryType, tok::objc_private,
|
|
|
|
atLoc);
|
|
|
|
|
2010-04-27 05:18:08 +08:00
|
|
|
ParseObjCInterfaceDeclList(CategoryType, tok::objc_not_keyword);
|
|
|
|
return CategoryType;
|
2007-08-21 05:31:48 +08:00
|
|
|
}
|
|
|
|
// Parse a class interface.
|
|
|
|
IdentifierInfo *superClassId = 0;
|
|
|
|
SourceLocation superClassLoc;
|
2007-08-23 06:17:26 +08:00
|
|
|
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.is(tok::colon)) { // a super class is specified.
|
2007-08-21 05:31:48 +08:00
|
|
|
ConsumeToken();
|
2009-11-19 00:26:39 +08:00
|
|
|
|
|
|
|
// Code completion of superclass names.
|
|
|
|
if (Tok.is(tok::code_completion)) {
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.CodeCompleteObjCSuperclass(getCurScope(), nameId, nameLoc);
|
2010-05-25 13:58:43 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
2009-11-19 00:26:39 +08:00
|
|
|
}
|
|
|
|
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::identifier)) {
|
2007-08-21 05:31:48 +08:00
|
|
|
Diag(Tok, diag::err_expected_ident); // missing super class name.
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2007-08-21 05:31:48 +08:00
|
|
|
}
|
|
|
|
superClassId = Tok.getIdentifierInfo();
|
|
|
|
superClassLoc = ConsumeToken();
|
|
|
|
}
|
|
|
|
// Next, we need to check for any protocol references.
|
2010-08-21 17:40:31 +08:00
|
|
|
llvm::SmallVector<Decl *, 8> ProtocolRefs;
|
2009-09-30 03:41:44 +08:00
|
|
|
llvm::SmallVector<SourceLocation, 8> ProtocolLocs;
|
|
|
|
SourceLocation LAngleLoc, EndProtoLoc;
|
2008-07-26 12:13:19 +08:00
|
|
|
if (Tok.is(tok::less) &&
|
2009-09-30 03:41:44 +08:00
|
|
|
ParseObjCProtocolReferences(ProtocolRefs, ProtocolLocs, true,
|
|
|
|
LAngleLoc, EndProtoLoc))
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *ClsType =
|
2009-09-09 23:08:12 +08:00
|
|
|
Actions.ActOnStartClassInterface(atLoc, nameId, nameLoc,
|
2008-07-26 12:13:19 +08:00
|
|
|
superClassId, superClassLoc,
|
2009-05-21 17:52:38 +08:00
|
|
|
ProtocolRefs.data(), ProtocolRefs.size(),
|
2010-01-16 23:02:53 +08:00
|
|
|
ProtocolLocs.data(),
|
2010-12-24 10:08:15 +08:00
|
|
|
EndProtoLoc, attrs.getList());
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.is(tok::l_brace))
|
2010-02-23 07:04:20 +08:00
|
|
|
ParseObjCClassInstanceVariables(ClsType, tok::objc_protected, atLoc);
|
2007-08-21 05:31:48 +08:00
|
|
|
|
2007-09-18 05:07:36 +08:00
|
|
|
ParseObjCInterfaceDeclList(ClsType, tok::objc_interface);
|
2010-04-27 05:18:08 +08:00
|
|
|
return ClsType;
|
2007-08-21 05:31:48 +08:00
|
|
|
}
|
|
|
|
|
2009-12-04 06:31:13 +08:00
|
|
|
/// The Objective-C property callback. This should be defined where
|
|
|
|
/// it's used, but instead it's been lifted to here to support VS2005.
|
|
|
|
struct Parser::ObjCPropertyCallback : FieldCallback {
|
|
|
|
Parser &P;
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *IDecl;
|
|
|
|
llvm::SmallVectorImpl<Decl *> &Props;
|
2009-12-04 06:31:13 +08:00
|
|
|
ObjCDeclSpec &OCDS;
|
|
|
|
SourceLocation AtLoc;
|
|
|
|
tok::ObjCKeywordKind MethodImplKind;
|
|
|
|
|
2010-08-21 17:40:31 +08:00
|
|
|
ObjCPropertyCallback(Parser &P, Decl *IDecl,
|
|
|
|
llvm::SmallVectorImpl<Decl *> &Props,
|
2009-12-04 06:31:13 +08:00
|
|
|
ObjCDeclSpec &OCDS, SourceLocation AtLoc,
|
|
|
|
tok::ObjCKeywordKind MethodImplKind) :
|
|
|
|
P(P), IDecl(IDecl), Props(Props), OCDS(OCDS), AtLoc(AtLoc),
|
|
|
|
MethodImplKind(MethodImplKind) {
|
|
|
|
}
|
|
|
|
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *invoke(FieldDeclarator &FD) {
|
2009-12-04 06:31:13 +08:00
|
|
|
if (FD.D.getIdentifier() == 0) {
|
|
|
|
P.Diag(AtLoc, diag::err_objc_property_requires_field_name)
|
|
|
|
<< FD.D.getSourceRange();
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2009-12-04 06:31:13 +08:00
|
|
|
}
|
|
|
|
if (FD.BitfieldSize) {
|
|
|
|
P.Diag(AtLoc, diag::err_objc_property_bitfield)
|
|
|
|
<< FD.D.getSourceRange();
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2009-12-04 06:31:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Install the property declarator into interfaceDecl.
|
|
|
|
IdentifierInfo *SelName =
|
|
|
|
OCDS.getGetterName() ? OCDS.getGetterName() : FD.D.getIdentifier();
|
|
|
|
|
|
|
|
Selector GetterSel =
|
|
|
|
P.PP.getSelectorTable().getNullarySelector(SelName);
|
|
|
|
IdentifierInfo *SetterName = OCDS.getSetterName();
|
|
|
|
Selector SetterSel;
|
|
|
|
if (SetterName)
|
|
|
|
SetterSel = P.PP.getSelectorTable().getSelector(1, &SetterName);
|
|
|
|
else
|
|
|
|
SetterSel = SelectorTable::constructSetterName(P.PP.getIdentifierTable(),
|
|
|
|
P.PP.getSelectorTable(),
|
|
|
|
FD.D.getIdentifier());
|
|
|
|
bool isOverridingProperty = false;
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *Property =
|
2010-07-03 01:43:08 +08:00
|
|
|
P.Actions.ActOnProperty(P.getCurScope(), AtLoc, FD, OCDS,
|
2009-12-04 06:31:13 +08:00
|
|
|
GetterSel, SetterSel, IDecl,
|
|
|
|
&isOverridingProperty,
|
|
|
|
MethodImplKind);
|
|
|
|
if (!isOverridingProperty)
|
|
|
|
Props.push_back(Property);
|
|
|
|
|
|
|
|
return Property;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2007-08-21 05:31:48 +08:00
|
|
|
/// objc-interface-decl-list:
|
|
|
|
/// empty
|
|
|
|
/// objc-interface-decl-list objc-property-decl [OBJC2]
|
2007-08-23 00:35:03 +08:00
|
|
|
/// objc-interface-decl-list objc-method-requirement [OBJC2]
|
2007-09-07 05:24:23 +08:00
|
|
|
/// objc-interface-decl-list objc-method-proto ';'
|
2007-08-21 05:31:48 +08:00
|
|
|
/// objc-interface-decl-list declaration
|
|
|
|
/// objc-interface-decl-list ';'
|
|
|
|
///
|
2007-08-23 00:35:03 +08:00
|
|
|
/// objc-method-requirement: [OBJC2]
|
|
|
|
/// @required
|
|
|
|
/// @optional
|
|
|
|
///
|
2010-08-21 17:40:31 +08:00
|
|
|
void Parser::ParseObjCInterfaceDeclList(Decl *interfaceDecl,
|
2007-12-28 03:57:00 +08:00
|
|
|
tok::ObjCKeywordKind contextKey) {
|
2010-08-21 17:40:31 +08:00
|
|
|
llvm::SmallVector<Decl *, 32> allMethods;
|
|
|
|
llvm::SmallVector<Decl *, 16> allProperties;
|
2009-03-30 00:50:03 +08:00
|
|
|
llvm::SmallVector<DeclGroupPtrTy, 8> allTUVariables;
|
2007-09-18 08:25:23 +08:00
|
|
|
tok::ObjCKeywordKind MethodImplKind = tok::objc_not_keyword;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-01-07 09:20:12 +08:00
|
|
|
SourceRange AtEnd;
|
2008-10-20 14:10:06 +08:00
|
|
|
|
2007-08-23 00:35:03 +08:00
|
|
|
while (1) {
|
2008-10-20 13:46:22 +08:00
|
|
|
// If this is a method prototype, parse it.
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.is(tok::minus) || Tok.is(tok::plus)) {
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *methodPrototype =
|
2007-10-10 01:51:17 +08:00
|
|
|
ParseObjCMethodPrototype(interfaceDecl, MethodImplKind);
|
2007-09-18 05:07:36 +08:00
|
|
|
allMethods.push_back(methodPrototype);
|
2007-09-07 05:24:23 +08:00
|
|
|
// Consume the ';' here, since ParseObjCMethodPrototype() is re-used for
|
|
|
|
// method definitions.
|
2009-02-16 06:24:30 +08:00
|
|
|
ExpectAndConsume(tok::semi, diag::err_expected_semi_after_method_proto,
|
|
|
|
"", tok::semi);
|
2007-08-23 00:35:03 +08:00
|
|
|
continue;
|
|
|
|
}
|
2010-04-03 07:15:40 +08:00
|
|
|
if (Tok.is(tok::l_paren)) {
|
|
|
|
Diag(Tok, diag::err_expected_minus_or_plus);
|
2010-08-21 17:40:31 +08:00
|
|
|
ParseObjCMethodDecl(Tok.getLocation(),
|
|
|
|
tok::minus,
|
|
|
|
interfaceDecl,
|
|
|
|
MethodImplKind);
|
2010-04-03 07:15:40 +08:00
|
|
|
continue;
|
|
|
|
}
|
2008-10-20 13:46:22 +08:00
|
|
|
// Ignore excess semicolons.
|
|
|
|
if (Tok.is(tok::semi)) {
|
2007-08-23 00:35:03 +08:00
|
|
|
ConsumeToken();
|
2008-10-20 13:46:22 +08:00
|
|
|
continue;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-10-20 14:10:06 +08:00
|
|
|
// If we got to the end of the file, exit the loop.
|
2008-10-20 13:46:22 +08:00
|
|
|
if (Tok.is(tok::eof))
|
2007-09-11 04:33:04 +08:00
|
|
|
break;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-01-14 05:24:21 +08:00
|
|
|
// Code completion within an Objective-C interface.
|
|
|
|
if (Tok.is(tok::code_completion)) {
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.CodeCompleteOrdinaryName(getCurScope(),
|
2010-08-27 07:41:50 +08:00
|
|
|
ObjCImpDecl? Sema::PCC_ObjCImplementation
|
|
|
|
: Sema::PCC_ObjCInterface);
|
2010-05-25 13:58:43 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
2010-01-14 05:24:21 +08:00
|
|
|
}
|
|
|
|
|
2008-10-20 13:46:22 +08:00
|
|
|
// If we don't have an @ directive, parse it as a function definition.
|
|
|
|
if (Tok.isNot(tok::at)) {
|
2009-01-09 12:34:13 +08:00
|
|
|
// The code below does not consume '}'s because it is afraid of eating the
|
|
|
|
// end of a namespace. Because of the way this code is structured, an
|
|
|
|
// erroneous r_brace would cause an infinite loop if not handled here.
|
|
|
|
if (Tok.is(tok::r_brace))
|
|
|
|
break;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-08-23 02:35:33 +08:00
|
|
|
// FIXME: as the name implies, this rule allows function definitions.
|
|
|
|
// We could pass a flag or check for functions during semantic analysis.
|
2010-12-24 10:08:15 +08:00
|
|
|
ParsedAttributes attrs;
|
|
|
|
allTUVariables.push_back(ParseDeclarationOrFunctionDefinition(attrs));
|
2008-10-20 13:46:22 +08:00
|
|
|
continue;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-10-20 13:46:22 +08:00
|
|
|
// Otherwise, we have an @ directive, eat the @.
|
|
|
|
SourceLocation AtLoc = ConsumeToken(); // the "@"
|
2009-12-07 17:27:33 +08:00
|
|
|
if (Tok.is(tok::code_completion)) {
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.CodeCompleteObjCAtDirective(getCurScope(), ObjCImpDecl, true);
|
2010-05-25 13:58:43 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
2009-12-07 17:27:33 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2008-10-20 13:57:40 +08:00
|
|
|
tok::ObjCKeywordKind DirectiveKind = Tok.getObjCKeywordID();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-10-20 13:57:40 +08:00
|
|
|
if (DirectiveKind == tok::objc_end) { // @end -> terminate list
|
2010-01-07 09:20:12 +08:00
|
|
|
AtEnd.setBegin(AtLoc);
|
|
|
|
AtEnd.setEnd(Tok.getLocation());
|
2008-10-20 13:46:22 +08:00
|
|
|
break;
|
2010-03-16 14:04:47 +08:00
|
|
|
} else if (DirectiveKind == tok::objc_not_keyword) {
|
|
|
|
Diag(Tok, diag::err_objc_unknown_at);
|
|
|
|
SkipUntil(tok::semi);
|
|
|
|
continue;
|
2008-10-20 14:10:06 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-10-20 14:10:06 +08:00
|
|
|
// Eat the identifier.
|
|
|
|
ConsumeToken();
|
|
|
|
|
2008-10-20 13:57:40 +08:00
|
|
|
switch (DirectiveKind) {
|
|
|
|
default:
|
2008-10-20 14:10:06 +08:00
|
|
|
// FIXME: If someone forgets an @end on a protocol, this loop will
|
|
|
|
// continue to eat up tons of stuff and spew lots of nonsense errors. It
|
|
|
|
// would probably be better to bail out if we saw an @class or @interface
|
|
|
|
// or something like that.
|
2008-10-20 15:22:18 +08:00
|
|
|
Diag(AtLoc, diag::err_objc_illegal_interface_qual);
|
2008-10-20 14:10:06 +08:00
|
|
|
// Skip until we see an '@' or '}' or ';'.
|
2008-10-20 13:57:40 +08:00
|
|
|
SkipUntil(tok::r_brace, tok::at);
|
|
|
|
break;
|
2010-11-02 08:44:43 +08:00
|
|
|
|
|
|
|
case tok::objc_implementation:
|
2010-11-10 04:38:00 +08:00
|
|
|
case tok::objc_interface:
|
2010-11-02 08:44:43 +08:00
|
|
|
Diag(Tok, diag::err_objc_missing_end);
|
|
|
|
ConsumeToken();
|
|
|
|
break;
|
|
|
|
|
2008-10-20 13:57:40 +08:00
|
|
|
case tok::objc_required:
|
|
|
|
case tok::objc_optional:
|
|
|
|
// This is only valid on protocols.
|
2008-10-20 14:10:06 +08:00
|
|
|
// FIXME: Should this check for ObjC2 being enabled?
|
2008-10-20 13:46:22 +08:00
|
|
|
if (contextKey != tok::objc_protocol)
|
2008-10-20 14:10:06 +08:00
|
|
|
Diag(AtLoc, diag::err_objc_directive_only_in_protocol);
|
2008-10-20 13:57:40 +08:00
|
|
|
else
|
2008-10-20 14:10:06 +08:00
|
|
|
MethodImplKind = DirectiveKind;
|
2008-10-20 13:57:40 +08:00
|
|
|
break;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-10-20 13:57:40 +08:00
|
|
|
case tok::objc_property:
|
2008-10-20 15:22:18 +08:00
|
|
|
if (!getLang().ObjC2)
|
2010-12-17 13:40:22 +08:00
|
|
|
Diag(AtLoc, diag::err_objc_properties_require_objc2);
|
2008-10-20 15:22:18 +08:00
|
|
|
|
2008-10-20 13:46:22 +08:00
|
|
|
ObjCDeclSpec OCDS;
|
2009-09-09 23:08:12 +08:00
|
|
|
// Parse property attribute list, if any.
|
2008-10-20 15:24:39 +08:00
|
|
|
if (Tok.is(tok::l_paren))
|
Fix a major inconsistency in the representation of Objective-C
classes, categories, protocols, and class extensions, where the
methods and properties of these entities would be inserted into the
DeclContext in an ordering that doesn't necessarily reflect source
order. The culprits were Sema::ActOnMethodDeclaration(), which did not
perform the insertion of the just-created method declaration into
the DeclContext for these Objective-C entities, and
Sema::ActOnAtEnd(), which inserted all method declarations at the
*end* of the DeclContext.
With this fix in hand, clean up the code-completion actions for
property setters/getters that worked around this brokenness in the AST.
Fixes <rdar://problem/8062781>, where this problem manifested as poor
token-annotation information, but this would have struck again in many
other places.
llvm-svn: 122347
2010-12-22 01:34:17 +08:00
|
|
|
ParseObjCPropertyAttribute(OCDS, interfaceDecl);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-12-04 06:31:13 +08:00
|
|
|
ObjCPropertyCallback Callback(*this, interfaceDecl, allProperties,
|
|
|
|
OCDS, AtLoc, MethodImplKind);
|
2009-11-03 10:38:08 +08:00
|
|
|
|
2008-10-20 13:46:22 +08:00
|
|
|
// Parse all the comma separated declarators.
|
|
|
|
DeclSpec DS;
|
2009-11-03 10:38:08 +08:00
|
|
|
ParseStructDeclaration(DS, Callback);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-10-20 14:15:13 +08:00
|
|
|
ExpectAndConsume(tok::semi, diag::err_expected_semi_decl_list, "",
|
|
|
|
tok::at);
|
2008-10-20 13:57:40 +08:00
|
|
|
break;
|
2007-09-06 07:30:30 +08:00
|
|
|
}
|
2007-08-23 00:35:03 +08:00
|
|
|
}
|
2008-10-20 14:10:06 +08:00
|
|
|
|
|
|
|
// We break out of the big loop in two cases: when we see @end or when we see
|
|
|
|
// EOF. In the former case, eat the @end. In the later case, emit an error.
|
2009-12-07 17:27:33 +08:00
|
|
|
if (Tok.is(tok::code_completion)) {
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.CodeCompleteObjCAtDirective(getCurScope(), ObjCImpDecl, true);
|
2010-05-25 13:58:43 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
2009-12-07 17:27:33 +08:00
|
|
|
} else if (Tok.isObjCAtKeyword(tok::objc_end))
|
2008-10-20 14:10:06 +08:00
|
|
|
ConsumeToken(); // the "end" identifier
|
|
|
|
else
|
|
|
|
Diag(Tok, diag::err_objc_missing_end);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-10-20 13:57:40 +08:00
|
|
|
// Insert collected methods declarations into the @interface object.
|
2008-10-20 14:10:06 +08:00
|
|
|
// This passes in an invalid SourceLocation for AtEndLoc when EOF is hit.
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.ActOnAtEnd(getCurScope(), AtEnd, interfaceDecl,
|
2009-09-09 23:08:12 +08:00
|
|
|
allMethods.data(), allMethods.size(),
|
2009-05-21 17:52:38 +08:00
|
|
|
allProperties.data(), allProperties.size(),
|
|
|
|
allTUVariables.data(), allTUVariables.size());
|
2007-08-23 00:35:03 +08:00
|
|
|
}
|
|
|
|
|
2007-09-01 00:11:31 +08:00
|
|
|
/// Parse property attribute declarations.
|
|
|
|
///
|
|
|
|
/// property-attr-decl: '(' property-attrlist ')'
|
|
|
|
/// property-attrlist:
|
|
|
|
/// property-attribute
|
|
|
|
/// property-attrlist ',' property-attribute
|
|
|
|
/// property-attribute:
|
|
|
|
/// getter '=' identifier
|
|
|
|
/// setter '=' identifier ':'
|
|
|
|
/// readonly
|
|
|
|
/// readwrite
|
|
|
|
/// assign
|
|
|
|
/// retain
|
|
|
|
/// copy
|
|
|
|
/// nonatomic
|
|
|
|
///
|
Fix a major inconsistency in the representation of Objective-C
classes, categories, protocols, and class extensions, where the
methods and properties of these entities would be inserted into the
DeclContext in an ordering that doesn't necessarily reflect source
order. The culprits were Sema::ActOnMethodDeclaration(), which did not
perform the insertion of the just-created method declaration into
the DeclContext for these Objective-C entities, and
Sema::ActOnAtEnd(), which inserted all method declarations at the
*end* of the DeclContext.
With this fix in hand, clean up the code-completion actions for
property setters/getters that worked around this brokenness in the AST.
Fixes <rdar://problem/8062781>, where this problem manifested as poor
token-annotation information, but this would have struck again in many
other places.
llvm-svn: 122347
2010-12-22 01:34:17 +08:00
|
|
|
void Parser::ParseObjCPropertyAttribute(ObjCDeclSpec &DS, Decl *ClassDecl) {
|
2008-10-20 15:24:39 +08:00
|
|
|
assert(Tok.getKind() == tok::l_paren);
|
2008-10-20 15:00:43 +08:00
|
|
|
SourceLocation LHSLoc = ConsumeParen(); // consume '('
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-10-20 15:15:22 +08:00
|
|
|
while (1) {
|
2009-10-09 05:55:05 +08:00
|
|
|
if (Tok.is(tok::code_completion)) {
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.CodeCompleteObjCPropertyFlags(getCurScope(), DS);
|
2010-05-25 13:58:43 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
2009-10-09 05:55:05 +08:00
|
|
|
}
|
2007-09-01 00:11:31 +08:00
|
|
|
const IdentifierInfo *II = Tok.getIdentifierInfo();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-10-20 15:22:18 +08:00
|
|
|
// If this is not an identifier at all, bail out early.
|
|
|
|
if (II == 0) {
|
|
|
|
MatchRHSPunctuation(tok::r_paren, LHSLoc);
|
|
|
|
return;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-10-20 15:37:22 +08:00
|
|
|
SourceLocation AttrName = ConsumeToken(); // consume last attribute name
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-11-20 12:42:34 +08:00
|
|
|
if (II->isStr("readonly"))
|
2008-10-20 15:39:53 +08:00
|
|
|
DS.setPropertyAttributes(ObjCDeclSpec::DQ_PR_readonly);
|
2008-11-20 12:42:34 +08:00
|
|
|
else if (II->isStr("assign"))
|
2008-10-20 15:39:53 +08:00
|
|
|
DS.setPropertyAttributes(ObjCDeclSpec::DQ_PR_assign);
|
2008-11-20 12:42:34 +08:00
|
|
|
else if (II->isStr("readwrite"))
|
2008-10-20 15:39:53 +08:00
|
|
|
DS.setPropertyAttributes(ObjCDeclSpec::DQ_PR_readwrite);
|
2008-11-20 12:42:34 +08:00
|
|
|
else if (II->isStr("retain"))
|
2008-10-20 15:39:53 +08:00
|
|
|
DS.setPropertyAttributes(ObjCDeclSpec::DQ_PR_retain);
|
2008-11-20 12:42:34 +08:00
|
|
|
else if (II->isStr("copy"))
|
2008-10-20 15:39:53 +08:00
|
|
|
DS.setPropertyAttributes(ObjCDeclSpec::DQ_PR_copy);
|
2008-11-20 12:42:34 +08:00
|
|
|
else if (II->isStr("nonatomic"))
|
2008-10-20 15:39:53 +08:00
|
|
|
DS.setPropertyAttributes(ObjCDeclSpec::DQ_PR_nonatomic);
|
2010-12-18 06:28:16 +08:00
|
|
|
else if (II->isStr("atomic"))
|
|
|
|
DS.setPropertyAttributes(ObjCDeclSpec::DQ_PR_atomic);
|
2008-11-20 12:42:34 +08:00
|
|
|
else if (II->isStr("getter") || II->isStr("setter")) {
|
2010-10-03 01:45:21 +08:00
|
|
|
bool IsSetter = II->getNameStart()[0] == 's';
|
|
|
|
|
2008-10-20 15:39:53 +08:00
|
|
|
// getter/setter require extra treatment.
|
2010-10-03 01:45:21 +08:00
|
|
|
unsigned DiagID = IsSetter ? diag::err_objc_expected_equal_for_setter :
|
|
|
|
diag::err_objc_expected_equal_for_getter;
|
|
|
|
|
|
|
|
if (ExpectAndConsume(tok::equal, DiagID, "", tok::r_paren))
|
2008-10-20 15:00:43 +08:00
|
|
|
return;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-11-19 15:41:15 +08:00
|
|
|
if (Tok.is(tok::code_completion)) {
|
2010-10-03 01:45:21 +08:00
|
|
|
if (IsSetter)
|
Fix a major inconsistency in the representation of Objective-C
classes, categories, protocols, and class extensions, where the
methods and properties of these entities would be inserted into the
DeclContext in an ordering that doesn't necessarily reflect source
order. The culprits were Sema::ActOnMethodDeclaration(), which did not
perform the insertion of the just-created method declaration into
the DeclContext for these Objective-C entities, and
Sema::ActOnAtEnd(), which inserted all method declarations at the
*end* of the DeclContext.
With this fix in hand, clean up the code-completion actions for
property setters/getters that worked around this brokenness in the AST.
Fixes <rdar://problem/8062781>, where this problem manifested as poor
token-annotation information, but this would have struck again in many
other places.
llvm-svn: 122347
2010-12-22 01:34:17 +08:00
|
|
|
Actions.CodeCompleteObjCPropertySetter(getCurScope(), ClassDecl);
|
2009-11-19 15:41:15 +08:00
|
|
|
else
|
Fix a major inconsistency in the representation of Objective-C
classes, categories, protocols, and class extensions, where the
methods and properties of these entities would be inserted into the
DeclContext in an ordering that doesn't necessarily reflect source
order. The culprits were Sema::ActOnMethodDeclaration(), which did not
perform the insertion of the just-created method declaration into
the DeclContext for these Objective-C entities, and
Sema::ActOnAtEnd(), which inserted all method declarations at the
*end* of the DeclContext.
With this fix in hand, clean up the code-completion actions for
property setters/getters that worked around this brokenness in the AST.
Fixes <rdar://problem/8062781>, where this problem manifested as poor
token-annotation information, but this would have struck again in many
other places.
llvm-svn: 122347
2010-12-22 01:34:17 +08:00
|
|
|
Actions.CodeCompleteObjCPropertyGetter(getCurScope(), ClassDecl);
|
2010-05-25 13:58:43 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
2009-11-19 15:41:15 +08:00
|
|
|
}
|
|
|
|
|
2010-10-03 01:45:21 +08:00
|
|
|
|
|
|
|
SourceLocation SelLoc;
|
|
|
|
IdentifierInfo *SelIdent = ParseObjCSelectorPiece(SelLoc);
|
|
|
|
|
|
|
|
if (!SelIdent) {
|
|
|
|
Diag(Tok, diag::err_objc_expected_selector_for_getter_setter)
|
|
|
|
<< IsSetter;
|
2008-10-20 15:24:39 +08:00
|
|
|
SkipUntil(tok::r_paren);
|
|
|
|
return;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-10-03 01:45:21 +08:00
|
|
|
if (IsSetter) {
|
2008-10-20 15:24:39 +08:00
|
|
|
DS.setPropertyAttributes(ObjCDeclSpec::DQ_PR_setter);
|
2010-10-03 01:45:21 +08:00
|
|
|
DS.setSetterName(SelIdent);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-02-16 06:20:11 +08:00
|
|
|
if (ExpectAndConsume(tok::colon,
|
|
|
|
diag::err_expected_colon_after_setter_name, "",
|
2008-10-20 15:37:22 +08:00
|
|
|
tok::r_paren))
|
2008-10-20 15:24:39 +08:00
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
DS.setPropertyAttributes(ObjCDeclSpec::DQ_PR_getter);
|
2010-10-03 01:45:21 +08:00
|
|
|
DS.setGetterName(SelIdent);
|
2008-10-20 15:24:39 +08:00
|
|
|
}
|
2008-10-20 15:39:53 +08:00
|
|
|
} else {
|
2008-11-19 15:49:38 +08:00
|
|
|
Diag(AttrName, diag::err_objc_expected_property_attr) << II;
|
2008-10-20 15:15:22 +08:00
|
|
|
SkipUntil(tok::r_paren);
|
|
|
|
return;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-10-20 15:37:22 +08:00
|
|
|
if (Tok.isNot(tok::comma))
|
|
|
|
break;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-10-20 15:37:22 +08:00
|
|
|
ConsumeToken();
|
2007-09-01 00:11:31 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-10-20 15:37:22 +08:00
|
|
|
MatchRHSPunctuation(tok::r_paren, LHSLoc);
|
2007-09-01 00:11:31 +08:00
|
|
|
}
|
|
|
|
|
2007-09-07 05:24:23 +08:00
|
|
|
/// objc-method-proto:
|
2009-09-09 23:08:12 +08:00
|
|
|
/// objc-instance-method objc-method-decl objc-method-attributes[opt]
|
2007-09-07 05:24:23 +08:00
|
|
|
/// objc-class-method objc-method-decl objc-method-attributes[opt]
|
2007-08-23 00:35:03 +08:00
|
|
|
///
|
|
|
|
/// objc-instance-method: '-'
|
|
|
|
/// objc-class-method: '+'
|
|
|
|
///
|
2007-08-23 02:35:33 +08:00
|
|
|
/// objc-method-attributes: [OBJC2]
|
|
|
|
/// __attribute__((deprecated))
|
|
|
|
///
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *Parser::ParseObjCMethodPrototype(Decl *IDecl,
|
|
|
|
tok::ObjCKeywordKind MethodImplKind) {
|
2007-10-10 01:51:17 +08:00
|
|
|
assert((Tok.is(tok::minus) || Tok.is(tok::plus)) && "expected +/-");
|
2007-08-23 00:35:03 +08:00
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
tok::TokenKind methodType = Tok.getKind();
|
2007-10-27 04:53:56 +08:00
|
|
|
SourceLocation mLoc = ConsumeToken();
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *MDecl = ParseObjCMethodDecl(mLoc, methodType, IDecl,MethodImplKind);
|
2007-09-07 05:24:23 +08:00
|
|
|
// Since this rule is used for both method declarations and definitions,
|
2007-09-11 04:51:04 +08:00
|
|
|
// the caller is (optionally) responsible for consuming the ';'.
|
2007-09-06 07:30:30 +08:00
|
|
|
return MDecl;
|
2007-08-23 00:35:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// objc-selector:
|
|
|
|
/// identifier
|
|
|
|
/// one of
|
|
|
|
/// enum struct union if else while do for switch case default
|
|
|
|
/// break continue return goto asm sizeof typeof __alignof
|
|
|
|
/// unsigned long const short volatile signed restrict _Complex
|
|
|
|
/// in out inout bycopy byref oneway int char float double void _Bool
|
|
|
|
///
|
2009-04-12 02:13:45 +08:00
|
|
|
IdentifierInfo *Parser::ParseObjCSelectorPiece(SourceLocation &SelectorLoc) {
|
2010-09-03 09:26:16 +08:00
|
|
|
|
2007-10-07 10:00:24 +08:00
|
|
|
switch (Tok.getKind()) {
|
|
|
|
default:
|
|
|
|
return 0;
|
2010-09-04 01:33:04 +08:00
|
|
|
case tok::ampamp:
|
|
|
|
case tok::ampequal:
|
|
|
|
case tok::amp:
|
|
|
|
case tok::pipe:
|
|
|
|
case tok::tilde:
|
|
|
|
case tok::exclaim:
|
|
|
|
case tok::exclaimequal:
|
|
|
|
case tok::pipepipe:
|
|
|
|
case tok::pipeequal:
|
|
|
|
case tok::caret:
|
|
|
|
case tok::caretequal: {
|
2010-09-04 02:01:09 +08:00
|
|
|
std::string ThisTok(PP.getSpelling(Tok));
|
2010-09-04 01:33:04 +08:00
|
|
|
if (isalpha(ThisTok[0])) {
|
|
|
|
IdentifierInfo *II = &PP.getIdentifierTable().get(ThisTok.data());
|
|
|
|
Tok.setKind(tok::identifier);
|
|
|
|
SelectorLoc = ConsumeToken();
|
|
|
|
return II;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-10-07 10:00:24 +08:00
|
|
|
case tok::identifier:
|
2008-08-24 05:00:01 +08:00
|
|
|
case tok::kw_asm:
|
2007-10-07 10:00:24 +08:00
|
|
|
case tok::kw_auto:
|
2007-11-15 13:25:19 +08:00
|
|
|
case tok::kw_bool:
|
2008-08-24 05:00:01 +08:00
|
|
|
case tok::kw_break:
|
|
|
|
case tok::kw_case:
|
|
|
|
case tok::kw_catch:
|
|
|
|
case tok::kw_char:
|
|
|
|
case tok::kw_class:
|
|
|
|
case tok::kw_const:
|
|
|
|
case tok::kw_const_cast:
|
|
|
|
case tok::kw_continue:
|
|
|
|
case tok::kw_default:
|
|
|
|
case tok::kw_delete:
|
|
|
|
case tok::kw_do:
|
|
|
|
case tok::kw_double:
|
|
|
|
case tok::kw_dynamic_cast:
|
|
|
|
case tok::kw_else:
|
|
|
|
case tok::kw_enum:
|
|
|
|
case tok::kw_explicit:
|
|
|
|
case tok::kw_export:
|
|
|
|
case tok::kw_extern:
|
|
|
|
case tok::kw_false:
|
|
|
|
case tok::kw_float:
|
|
|
|
case tok::kw_for:
|
|
|
|
case tok::kw_friend:
|
|
|
|
case tok::kw_goto:
|
|
|
|
case tok::kw_if:
|
|
|
|
case tok::kw_inline:
|
|
|
|
case tok::kw_int:
|
|
|
|
case tok::kw_long:
|
|
|
|
case tok::kw_mutable:
|
|
|
|
case tok::kw_namespace:
|
|
|
|
case tok::kw_new:
|
|
|
|
case tok::kw_operator:
|
|
|
|
case tok::kw_private:
|
|
|
|
case tok::kw_protected:
|
|
|
|
case tok::kw_public:
|
|
|
|
case tok::kw_register:
|
|
|
|
case tok::kw_reinterpret_cast:
|
|
|
|
case tok::kw_restrict:
|
|
|
|
case tok::kw_return:
|
|
|
|
case tok::kw_short:
|
|
|
|
case tok::kw_signed:
|
|
|
|
case tok::kw_sizeof:
|
|
|
|
case tok::kw_static:
|
|
|
|
case tok::kw_static_cast:
|
|
|
|
case tok::kw_struct:
|
|
|
|
case tok::kw_switch:
|
|
|
|
case tok::kw_template:
|
|
|
|
case tok::kw_this:
|
|
|
|
case tok::kw_throw:
|
|
|
|
case tok::kw_true:
|
|
|
|
case tok::kw_try:
|
|
|
|
case tok::kw_typedef:
|
|
|
|
case tok::kw_typeid:
|
|
|
|
case tok::kw_typename:
|
|
|
|
case tok::kw_typeof:
|
|
|
|
case tok::kw_union:
|
|
|
|
case tok::kw_unsigned:
|
|
|
|
case tok::kw_using:
|
|
|
|
case tok::kw_virtual:
|
|
|
|
case tok::kw_void:
|
|
|
|
case tok::kw_volatile:
|
|
|
|
case tok::kw_wchar_t:
|
|
|
|
case tok::kw_while:
|
2007-10-07 10:00:24 +08:00
|
|
|
case tok::kw__Bool:
|
|
|
|
case tok::kw__Complex:
|
2008-08-24 05:00:01 +08:00
|
|
|
case tok::kw___alignof:
|
2007-10-07 10:00:24 +08:00
|
|
|
IdentifierInfo *II = Tok.getIdentifierInfo();
|
2007-10-11 08:55:41 +08:00
|
|
|
SelectorLoc = ConsumeToken();
|
2007-10-07 10:00:24 +08:00
|
|
|
return II;
|
2007-09-28 03:52:15 +08:00
|
|
|
}
|
2007-08-23 00:35:03 +08:00
|
|
|
}
|
|
|
|
|
2008-01-03 06:54:34 +08:00
|
|
|
/// objc-for-collection-in: 'in'
|
|
|
|
///
|
2008-01-05 07:04:08 +08:00
|
|
|
bool Parser::isTokIdentifier_in() const {
|
2008-01-04 01:55:25 +08:00
|
|
|
// FIXME: May have to do additional look-ahead to only allow for
|
|
|
|
// valid tokens following an 'in'; such as an identifier, unary operators,
|
|
|
|
// '[' etc.
|
2009-09-09 23:08:12 +08:00
|
|
|
return (getLang().ObjC2 && Tok.is(tok::identifier) &&
|
2008-08-23 10:02:23 +08:00
|
|
|
Tok.getIdentifierInfo() == ObjCTypeQuals[objc_in]);
|
2008-01-03 06:54:34 +08:00
|
|
|
}
|
|
|
|
|
2008-01-08 03:49:32 +08:00
|
|
|
/// ParseObjCTypeQualifierList - This routine parses the objective-c's type
|
2007-12-12 14:56:32 +08:00
|
|
|
/// qualifier list and builds their bitmask representation in the input
|
|
|
|
/// argument.
|
2007-08-23 00:35:03 +08:00
|
|
|
///
|
|
|
|
/// objc-type-qualifiers:
|
|
|
|
/// objc-type-qualifier
|
|
|
|
/// objc-type-qualifiers objc-type-qualifier
|
|
|
|
///
|
2010-08-24 09:06:58 +08:00
|
|
|
void Parser::ParseObjCTypeQualifierList(ObjCDeclSpec &DS, bool IsParameter) {
|
2007-12-12 14:56:32 +08:00
|
|
|
while (1) {
|
2010-08-24 09:06:58 +08:00
|
|
|
if (Tok.is(tok::code_completion)) {
|
2011-02-16 06:19:42 +08:00
|
|
|
Actions.CodeCompleteObjCPassingType(getCurScope(), DS, IsParameter);
|
2010-08-24 09:06:58 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
|
|
|
}
|
|
|
|
|
2007-12-28 03:57:00 +08:00
|
|
|
if (Tok.isNot(tok::identifier))
|
2007-12-12 14:56:32 +08:00
|
|
|
return;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-12-12 14:56:32 +08:00
|
|
|
const IdentifierInfo *II = Tok.getIdentifierInfo();
|
|
|
|
for (unsigned i = 0; i != objc_NumQuals; ++i) {
|
2008-01-08 03:49:32 +08:00
|
|
|
if (II != ObjCTypeQuals[i])
|
2007-12-12 14:56:32 +08:00
|
|
|
continue;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-01-08 03:49:32 +08:00
|
|
|
ObjCDeclSpec::ObjCDeclQualifier Qual;
|
2007-12-12 14:56:32 +08:00
|
|
|
switch (i) {
|
|
|
|
default: assert(0 && "Unknown decl qualifier");
|
2008-01-08 03:49:32 +08:00
|
|
|
case objc_in: Qual = ObjCDeclSpec::DQ_In; break;
|
|
|
|
case objc_out: Qual = ObjCDeclSpec::DQ_Out; break;
|
|
|
|
case objc_inout: Qual = ObjCDeclSpec::DQ_Inout; break;
|
|
|
|
case objc_oneway: Qual = ObjCDeclSpec::DQ_Oneway; break;
|
|
|
|
case objc_bycopy: Qual = ObjCDeclSpec::DQ_Bycopy; break;
|
|
|
|
case objc_byref: Qual = ObjCDeclSpec::DQ_Byref; break;
|
2007-12-12 14:56:32 +08:00
|
|
|
}
|
2008-01-08 03:49:32 +08:00
|
|
|
DS.setObjCDeclQualifier(Qual);
|
2007-12-12 14:56:32 +08:00
|
|
|
ConsumeToken();
|
|
|
|
II = 0;
|
|
|
|
break;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-12-12 14:56:32 +08:00
|
|
|
// If this wasn't a recognized qualifier, bail out.
|
|
|
|
if (II) return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// objc-type-name:
|
|
|
|
/// '(' objc-type-qualifiers[opt] type-name ')'
|
|
|
|
/// '(' objc-type-qualifiers[opt] ')'
|
|
|
|
///
|
2010-08-24 13:47:05 +08:00
|
|
|
ParsedType Parser::ParseObjCTypeName(ObjCDeclSpec &DS, bool IsParameter) {
|
2007-10-10 01:51:17 +08:00
|
|
|
assert(Tok.is(tok::l_paren) && "expected (");
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-10-22 11:52:06 +08:00
|
|
|
SourceLocation LParenLoc = ConsumeParen();
|
2008-08-23 09:48:03 +08:00
|
|
|
SourceLocation TypeStartLoc = Tok.getLocation();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-11-01 05:59:43 +08:00
|
|
|
// Parse type qualifiers, in, inout, etc.
|
2010-08-24 09:06:58 +08:00
|
|
|
ParseObjCTypeQualifierList(DS, IsParameter);
|
2007-08-23 07:18:22 +08:00
|
|
|
|
2010-08-24 13:47:05 +08:00
|
|
|
ParsedType Ty;
|
2009-02-19 01:45:20 +08:00
|
|
|
if (isTypeSpecifierQualifier()) {
|
|
|
|
TypeResult TypeSpec = ParseTypeName();
|
|
|
|
if (!TypeSpec.isInvalid())
|
|
|
|
Ty = TypeSpec.get();
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-10-22 11:52:06 +08:00
|
|
|
if (Tok.is(tok::r_paren))
|
|
|
|
ConsumeParen();
|
|
|
|
else if (Tok.getLocation() == TypeStartLoc) {
|
2008-08-23 09:48:03 +08:00
|
|
|
// If we didn't eat any tokens, then this isn't a type.
|
2008-11-18 15:48:38 +08:00
|
|
|
Diag(Tok, diag::err_expected_type);
|
2008-10-22 11:52:06 +08:00
|
|
|
SkipUntil(tok::r_paren);
|
|
|
|
} else {
|
|
|
|
// Otherwise, we found *something*, but didn't get a ')' in the right
|
|
|
|
// place. Emit an error then return what we have as the type.
|
|
|
|
MatchRHSPunctuation(tok::r_paren, LParenLoc);
|
2007-08-23 00:35:03 +08:00
|
|
|
}
|
2007-09-06 07:30:30 +08:00
|
|
|
return Ty;
|
2007-08-23 00:35:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// objc-method-decl:
|
|
|
|
/// objc-selector
|
2007-08-23 02:35:33 +08:00
|
|
|
/// objc-keyword-selector objc-parmlist[opt]
|
2007-08-23 00:35:03 +08:00
|
|
|
/// objc-type-name objc-selector
|
2007-08-23 02:35:33 +08:00
|
|
|
/// objc-type-name objc-keyword-selector objc-parmlist[opt]
|
2007-08-23 00:35:03 +08:00
|
|
|
///
|
|
|
|
/// objc-keyword-selector:
|
2009-09-09 23:08:12 +08:00
|
|
|
/// objc-keyword-decl
|
2007-08-23 00:35:03 +08:00
|
|
|
/// objc-keyword-selector objc-keyword-decl
|
|
|
|
///
|
|
|
|
/// objc-keyword-decl:
|
2007-08-23 06:17:26 +08:00
|
|
|
/// objc-selector ':' objc-type-name objc-keyword-attributes[opt] identifier
|
|
|
|
/// objc-selector ':' objc-keyword-attributes[opt] identifier
|
|
|
|
/// ':' objc-type-name objc-keyword-attributes[opt] identifier
|
|
|
|
/// ':' objc-keyword-attributes[opt] identifier
|
2007-08-23 00:35:03 +08:00
|
|
|
///
|
2007-08-23 02:35:33 +08:00
|
|
|
/// objc-parmlist:
|
|
|
|
/// objc-parms objc-ellipsis[opt]
|
2007-08-23 00:35:03 +08:00
|
|
|
///
|
2007-08-23 02:35:33 +08:00
|
|
|
/// objc-parms:
|
|
|
|
/// objc-parms , parameter-declaration
|
2007-08-23 00:35:03 +08:00
|
|
|
///
|
2007-08-23 02:35:33 +08:00
|
|
|
/// objc-ellipsis:
|
2007-08-23 00:35:03 +08:00
|
|
|
/// , ...
|
|
|
|
///
|
2007-08-23 06:17:26 +08:00
|
|
|
/// objc-keyword-attributes: [OBJC2]
|
|
|
|
/// __attribute__((unused))
|
|
|
|
///
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *Parser::ParseObjCMethodDecl(SourceLocation mLoc,
|
2010-08-24 02:23:48 +08:00
|
|
|
tok::TokenKind mType,
|
|
|
|
Decl *IDecl,
|
|
|
|
tok::ObjCKeywordKind MethodImplKind) {
|
2009-11-04 10:18:39 +08:00
|
|
|
ParsingDeclRAIIObject PD(*this);
|
|
|
|
|
2010-04-07 08:21:17 +08:00
|
|
|
if (Tok.is(tok::code_completion)) {
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.CodeCompleteObjCMethodDecl(getCurScope(), mType == tok::minus,
|
2010-08-24 13:47:05 +08:00
|
|
|
/*ReturnType=*/ ParsedType(), IDecl);
|
2010-05-25 13:58:43 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
2010-04-07 08:21:17 +08:00
|
|
|
}
|
|
|
|
|
2008-08-23 09:48:03 +08:00
|
|
|
// Parse the return type if present.
|
2010-08-24 13:47:05 +08:00
|
|
|
ParsedType ReturnType;
|
2008-01-08 03:49:32 +08:00
|
|
|
ObjCDeclSpec DSRet;
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.is(tok::l_paren))
|
2010-08-24 09:06:58 +08:00
|
|
|
ReturnType = ParseObjCTypeName(DSRet, false);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-02-19 07:05:16 +08:00
|
|
|
// If attributes exist before the method, parse them.
|
2010-12-24 10:08:15 +08:00
|
|
|
ParsedAttributes attrs;
|
|
|
|
if (getLang().ObjC2)
|
|
|
|
MaybeParseGNUAttributes(attrs);
|
2010-02-19 07:05:16 +08:00
|
|
|
|
2010-04-07 08:21:17 +08:00
|
|
|
if (Tok.is(tok::code_completion)) {
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.CodeCompleteObjCMethodDecl(getCurScope(), mType == tok::minus,
|
2010-04-07 08:21:17 +08:00
|
|
|
ReturnType, IDecl);
|
2010-05-25 13:58:43 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
2010-04-07 08:21:17 +08:00
|
|
|
}
|
|
|
|
|
2010-02-19 07:05:16 +08:00
|
|
|
// Now parse the selector.
|
2007-10-27 04:53:56 +08:00
|
|
|
SourceLocation selLoc;
|
2009-04-12 02:13:45 +08:00
|
|
|
IdentifierInfo *SelIdent = ParseObjCSelectorPiece(selLoc);
|
2008-08-23 09:48:03 +08:00
|
|
|
|
2009-02-12 04:43:13 +08:00
|
|
|
// An unnamed colon is valid.
|
|
|
|
if (!SelIdent && Tok.isNot(tok::colon)) { // missing selector name.
|
2008-11-18 15:48:38 +08:00
|
|
|
Diag(Tok, diag::err_expected_selector_for_method)
|
|
|
|
<< SourceRange(mLoc, Tok.getLocation());
|
2008-08-23 09:48:03 +08:00
|
|
|
// Skip until we get a ; or {}.
|
|
|
|
SkipUntil(tok::r_brace);
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2008-08-23 09:48:03 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-04-08 08:30:06 +08:00
|
|
|
llvm::SmallVector<DeclaratorChunk::ParamInfo, 8> CParamInfo;
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::colon)) {
|
2007-10-07 10:00:24 +08:00
|
|
|
// If attributes exist after the method, parse them.
|
2010-12-24 10:08:15 +08:00
|
|
|
if (getLang().ObjC2)
|
|
|
|
MaybeParseGNUAttributes(attrs);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-10-07 10:00:24 +08:00
|
|
|
Selector Sel = PP.getSelectorTable().getNullarySelector(SelIdent);
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *Result
|
2011-02-10 06:20:01 +08:00
|
|
|
= Actions.ActOnMethodDeclaration(getCurScope(), mLoc, Tok.getLocation(),
|
2007-11-10 03:52:12 +08:00
|
|
|
mType, IDecl, DSRet, ReturnType, Sel,
|
2010-04-08 08:30:06 +08:00
|
|
|
0,
|
|
|
|
CParamInfo.data(), CParamInfo.size(),
|
2010-12-24 10:08:15 +08:00
|
|
|
attrs.getList(), MethodImplKind);
|
2009-11-04 10:18:39 +08:00
|
|
|
PD.complete(Result);
|
|
|
|
return Result;
|
2007-10-07 10:00:24 +08:00
|
|
|
}
|
2007-09-06 07:30:30 +08:00
|
|
|
|
Add SelectorInfo (similar in spirit to IdentifierInfo). The key difference is SelectorInfo is not string-oriented, it is a unique aggregate of IdentifierInfo's (using a folding set). SelectorInfo also has a richer API that simplifies the parser/action interface. 3 noteworthy benefits:
#1: It is cleaner. I never "liked" storing keyword selectors (i.e. foo:bar:baz) in the IdentifierTable.
#2: It is more space efficient. Since Cocoa keyword selectors can be quite long, this technique is space saving. For Cocoa.h, pulling the keyword selectors out saves ~180k. The cost of the SelectorInfo data is ~100k. Saves ~80k, or 43%.
#3: It results in many API simplifications. Here are some highlights:
- Removed 3 actions (ActOnKeywordMessage, ActOnUnaryMessage, & one flavor of ObjcBuildMethodDeclaration that was specific to unary messages).
- Removed 3 funky structs from DeclSpec.h (ObjcKeywordMessage, ObjcKeywordDecl, and ObjcKeywordInfo).
- Removed 2 ivars and 2 constructors from ObjCMessageExpr (fyi, this space savings has not been measured).
I am happy with the way it turned out (though it took a bit more hacking than I expected). Given the central role of selectors in ObjC, making sure this is "right" will pay dividends later.
Thanks to Chris for talking this through with me and suggesting this approach.
llvm-svn: 42395
2007-09-27 22:38:14 +08:00
|
|
|
llvm::SmallVector<IdentifierInfo *, 12> KeyIdents;
|
2010-08-27 07:41:50 +08:00
|
|
|
llvm::SmallVector<Sema::ObjCArgInfo, 12> ArgInfos;
|
2011-02-10 06:20:01 +08:00
|
|
|
ParseScope PrototypeScope(this,
|
|
|
|
Scope::FunctionPrototypeScope|Scope::DeclScope);
|
|
|
|
|
2007-10-07 10:00:24 +08:00
|
|
|
while (1) {
|
2010-08-27 07:41:50 +08:00
|
|
|
Sema::ObjCArgInfo ArgInfo;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-10-07 10:00:24 +08:00
|
|
|
// Each iteration parses a single keyword argument.
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::colon)) {
|
2007-10-07 10:00:24 +08:00
|
|
|
Diag(Tok, diag::err_expected_colon);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
ConsumeToken(); // Eat the ':'.
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-08-24 13:47:05 +08:00
|
|
|
ArgInfo.Type = ParsedType();
|
2009-04-12 02:57:04 +08:00
|
|
|
if (Tok.is(tok::l_paren)) // Parse the argument type if present.
|
2010-08-24 09:06:58 +08:00
|
|
|
ArgInfo.Type = ParseObjCTypeName(ArgInfo.DeclSpec, true);
|
2009-04-12 02:57:04 +08:00
|
|
|
|
2007-10-07 10:00:24 +08:00
|
|
|
// If attributes exist before the argument name, parse them.
|
2009-04-12 02:57:04 +08:00
|
|
|
ArgInfo.ArgAttrs = 0;
|
2010-12-24 10:08:15 +08:00
|
|
|
if (getLang().ObjC2) {
|
|
|
|
ParsedAttributes attrs;
|
|
|
|
MaybeParseGNUAttributes(attrs);
|
|
|
|
ArgInfo.ArgAttrs = attrs.getList();
|
|
|
|
}
|
2007-08-23 06:17:26 +08:00
|
|
|
|
2010-07-09 07:37:41 +08:00
|
|
|
// Code completion for the next piece of the selector.
|
|
|
|
if (Tok.is(tok::code_completion)) {
|
|
|
|
ConsumeCodeCompletionToken();
|
|
|
|
KeyIdents.push_back(SelIdent);
|
|
|
|
Actions.CodeCompleteObjCMethodDeclSelector(getCurScope(),
|
|
|
|
mType == tok::minus,
|
|
|
|
/*AtParameterName=*/true,
|
|
|
|
ReturnType,
|
|
|
|
KeyIdents.data(),
|
|
|
|
KeyIdents.size());
|
|
|
|
KeyIdents.pop_back();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::identifier)) {
|
2007-10-07 10:00:24 +08:00
|
|
|
Diag(Tok, diag::err_expected_ident); // missing argument name.
|
|
|
|
break;
|
2007-08-23 02:35:33 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-12 02:57:04 +08:00
|
|
|
ArgInfo.Name = Tok.getIdentifierInfo();
|
|
|
|
ArgInfo.NameLoc = Tok.getLocation();
|
2007-10-07 10:00:24 +08:00
|
|
|
ConsumeToken(); // Eat the identifier.
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-12 02:57:04 +08:00
|
|
|
ArgInfos.push_back(ArgInfo);
|
|
|
|
KeyIdents.push_back(SelIdent);
|
|
|
|
|
2010-07-09 07:20:03 +08:00
|
|
|
// Code completion for the next piece of the selector.
|
|
|
|
if (Tok.is(tok::code_completion)) {
|
|
|
|
ConsumeCodeCompletionToken();
|
|
|
|
Actions.CodeCompleteObjCMethodDeclSelector(getCurScope(),
|
|
|
|
mType == tok::minus,
|
2010-07-09 07:37:41 +08:00
|
|
|
/*AtParameterName=*/false,
|
2010-07-09 07:20:03 +08:00
|
|
|
ReturnType,
|
|
|
|
KeyIdents.data(),
|
|
|
|
KeyIdents.size());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2007-10-07 10:00:24 +08:00
|
|
|
// Check for another keyword selector.
|
2007-10-11 08:55:41 +08:00
|
|
|
SourceLocation Loc;
|
2009-04-12 02:13:45 +08:00
|
|
|
SelIdent = ParseObjCSelectorPiece(Loc);
|
2007-10-10 01:51:17 +08:00
|
|
|
if (!SelIdent && Tok.isNot(tok::colon))
|
2007-10-07 10:00:24 +08:00
|
|
|
break;
|
|
|
|
// We have a selector or a colon, continue parsing.
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-11-15 20:35:21 +08:00
|
|
|
bool isVariadic = false;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-10-07 10:00:24 +08:00
|
|
|
// Parse the (optional) parameter list.
|
2007-10-10 01:51:17 +08:00
|
|
|
while (Tok.is(tok::comma)) {
|
2007-10-07 10:00:24 +08:00
|
|
|
ConsumeToken();
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.is(tok::ellipsis)) {
|
2007-11-15 20:35:21 +08:00
|
|
|
isVariadic = true;
|
2007-08-23 02:35:33 +08:00
|
|
|
ConsumeToken();
|
2007-10-07 10:00:24 +08:00
|
|
|
break;
|
2007-08-23 02:35:33 +08:00
|
|
|
}
|
2007-10-07 10:00:24 +08:00
|
|
|
DeclSpec DS;
|
|
|
|
ParseDeclarationSpecifiers(DS);
|
2009-09-09 23:08:12 +08:00
|
|
|
// Parse the declarator.
|
2007-10-07 10:00:24 +08:00
|
|
|
Declarator ParmDecl(DS, Declarator::PrototypeContext);
|
|
|
|
ParseDeclarator(ParmDecl);
|
2010-04-08 08:30:06 +08:00
|
|
|
IdentifierInfo *ParmII = ParmDecl.getIdentifier();
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *Param = Actions.ActOnParamDeclarator(getCurScope(), ParmDecl);
|
2010-04-08 08:30:06 +08:00
|
|
|
CParamInfo.push_back(DeclaratorChunk::ParamInfo(ParmII,
|
|
|
|
ParmDecl.getIdentifierLoc(),
|
|
|
|
Param,
|
|
|
|
0));
|
|
|
|
|
2007-08-23 02:35:33 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-10-12 08:21:25 +08:00
|
|
|
// FIXME: Add support for optional parameter list...
|
2007-09-11 04:33:04 +08:00
|
|
|
// If attributes exist after the method, parse them.
|
2010-12-24 10:08:15 +08:00
|
|
|
if (getLang().ObjC2)
|
|
|
|
MaybeParseGNUAttributes(attrs);
|
2011-02-10 06:20:01 +08:00
|
|
|
|
|
|
|
if (KeyIdents.size() == 0) {
|
|
|
|
// Leave prototype scope.
|
|
|
|
PrototypeScope.Exit();
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2011-02-10 06:20:01 +08:00
|
|
|
}
|
|
|
|
|
2007-10-07 10:00:24 +08:00
|
|
|
Selector Sel = PP.getSelectorTable().getSelector(KeyIdents.size(),
|
|
|
|
&KeyIdents[0]);
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *Result
|
2011-02-10 06:20:01 +08:00
|
|
|
= Actions.ActOnMethodDeclaration(getCurScope(), mLoc, Tok.getLocation(),
|
2009-06-25 01:00:18 +08:00
|
|
|
mType, IDecl, DSRet, ReturnType, Sel,
|
2010-04-08 08:30:06 +08:00
|
|
|
&ArgInfos[0],
|
|
|
|
CParamInfo.data(), CParamInfo.size(),
|
2010-12-24 10:08:15 +08:00
|
|
|
attrs.getList(),
|
2007-11-15 20:35:21 +08:00
|
|
|
MethodImplKind, isVariadic);
|
2011-02-10 06:20:01 +08:00
|
|
|
// Leave prototype scope.
|
|
|
|
PrototypeScope.Exit();
|
|
|
|
|
2009-11-04 10:18:39 +08:00
|
|
|
PD.complete(Result);
|
|
|
|
return Result;
|
2007-08-23 00:35:03 +08:00
|
|
|
}
|
|
|
|
|
2008-07-26 12:03:38 +08:00
|
|
|
/// objc-protocol-refs:
|
|
|
|
/// '<' identifier-list '>'
|
|
|
|
///
|
|
|
|
bool Parser::
|
2010-08-21 17:40:31 +08:00
|
|
|
ParseObjCProtocolReferences(llvm::SmallVectorImpl<Decl *> &Protocols,
|
2009-09-30 03:41:44 +08:00
|
|
|
llvm::SmallVectorImpl<SourceLocation> &ProtocolLocs,
|
|
|
|
bool WarnOnDeclarations,
|
|
|
|
SourceLocation &LAngleLoc, SourceLocation &EndLoc) {
|
2008-07-26 12:03:38 +08:00
|
|
|
assert(Tok.is(tok::less) && "expected <");
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-09-30 03:41:44 +08:00
|
|
|
LAngleLoc = ConsumeToken(); // the "<"
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-07-26 12:03:38 +08:00
|
|
|
llvm::SmallVector<IdentifierLocPair, 8> ProtocolIdents;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-07-26 12:03:38 +08:00
|
|
|
while (1) {
|
2009-11-18 12:19:12 +08:00
|
|
|
if (Tok.is(tok::code_completion)) {
|
|
|
|
Actions.CodeCompleteObjCProtocolReferences(ProtocolIdents.data(),
|
|
|
|
ProtocolIdents.size());
|
2010-05-25 13:58:43 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
2009-11-18 12:19:12 +08:00
|
|
|
}
|
|
|
|
|
2008-07-26 12:03:38 +08:00
|
|
|
if (Tok.isNot(tok::identifier)) {
|
|
|
|
Diag(Tok, diag::err_expected_ident);
|
|
|
|
SkipUntil(tok::greater);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
ProtocolIdents.push_back(std::make_pair(Tok.getIdentifierInfo(),
|
|
|
|
Tok.getLocation()));
|
2009-09-30 03:41:44 +08:00
|
|
|
ProtocolLocs.push_back(Tok.getLocation());
|
2008-07-26 12:03:38 +08:00
|
|
|
ConsumeToken();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-07-26 12:03:38 +08:00
|
|
|
if (Tok.isNot(tok::comma))
|
|
|
|
break;
|
|
|
|
ConsumeToken();
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-07-26 12:03:38 +08:00
|
|
|
// Consume the '>'.
|
|
|
|
if (Tok.isNot(tok::greater)) {
|
|
|
|
Diag(Tok, diag::err_expected_greater);
|
|
|
|
return true;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-07-26 12:03:38 +08:00
|
|
|
EndLoc = ConsumeAnyToken();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-07-26 12:03:38 +08:00
|
|
|
// Convert the list of protocols identifiers into a list of protocol decls.
|
|
|
|
Actions.FindProtocolDeclaration(WarnOnDeclarations,
|
|
|
|
&ProtocolIdents[0], ProtocolIdents.size(),
|
|
|
|
Protocols);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2010-10-22 07:17:00 +08:00
|
|
|
/// \brief Parse the Objective-C protocol qualifiers that follow a typename
|
|
|
|
/// in a decl-specifier-seq, starting at the '<'.
|
2010-11-20 01:10:50 +08:00
|
|
|
bool Parser::ParseObjCProtocolQualifiers(DeclSpec &DS) {
|
2010-10-22 07:17:00 +08:00
|
|
|
assert(Tok.is(tok::less) && "Protocol qualifiers start with '<'");
|
|
|
|
assert(getLang().ObjC1 && "Protocol qualifiers only exist in Objective-C");
|
|
|
|
SourceLocation LAngleLoc, EndProtoLoc;
|
|
|
|
llvm::SmallVector<Decl *, 8> ProtocolDecl;
|
|
|
|
llvm::SmallVector<SourceLocation, 8> ProtocolLocs;
|
2010-11-20 01:10:50 +08:00
|
|
|
bool Result = ParseObjCProtocolReferences(ProtocolDecl, ProtocolLocs, false,
|
|
|
|
LAngleLoc, EndProtoLoc);
|
2010-10-22 07:17:00 +08:00
|
|
|
DS.setProtocolQualifiers(ProtocolDecl.data(), ProtocolDecl.size(),
|
|
|
|
ProtocolLocs.data(), LAngleLoc);
|
|
|
|
if (EndProtoLoc.isValid())
|
|
|
|
DS.SetRangeEnd(EndProtoLoc);
|
2010-11-20 01:10:50 +08:00
|
|
|
return Result;
|
2010-10-22 07:17:00 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-08-21 05:31:48 +08:00
|
|
|
/// objc-class-instance-variables:
|
|
|
|
/// '{' objc-instance-variable-decl-list[opt] '}'
|
|
|
|
///
|
|
|
|
/// objc-instance-variable-decl-list:
|
|
|
|
/// objc-visibility-spec
|
|
|
|
/// objc-instance-variable-decl ';'
|
|
|
|
/// ';'
|
|
|
|
/// objc-instance-variable-decl-list objc-visibility-spec
|
|
|
|
/// objc-instance-variable-decl-list objc-instance-variable-decl ';'
|
|
|
|
/// objc-instance-variable-decl-list ';'
|
|
|
|
///
|
|
|
|
/// objc-visibility-spec:
|
|
|
|
/// @private
|
|
|
|
/// @protected
|
|
|
|
/// @public
|
2007-08-22 05:17:12 +08:00
|
|
|
/// @package [OBJC2]
|
2007-08-21 05:31:48 +08:00
|
|
|
///
|
|
|
|
/// objc-instance-variable-decl:
|
2009-09-09 23:08:12 +08:00
|
|
|
/// struct-declaration
|
2007-08-21 05:31:48 +08:00
|
|
|
///
|
2010-08-21 17:40:31 +08:00
|
|
|
void Parser::ParseObjCClassInstanceVariables(Decl *interfaceDecl,
|
2010-02-23 07:04:20 +08:00
|
|
|
tok::ObjCKeywordKind visibility,
|
2007-10-30 05:38:07 +08:00
|
|
|
SourceLocation atLoc) {
|
2007-10-10 01:51:17 +08:00
|
|
|
assert(Tok.is(tok::l_brace) && "expected {");
|
2010-08-21 17:40:31 +08:00
|
|
|
llvm::SmallVector<Decl *, 32> AllIvarDecls;
|
2008-04-10 14:46:29 +08:00
|
|
|
|
2009-01-13 02:45:55 +08:00
|
|
|
ParseScope ClassScope(this, Scope::DeclScope|Scope::ClassScope);
|
Unify the code for defining tags in C and C++, so that we always
introduce a Scope for the body of a tag. This reduces the number of
semantic differences between C and C++ structs and unions, and will
help with other features (e.g., anonymous unions) in C. Some important
points:
- Fields are now in the "member" namespace (IDNS_Member), to keep
them separate from tags and ordinary names in C. See the new test
in Sema/member-reference.c for an example of why this matters. In
C++, ordinary and member name lookup will find members in both the
ordinary and member namespace, so the difference between
IDNS_Member and IDNS_Ordinary is erased by Sema::LookupDecl (but
only in C++!).
- We always introduce a Scope and push a DeclContext when we're
defining a tag, in both C and C++. Previously, we had different
actions and different Scope/CurContext behavior for enums, C
structs/unions, and C++ structs/unions/classes. Now, it's one pair
of actions. (Yay!)
There's still some fuzziness in the handling of struct/union/enum
definitions within other struct/union/enum definitions in C. We'll
need to do some more cleanup to eliminate some reliance on CurContext
before we can solve this issue for real. What we want is for something
like this:
struct X {
struct T { int x; } t;
};
to introduce T into translation unit scope (placing it at the
appropriate point in the IdentifierResolver chain, too), but it should
still have struct X as its lexical declaration
context. PushOnScopeChains isn't smart enough to do that yet, though,
so there's a FIXME test in nested-redef.c
llvm-svn: 61940
2009-01-09 04:45:30 +08:00
|
|
|
|
2007-08-22 05:17:12 +08:00
|
|
|
SourceLocation LBraceLoc = ConsumeBrace(); // the "{"
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-08-22 05:17:12 +08:00
|
|
|
// While we still have something to read, read the instance variables.
|
2007-10-10 01:51:17 +08:00
|
|
|
while (Tok.isNot(tok::r_brace) && Tok.isNot(tok::eof)) {
|
2007-08-22 05:17:12 +08:00
|
|
|
// Each iteration of this loop reads one objc-instance-variable-decl.
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-08-22 05:17:12 +08:00
|
|
|
// Check for extraneous top-level semicolon.
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.is(tok::semi)) {
|
2010-06-17 07:08:59 +08:00
|
|
|
Diag(Tok, diag::ext_extra_ivar_semi)
|
2010-04-01 01:46:05 +08:00
|
|
|
<< FixItHint::CreateRemoval(Tok.getLocation());
|
2007-08-22 05:17:12 +08:00
|
|
|
ConsumeToken();
|
|
|
|
continue;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-08-22 05:17:12 +08:00
|
|
|
// Set the default visibility to private.
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.is(tok::at)) { // parse objc-visibility-spec
|
2007-08-22 05:17:12 +08:00
|
|
|
ConsumeToken(); // eat the @ sign
|
2010-01-14 05:54:15 +08:00
|
|
|
|
|
|
|
if (Tok.is(tok::code_completion)) {
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.CodeCompleteObjCAtVisibility(getCurScope());
|
2010-05-25 13:58:43 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
2010-01-14 05:54:15 +08:00
|
|
|
}
|
|
|
|
|
2007-08-24 02:16:40 +08:00
|
|
|
switch (Tok.getObjCKeywordID()) {
|
2007-08-22 05:17:12 +08:00
|
|
|
case tok::objc_private:
|
|
|
|
case tok::objc_public:
|
|
|
|
case tok::objc_protected:
|
|
|
|
case tok::objc_package:
|
2007-08-24 02:16:40 +08:00
|
|
|
visibility = Tok.getObjCKeywordID();
|
2007-08-22 05:17:12 +08:00
|
|
|
ConsumeToken();
|
2009-09-09 23:08:12 +08:00
|
|
|
continue;
|
2007-08-22 05:17:12 +08:00
|
|
|
default:
|
|
|
|
Diag(Tok, diag::err_objc_illegal_visibility_spec);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-01-14 05:54:15 +08:00
|
|
|
if (Tok.is(tok::code_completion)) {
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.CodeCompleteOrdinaryName(getCurScope(),
|
2010-08-27 07:41:50 +08:00
|
|
|
Sema::PCC_ObjCInstanceVariableList);
|
2010-05-25 13:58:43 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
2010-01-14 05:54:15 +08:00
|
|
|
}
|
|
|
|
|
2009-11-03 10:38:08 +08:00
|
|
|
struct ObjCIvarCallback : FieldCallback {
|
|
|
|
Parser &P;
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *IDecl;
|
2009-11-03 10:38:08 +08:00
|
|
|
tok::ObjCKeywordKind visibility;
|
2010-08-21 17:40:31 +08:00
|
|
|
llvm::SmallVectorImpl<Decl *> &AllIvarDecls;
|
2009-11-03 10:38:08 +08:00
|
|
|
|
2010-08-21 17:40:31 +08:00
|
|
|
ObjCIvarCallback(Parser &P, Decl *IDecl, tok::ObjCKeywordKind V,
|
|
|
|
llvm::SmallVectorImpl<Decl *> &AllIvarDecls) :
|
2009-11-03 10:38:08 +08:00
|
|
|
P(P), IDecl(IDecl), visibility(V), AllIvarDecls(AllIvarDecls) {
|
|
|
|
}
|
|
|
|
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *invoke(FieldDeclarator &FD) {
|
2009-11-03 10:38:08 +08:00
|
|
|
// Install the declarator into the interface decl.
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *Field
|
2010-07-03 01:43:08 +08:00
|
|
|
= P.Actions.ActOnIvar(P.getCurScope(),
|
2009-11-03 10:38:08 +08:00
|
|
|
FD.D.getDeclSpec().getSourceRange().getBegin(),
|
|
|
|
IDecl, FD.D, FD.BitfieldSize, visibility);
|
2010-04-07 06:43:48 +08:00
|
|
|
if (Field)
|
|
|
|
AllIvarDecls.push_back(Field);
|
2009-11-03 10:38:08 +08:00
|
|
|
return Field;
|
|
|
|
}
|
|
|
|
} Callback(*this, interfaceDecl, visibility, AllIvarDecls);
|
2010-08-24 06:46:52 +08:00
|
|
|
|
2008-04-10 14:46:29 +08:00
|
|
|
// Parse all the comma separated declarators.
|
|
|
|
DeclSpec DS;
|
2009-11-03 10:38:08 +08:00
|
|
|
ParseStructDeclaration(DS, Callback);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.is(tok::semi)) {
|
2007-08-22 05:17:12 +08:00
|
|
|
ConsumeToken();
|
|
|
|
} else {
|
|
|
|
Diag(Tok, diag::err_expected_semi_decl_list);
|
|
|
|
// Skip to end of block or statement
|
|
|
|
SkipUntil(tok::r_brace, true, true);
|
|
|
|
}
|
|
|
|
}
|
2007-10-30 05:38:07 +08:00
|
|
|
SourceLocation RBraceLoc = MatchRHSPunctuation(tok::r_brace, LBraceLoc);
|
2010-08-24 06:46:52 +08:00
|
|
|
Actions.ActOnLastBitfield(RBraceLoc, interfaceDecl, AllIvarDecls);
|
2007-11-01 06:11:35 +08:00
|
|
|
// Call ActOnFields() even if we don't have any decls. This is useful
|
|
|
|
// for code rewriting tools that need to be aware of the empty list.
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.ActOnFields(getCurScope(), atLoc, interfaceDecl,
|
2009-05-21 17:52:38 +08:00
|
|
|
AllIvarDecls.data(), AllIvarDecls.size(),
|
2008-10-03 10:03:53 +08:00
|
|
|
LBraceLoc, RBraceLoc, 0);
|
2007-08-22 05:17:12 +08:00
|
|
|
return;
|
2006-11-05 10:08:13 +08:00
|
|
|
}
|
2007-08-21 05:31:48 +08:00
|
|
|
|
|
|
|
/// objc-protocol-declaration:
|
|
|
|
/// objc-protocol-definition
|
|
|
|
/// objc-protocol-forward-reference
|
|
|
|
///
|
|
|
|
/// objc-protocol-definition:
|
2009-09-09 23:08:12 +08:00
|
|
|
/// @protocol identifier
|
|
|
|
/// objc-protocol-refs[opt]
|
|
|
|
/// objc-interface-decl-list
|
2007-08-21 05:31:48 +08:00
|
|
|
/// @end
|
|
|
|
///
|
|
|
|
/// objc-protocol-forward-reference:
|
|
|
|
/// @protocol identifier-list ';'
|
|
|
|
///
|
|
|
|
/// "@protocol identifier ;" should be resolved as "@protocol
|
2007-09-07 05:24:23 +08:00
|
|
|
/// identifier-list ;": objc-interface-decl-list may not start with a
|
2007-08-21 05:31:48 +08:00
|
|
|
/// semicolon in the first alternative if objc-protocol-refs are omitted.
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *Parser::ParseObjCAtProtocolDeclaration(SourceLocation AtLoc,
|
2010-12-24 10:08:15 +08:00
|
|
|
ParsedAttributes &attrs) {
|
2007-08-24 02:16:40 +08:00
|
|
|
assert(Tok.isObjCAtKeyword(tok::objc_protocol) &&
|
2007-08-23 06:17:26 +08:00
|
|
|
"ParseObjCAtProtocolDeclaration(): Expected @protocol");
|
|
|
|
ConsumeToken(); // the "protocol" identifier
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-11-18 12:49:41 +08:00
|
|
|
if (Tok.is(tok::code_completion)) {
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.CodeCompleteObjCProtocolDecl(getCurScope());
|
2010-05-25 13:58:43 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
2009-11-18 12:49:41 +08:00
|
|
|
}
|
|
|
|
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::identifier)) {
|
2007-08-23 06:17:26 +08:00
|
|
|
Diag(Tok, diag::err_expected_ident); // missing protocol name.
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2007-08-23 06:17:26 +08:00
|
|
|
}
|
|
|
|
// Save the protocol name, then consume it.
|
|
|
|
IdentifierInfo *protocolName = Tok.getIdentifierInfo();
|
|
|
|
SourceLocation nameLoc = ConsumeToken();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.is(tok::semi)) { // forward declaration of one protocol.
|
2008-07-22 06:17:28 +08:00
|
|
|
IdentifierLocPair ProtoInfo(protocolName, nameLoc);
|
2007-08-23 06:17:26 +08:00
|
|
|
ConsumeToken();
|
2009-09-09 23:08:12 +08:00
|
|
|
return Actions.ActOnForwardProtocolDeclaration(AtLoc, &ProtoInfo, 1,
|
2010-12-24 10:08:15 +08:00
|
|
|
attrs.getList());
|
2007-08-23 06:17:26 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.is(tok::comma)) { // list of forward declarations.
|
2008-07-22 06:17:28 +08:00
|
|
|
llvm::SmallVector<IdentifierLocPair, 8> ProtocolRefs;
|
|
|
|
ProtocolRefs.push_back(std::make_pair(protocolName, nameLoc));
|
|
|
|
|
2007-08-23 06:17:26 +08:00
|
|
|
// Parse the list of forward declarations.
|
|
|
|
while (1) {
|
|
|
|
ConsumeToken(); // the ','
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::identifier)) {
|
2007-08-23 06:17:26 +08:00
|
|
|
Diag(Tok, diag::err_expected_ident);
|
|
|
|
SkipUntil(tok::semi);
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2007-08-23 06:17:26 +08:00
|
|
|
}
|
2008-07-22 06:17:28 +08:00
|
|
|
ProtocolRefs.push_back(IdentifierLocPair(Tok.getIdentifierInfo(),
|
|
|
|
Tok.getLocation()));
|
2007-08-23 06:17:26 +08:00
|
|
|
ConsumeToken(); // the identifier
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::comma))
|
2007-08-23 06:17:26 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
// Consume the ';'.
|
|
|
|
if (ExpectAndConsume(tok::semi, diag::err_expected_semi_after, "@protocol"))
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-10-11 01:32:04 +08:00
|
|
|
return Actions.ActOnForwardProtocolDeclaration(AtLoc,
|
2009-09-09 23:08:12 +08:00
|
|
|
&ProtocolRefs[0],
|
2008-12-17 09:07:27 +08:00
|
|
|
ProtocolRefs.size(),
|
2010-12-24 10:08:15 +08:00
|
|
|
attrs.getList());
|
2008-07-22 06:17:28 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-08-23 06:17:26 +08:00
|
|
|
// Last, and definitely not least, parse a protocol declaration.
|
2009-09-30 03:41:44 +08:00
|
|
|
SourceLocation LAngleLoc, EndProtoLoc;
|
2008-07-22 06:17:28 +08:00
|
|
|
|
2010-08-21 17:40:31 +08:00
|
|
|
llvm::SmallVector<Decl *, 8> ProtocolRefs;
|
2009-09-30 03:41:44 +08:00
|
|
|
llvm::SmallVector<SourceLocation, 8> ProtocolLocs;
|
2008-07-22 06:17:28 +08:00
|
|
|
if (Tok.is(tok::less) &&
|
2009-09-30 03:41:44 +08:00
|
|
|
ParseObjCProtocolReferences(ProtocolRefs, ProtocolLocs, false,
|
|
|
|
LAngleLoc, EndProtoLoc))
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *ProtoType =
|
2008-07-26 12:03:38 +08:00
|
|
|
Actions.ActOnStartProtocolInterface(AtLoc, protocolName, nameLoc,
|
2009-05-21 17:52:38 +08:00
|
|
|
ProtocolRefs.data(),
|
|
|
|
ProtocolRefs.size(),
|
2010-01-16 23:02:53 +08:00
|
|
|
ProtocolLocs.data(),
|
2010-12-24 10:08:15 +08:00
|
|
|
EndProtoLoc, attrs.getList());
|
2007-09-18 05:07:36 +08:00
|
|
|
ParseObjCInterfaceDeclList(ProtoType, tok::objc_protocol);
|
2008-10-20 14:10:06 +08:00
|
|
|
return ProtoType;
|
2007-08-21 05:31:48 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// objc-implementation:
|
|
|
|
/// objc-class-implementation-prologue
|
|
|
|
/// objc-category-implementation-prologue
|
|
|
|
///
|
|
|
|
/// objc-class-implementation-prologue:
|
|
|
|
/// @implementation identifier objc-superclass[opt]
|
|
|
|
/// objc-class-instance-variables[opt]
|
|
|
|
///
|
|
|
|
/// objc-category-implementation-prologue:
|
|
|
|
/// @implementation identifier ( identifier )
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *Parser::ParseObjCAtImplementationDeclaration(
|
2007-09-01 08:26:16 +08:00
|
|
|
SourceLocation atLoc) {
|
|
|
|
assert(Tok.isObjCAtKeyword(tok::objc_implementation) &&
|
|
|
|
"ParseObjCAtImplementationDeclaration(): Expected @implementation");
|
|
|
|
ConsumeToken(); // the "implementation" identifier
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-11-19 00:26:39 +08:00
|
|
|
// Code completion after '@implementation'.
|
|
|
|
if (Tok.is(tok::code_completion)) {
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.CodeCompleteObjCImplementationDecl(getCurScope());
|
2010-05-25 13:58:43 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
2009-11-19 00:26:39 +08:00
|
|
|
}
|
|
|
|
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::identifier)) {
|
2007-09-01 08:26:16 +08:00
|
|
|
Diag(Tok, diag::err_expected_ident); // missing class or category name.
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2007-09-01 08:26:16 +08:00
|
|
|
}
|
|
|
|
// We have a class or category name - consume it.
|
2007-09-26 02:38:09 +08:00
|
|
|
IdentifierInfo *nameId = Tok.getIdentifierInfo();
|
2007-09-01 08:26:16 +08:00
|
|
|
SourceLocation nameLoc = ConsumeToken(); // consume class or category name
|
2009-09-09 23:08:12 +08:00
|
|
|
|
|
|
|
if (Tok.is(tok::l_paren)) {
|
2007-09-01 08:26:16 +08:00
|
|
|
// we have a category implementation.
|
2011-01-18 10:00:16 +08:00
|
|
|
ConsumeParen();
|
2007-09-01 08:26:16 +08:00
|
|
|
SourceLocation categoryLoc, rparenLoc;
|
|
|
|
IdentifierInfo *categoryId = 0;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-11-19 03:08:43 +08:00
|
|
|
if (Tok.is(tok::code_completion)) {
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.CodeCompleteObjCImplementationCategory(getCurScope(), nameId, nameLoc);
|
2010-05-25 13:58:43 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
2009-11-19 03:08:43 +08:00
|
|
|
}
|
|
|
|
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.is(tok::identifier)) {
|
2007-09-01 08:26:16 +08:00
|
|
|
categoryId = Tok.getIdentifierInfo();
|
|
|
|
categoryLoc = ConsumeToken();
|
|
|
|
} else {
|
|
|
|
Diag(Tok, diag::err_expected_ident); // missing category name.
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2009-09-09 23:08:12 +08:00
|
|
|
}
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::r_paren)) {
|
2007-09-01 08:26:16 +08:00
|
|
|
Diag(Tok, diag::err_expected_rparen);
|
|
|
|
SkipUntil(tok::r_paren, false); // don't stop at ';'
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2007-09-01 08:26:16 +08:00
|
|
|
}
|
|
|
|
rparenLoc = ConsumeParen();
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *ImplCatType = Actions.ActOnStartCategoryImplementation(
|
2009-09-09 23:08:12 +08:00
|
|
|
atLoc, nameId, nameLoc, categoryId,
|
2007-10-03 00:38:50 +08:00
|
|
|
categoryLoc);
|
2008-01-08 03:49:32 +08:00
|
|
|
ObjCImpDecl = ImplCatType;
|
2009-11-17 02:57:01 +08:00
|
|
|
PendingObjCImpDecl.push_back(ObjCImpDecl);
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2007-09-01 08:26:16 +08:00
|
|
|
}
|
|
|
|
// We have a class implementation
|
2007-09-26 02:38:09 +08:00
|
|
|
SourceLocation superClassLoc;
|
|
|
|
IdentifierInfo *superClassId = 0;
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.is(tok::colon)) {
|
2007-09-01 08:26:16 +08:00
|
|
|
// We have a super class
|
|
|
|
ConsumeToken();
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::identifier)) {
|
2007-09-01 08:26:16 +08:00
|
|
|
Diag(Tok, diag::err_expected_ident); // missing super class name.
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2007-09-01 08:26:16 +08:00
|
|
|
}
|
2007-09-26 02:38:09 +08:00
|
|
|
superClassId = Tok.getIdentifierInfo();
|
|
|
|
superClassLoc = ConsumeToken(); // Consume super class name
|
2007-09-01 08:26:16 +08:00
|
|
|
}
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *ImplClsType = Actions.ActOnStartClassImplementation(
|
2007-12-28 03:57:00 +08:00
|
|
|
atLoc, nameId, nameLoc,
|
2007-09-26 02:38:09 +08:00
|
|
|
superClassId, superClassLoc);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-10-30 05:38:07 +08:00
|
|
|
if (Tok.is(tok::l_brace)) // we have ivars
|
2010-02-23 07:04:20 +08:00
|
|
|
ParseObjCClassInstanceVariables(ImplClsType/*FIXME*/,
|
2010-03-23 03:04:14 +08:00
|
|
|
tok::objc_private, atLoc);
|
2008-01-08 03:49:32 +08:00
|
|
|
ObjCImpDecl = ImplClsType;
|
2009-11-17 02:57:01 +08:00
|
|
|
PendingObjCImpDecl.push_back(ObjCImpDecl);
|
|
|
|
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2006-11-05 10:08:13 +08:00
|
|
|
}
|
2007-10-30 05:38:07 +08:00
|
|
|
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *Parser::ParseObjCAtEndDeclaration(SourceRange atEnd) {
|
2007-09-01 08:26:16 +08:00
|
|
|
assert(Tok.isObjCAtKeyword(tok::objc_end) &&
|
|
|
|
"ParseObjCAtEndDeclaration(): Expected @end");
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *Result = ObjCImpDecl;
|
2007-09-01 08:26:16 +08:00
|
|
|
ConsumeToken(); // the "end" identifier
|
2009-03-05 06:30:12 +08:00
|
|
|
if (ObjCImpDecl) {
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.ActOnAtEnd(getCurScope(), atEnd, ObjCImpDecl);
|
2010-08-21 17:40:31 +08:00
|
|
|
ObjCImpDecl = 0;
|
2009-11-17 02:57:01 +08:00
|
|
|
PendingObjCImpDecl.pop_back();
|
2009-03-05 06:30:12 +08:00
|
|
|
}
|
2010-01-07 09:20:12 +08:00
|
|
|
else {
|
|
|
|
// missing @implementation
|
|
|
|
Diag(atEnd.getBegin(), diag::warn_expected_implementation);
|
|
|
|
}
|
2009-03-05 06:30:12 +08:00
|
|
|
return Result;
|
2006-11-05 10:08:13 +08:00
|
|
|
}
|
2007-09-05 03:26:51 +08:00
|
|
|
|
2010-07-23 02:24:20 +08:00
|
|
|
Parser::DeclGroupPtrTy Parser::FinishPendingObjCActions() {
|
|
|
|
Actions.DiagnoseUseOfUnimplementedSelectors();
|
2009-11-17 02:57:01 +08:00
|
|
|
if (PendingObjCImpDecl.empty())
|
2010-08-21 17:40:31 +08:00
|
|
|
return Actions.ConvertDeclToDeclGroup(0);
|
|
|
|
Decl *ImpDecl = PendingObjCImpDecl.pop_back_val();
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.ActOnAtEnd(getCurScope(), SourceRange(), ImpDecl);
|
2009-11-17 02:57:01 +08:00
|
|
|
return Actions.ConvertDeclToDeclGroup(ImpDecl);
|
|
|
|
}
|
|
|
|
|
2007-09-05 03:26:51 +08:00
|
|
|
/// compatibility-alias-decl:
|
|
|
|
/// @compatibility_alias alias-name class-name ';'
|
|
|
|
///
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *Parser::ParseObjCAtAliasDeclaration(SourceLocation atLoc) {
|
2007-09-05 03:26:51 +08:00
|
|
|
assert(Tok.isObjCAtKeyword(tok::objc_compatibility_alias) &&
|
|
|
|
"ParseObjCAtAliasDeclaration(): Expected @compatibility_alias");
|
|
|
|
ConsumeToken(); // consume compatibility_alias
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::identifier)) {
|
2007-09-05 03:26:51 +08:00
|
|
|
Diag(Tok, diag::err_expected_ident);
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2007-09-05 03:26:51 +08:00
|
|
|
}
|
2007-10-12 07:42:27 +08:00
|
|
|
IdentifierInfo *aliasId = Tok.getIdentifierInfo();
|
|
|
|
SourceLocation aliasLoc = ConsumeToken(); // consume alias-name
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::identifier)) {
|
2007-09-05 03:26:51 +08:00
|
|
|
Diag(Tok, diag::err_expected_ident);
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2007-09-05 03:26:51 +08:00
|
|
|
}
|
2007-10-12 07:42:27 +08:00
|
|
|
IdentifierInfo *classId = Tok.getIdentifierInfo();
|
|
|
|
SourceLocation classLoc = ConsumeToken(); // consume class-name;
|
2011-01-05 09:10:06 +08:00
|
|
|
ExpectAndConsume(tok::semi, diag::err_expected_semi_after,
|
|
|
|
"@compatibility_alias");
|
2009-03-29 03:18:32 +08:00
|
|
|
return Actions.ActOnCompatiblityAlias(atLoc, aliasId, aliasLoc,
|
|
|
|
classId, classLoc);
|
2006-11-05 10:08:13 +08:00
|
|
|
}
|
|
|
|
|
2007-09-01 08:26:16 +08:00
|
|
|
/// property-synthesis:
|
|
|
|
/// @synthesize property-ivar-list ';'
|
|
|
|
///
|
|
|
|
/// property-ivar-list:
|
|
|
|
/// property-ivar
|
|
|
|
/// property-ivar-list ',' property-ivar
|
|
|
|
///
|
|
|
|
/// property-ivar:
|
|
|
|
/// identifier
|
|
|
|
/// identifier '=' identifier
|
|
|
|
///
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *Parser::ParseObjCPropertySynthesize(SourceLocation atLoc) {
|
2007-09-01 08:26:16 +08:00
|
|
|
assert(Tok.isObjCAtKeyword(tok::objc_synthesize) &&
|
|
|
|
"ParseObjCPropertyDynamic(): Expected '@synthesize'");
|
2011-01-18 10:00:16 +08:00
|
|
|
ConsumeToken(); // consume synthesize
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-11-19 03:45:45 +08:00
|
|
|
while (true) {
|
2009-11-19 06:32:06 +08:00
|
|
|
if (Tok.is(tok::code_completion)) {
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.CodeCompleteObjCPropertyDefinition(getCurScope(), ObjCImpDecl);
|
2010-05-25 13:58:43 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
2009-11-19 06:32:06 +08:00
|
|
|
}
|
|
|
|
|
2009-11-19 03:45:45 +08:00
|
|
|
if (Tok.isNot(tok::identifier)) {
|
|
|
|
Diag(Tok, diag::err_synthesized_property_name);
|
|
|
|
SkipUntil(tok::semi);
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2009-11-19 03:45:45 +08:00
|
|
|
}
|
|
|
|
|
2008-04-18 08:19:30 +08:00
|
|
|
IdentifierInfo *propertyIvar = 0;
|
|
|
|
IdentifierInfo *propertyId = Tok.getIdentifierInfo();
|
|
|
|
SourceLocation propertyLoc = ConsumeToken(); // consume property name
|
2010-11-17 09:03:52 +08:00
|
|
|
SourceLocation propertyIvarLoc;
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.is(tok::equal)) {
|
2007-09-01 08:26:16 +08:00
|
|
|
// property '=' ivar-name
|
|
|
|
ConsumeToken(); // consume '='
|
2009-11-19 06:32:06 +08:00
|
|
|
|
|
|
|
if (Tok.is(tok::code_completion)) {
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.CodeCompleteObjCPropertySynthesizeIvar(getCurScope(), propertyId,
|
2009-11-19 06:32:06 +08:00
|
|
|
ObjCImpDecl);
|
2010-05-25 13:58:43 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
2009-11-19 06:32:06 +08:00
|
|
|
}
|
|
|
|
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::identifier)) {
|
2007-09-01 08:26:16 +08:00
|
|
|
Diag(Tok, diag::err_expected_ident);
|
|
|
|
break;
|
|
|
|
}
|
2008-04-18 08:19:30 +08:00
|
|
|
propertyIvar = Tok.getIdentifierInfo();
|
2010-11-17 09:03:52 +08:00
|
|
|
propertyIvarLoc = ConsumeToken(); // consume ivar-name
|
2007-09-01 08:26:16 +08:00
|
|
|
}
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.ActOnPropertyImplDecl(getCurScope(), atLoc, propertyLoc, true, ObjCImpDecl,
|
2010-11-17 09:03:52 +08:00
|
|
|
propertyId, propertyIvar, propertyIvarLoc);
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::comma))
|
2007-09-01 08:26:16 +08:00
|
|
|
break;
|
|
|
|
ConsumeToken(); // consume ','
|
|
|
|
}
|
2011-01-05 09:10:06 +08:00
|
|
|
ExpectAndConsume(tok::semi, diag::err_expected_semi_after, "@synthesize");
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2007-09-01 08:26:16 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// property-dynamic:
|
|
|
|
/// @dynamic property-list
|
|
|
|
///
|
|
|
|
/// property-list:
|
|
|
|
/// identifier
|
|
|
|
/// property-list ',' identifier
|
|
|
|
///
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *Parser::ParseObjCPropertyDynamic(SourceLocation atLoc) {
|
2007-09-01 08:26:16 +08:00
|
|
|
assert(Tok.isObjCAtKeyword(tok::objc_dynamic) &&
|
|
|
|
"ParseObjCPropertyDynamic(): Expected '@dynamic'");
|
2011-01-18 10:00:16 +08:00
|
|
|
ConsumeToken(); // consume dynamic
|
2009-11-19 06:56:13 +08:00
|
|
|
while (true) {
|
|
|
|
if (Tok.is(tok::code_completion)) {
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.CodeCompleteObjCPropertyDefinition(getCurScope(), ObjCImpDecl);
|
2010-05-25 13:58:43 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
2009-11-19 06:56:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (Tok.isNot(tok::identifier)) {
|
|
|
|
Diag(Tok, diag::err_expected_ident);
|
|
|
|
SkipUntil(tok::semi);
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2009-11-19 06:56:13 +08:00
|
|
|
}
|
|
|
|
|
2008-04-22 05:05:54 +08:00
|
|
|
IdentifierInfo *propertyId = Tok.getIdentifierInfo();
|
|
|
|
SourceLocation propertyLoc = ConsumeToken(); // consume property name
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.ActOnPropertyImplDecl(getCurScope(), atLoc, propertyLoc, false, ObjCImpDecl,
|
2010-11-17 09:03:52 +08:00
|
|
|
propertyId, 0, SourceLocation());
|
2008-04-22 05:05:54 +08:00
|
|
|
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::comma))
|
2007-09-01 08:26:16 +08:00
|
|
|
break;
|
|
|
|
ConsumeToken(); // consume ','
|
|
|
|
}
|
2011-01-05 09:10:06 +08:00
|
|
|
ExpectAndConsume(tok::semi, diag::err_expected_semi_after, "@dynamic");
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2007-09-01 08:26:16 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-09-20 03:14:32 +08:00
|
|
|
/// objc-throw-statement:
|
|
|
|
/// throw expression[opt];
|
|
|
|
///
|
2010-08-24 14:29:42 +08:00
|
|
|
StmtResult Parser::ParseObjCThrowStmt(SourceLocation atLoc) {
|
|
|
|
ExprResult Res;
|
2007-09-20 03:14:32 +08:00
|
|
|
ConsumeToken(); // consume throw
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::semi)) {
|
2007-11-07 10:00:49 +08:00
|
|
|
Res = ParseExpression();
|
2008-12-09 21:15:23 +08:00
|
|
|
if (Res.isInvalid()) {
|
2007-09-20 03:14:32 +08:00
|
|
|
SkipUntil(tok::semi);
|
2008-12-12 04:12:42 +08:00
|
|
|
return StmtError();
|
2007-09-20 03:14:32 +08:00
|
|
|
}
|
|
|
|
}
|
2010-04-21 05:21:51 +08:00
|
|
|
// consume ';'
|
|
|
|
ExpectAndConsume(tok::semi, diag::err_expected_semi_after, "@throw");
|
2010-08-24 07:25:46 +08:00
|
|
|
return Actions.ActOnObjCAtThrowStmt(atLoc, Res.take(), getCurScope());
|
2007-09-20 03:14:32 +08:00
|
|
|
}
|
|
|
|
|
2008-01-30 02:21:32 +08:00
|
|
|
/// objc-synchronized-statement:
|
2008-01-31 01:38:29 +08:00
|
|
|
/// @synchronized '(' expression ')' compound-statement
|
2008-01-30 02:21:32 +08:00
|
|
|
///
|
2010-08-24 14:29:42 +08:00
|
|
|
StmtResult
|
2008-12-12 04:12:42 +08:00
|
|
|
Parser::ParseObjCSynchronizedStmt(SourceLocation atLoc) {
|
2008-01-30 03:14:59 +08:00
|
|
|
ConsumeToken(); // consume synchronized
|
|
|
|
if (Tok.isNot(tok::l_paren)) {
|
2008-11-18 15:48:38 +08:00
|
|
|
Diag(Tok, diag::err_expected_lparen_after) << "@synchronized";
|
2008-12-12 04:12:42 +08:00
|
|
|
return StmtError();
|
2008-01-30 03:14:59 +08:00
|
|
|
}
|
|
|
|
ConsumeParen(); // '('
|
2010-08-24 14:29:42 +08:00
|
|
|
ExprResult Res(ParseExpression());
|
2008-12-09 21:15:23 +08:00
|
|
|
if (Res.isInvalid()) {
|
2008-01-30 03:14:59 +08:00
|
|
|
SkipUntil(tok::semi);
|
2008-12-12 04:12:42 +08:00
|
|
|
return StmtError();
|
2008-01-30 03:14:59 +08:00
|
|
|
}
|
|
|
|
if (Tok.isNot(tok::r_paren)) {
|
2008-11-18 15:48:38 +08:00
|
|
|
Diag(Tok, diag::err_expected_lbrace);
|
2008-12-12 04:12:42 +08:00
|
|
|
return StmtError();
|
2008-01-30 03:14:59 +08:00
|
|
|
}
|
|
|
|
ConsumeParen(); // ')'
|
2008-01-31 01:38:29 +08:00
|
|
|
if (Tok.isNot(tok::l_brace)) {
|
2008-11-18 15:48:38 +08:00
|
|
|
Diag(Tok, diag::err_expected_lbrace);
|
2008-12-12 04:12:42 +08:00
|
|
|
return StmtError();
|
2008-01-31 01:38:29 +08:00
|
|
|
}
|
2008-06-05 04:36:13 +08:00
|
|
|
// Enter a scope to hold everything within the compound stmt. Compound
|
|
|
|
// statements can always hold declarations.
|
2008-12-10 14:34:36 +08:00
|
|
|
ParseScope BodyScope(this, Scope::DeclScope);
|
2008-06-05 04:36:13 +08:00
|
|
|
|
2010-08-24 14:29:42 +08:00
|
|
|
StmtResult SynchBody(ParseCompoundStatementBody());
|
2008-12-09 21:15:23 +08:00
|
|
|
|
2008-12-10 14:34:36 +08:00
|
|
|
BodyScope.Exit();
|
2008-12-09 21:15:23 +08:00
|
|
|
if (SynchBody.isInvalid())
|
2008-01-30 03:14:59 +08:00
|
|
|
SynchBody = Actions.ActOnNullStmt(Tok.getLocation());
|
2010-08-24 07:25:46 +08:00
|
|
|
return Actions.ActOnObjCAtSynchronizedStmt(atLoc, Res.take(), SynchBody.take());
|
2008-01-30 02:21:32 +08:00
|
|
|
}
|
|
|
|
|
2007-09-20 03:14:32 +08:00
|
|
|
/// objc-try-catch-statement:
|
|
|
|
/// @try compound-statement objc-catch-list[opt]
|
|
|
|
/// @try compound-statement objc-catch-list[opt] @finally compound-statement
|
|
|
|
///
|
|
|
|
/// objc-catch-list:
|
|
|
|
/// @catch ( parameter-declaration ) compound-statement
|
|
|
|
/// objc-catch-list @catch ( catch-parameter-declaration ) compound-statement
|
|
|
|
/// catch-parameter-declaration:
|
|
|
|
/// parameter-declaration
|
|
|
|
/// '...' [OBJC2]
|
|
|
|
///
|
2010-08-24 14:29:42 +08:00
|
|
|
StmtResult Parser::ParseObjCTryStmt(SourceLocation atLoc) {
|
2007-09-20 03:14:32 +08:00
|
|
|
bool catch_or_finally_seen = false;
|
2008-12-12 04:12:42 +08:00
|
|
|
|
2007-09-20 03:14:32 +08:00
|
|
|
ConsumeToken(); // consume try
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::l_brace)) {
|
2008-11-18 15:48:38 +08:00
|
|
|
Diag(Tok, diag::err_expected_lbrace);
|
2008-12-12 04:12:42 +08:00
|
|
|
return StmtError();
|
2007-09-20 03:14:32 +08:00
|
|
|
}
|
2010-04-24 06:50:49 +08:00
|
|
|
StmtVector CatchStmts(Actions);
|
2010-08-24 14:29:42 +08:00
|
|
|
StmtResult FinallyStmt;
|
2008-12-10 14:34:36 +08:00
|
|
|
ParseScope TryScope(this, Scope::DeclScope);
|
2010-08-24 14:29:42 +08:00
|
|
|
StmtResult TryBody(ParseCompoundStatementBody());
|
2008-12-10 14:34:36 +08:00
|
|
|
TryScope.Exit();
|
2008-12-09 21:15:23 +08:00
|
|
|
if (TryBody.isInvalid())
|
2007-11-02 23:39:31 +08:00
|
|
|
TryBody = Actions.ActOnNullStmt(Tok.getLocation());
|
2008-11-26 06:21:31 +08:00
|
|
|
|
2007-10-10 01:51:17 +08:00
|
|
|
while (Tok.is(tok::at)) {
|
2008-03-10 14:06:04 +08:00
|
|
|
// At this point, we need to lookahead to determine if this @ is the start
|
|
|
|
// of an @catch or @finally. We don't want to consume the @ token if this
|
|
|
|
// is an @try or @encode or something else.
|
|
|
|
Token AfterAt = GetLookAheadToken(1);
|
|
|
|
if (!AfterAt.isObjCAtKeyword(tok::objc_catch) &&
|
|
|
|
!AfterAt.isObjCAtKeyword(tok::objc_finally))
|
|
|
|
break;
|
2008-12-09 21:15:23 +08:00
|
|
|
|
2007-11-02 08:18:53 +08:00
|
|
|
SourceLocation AtCatchFinallyLoc = ConsumeToken();
|
2007-12-28 03:57:00 +08:00
|
|
|
if (Tok.isObjCAtKeyword(tok::objc_catch)) {
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *FirstPart = 0;
|
2007-11-02 07:59:59 +08:00
|
|
|
ConsumeToken(); // consume catch
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.is(tok::l_paren)) {
|
2007-09-20 03:14:32 +08:00
|
|
|
ConsumeParen();
|
2009-02-12 04:05:44 +08:00
|
|
|
ParseScope CatchScope(this, Scope::DeclScope|Scope::AtCatchScope);
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::ellipsis)) {
|
2007-09-20 03:14:32 +08:00
|
|
|
DeclSpec DS;
|
|
|
|
ParseDeclarationSpecifiers(DS);
|
2008-12-12 04:12:42 +08:00
|
|
|
// For some odd reason, the name of the exception variable is
|
2009-09-09 23:08:12 +08:00
|
|
|
// optional. As a result, we need to use "PrototypeContext", because
|
2009-03-04 03:52:17 +08:00
|
|
|
// we must accept either 'declarator' or 'abstract-declarator' here.
|
|
|
|
Declarator ParmDecl(DS, Declarator::PrototypeContext);
|
|
|
|
ParseDeclarator(ParmDecl);
|
|
|
|
|
2010-04-24 07:01:43 +08:00
|
|
|
// Inform the actions module about the declarator, so it
|
2009-03-04 03:52:17 +08:00
|
|
|
// gets added to the current scope.
|
2010-07-03 01:43:08 +08:00
|
|
|
FirstPart = Actions.ActOnObjCExceptionDecl(getCurScope(), ParmDecl);
|
2008-02-06 05:27:35 +08:00
|
|
|
} else
|
2007-09-20 03:14:32 +08:00
|
|
|
ConsumeToken(); // consume '...'
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-08 06:56:58 +08:00
|
|
|
SourceLocation RParenLoc;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-08 06:56:58 +08:00
|
|
|
if (Tok.is(tok::r_paren))
|
|
|
|
RParenLoc = ConsumeParen();
|
|
|
|
else // Skip over garbage, until we get to ')'. Eat the ')'.
|
|
|
|
SkipUntil(tok::r_paren, true, false);
|
|
|
|
|
2010-08-24 14:29:42 +08:00
|
|
|
StmtResult CatchBody(true);
|
2008-02-15 03:27:54 +08:00
|
|
|
if (Tok.is(tok::l_brace))
|
|
|
|
CatchBody = ParseCompoundStatementBody();
|
|
|
|
else
|
|
|
|
Diag(Tok, diag::err_expected_lbrace);
|
2008-12-09 21:15:23 +08:00
|
|
|
if (CatchBody.isInvalid())
|
2007-11-02 07:59:59 +08:00
|
|
|
CatchBody = Actions.ActOnNullStmt(Tok.getLocation());
|
2010-04-24 06:50:49 +08:00
|
|
|
|
2010-08-24 14:29:42 +08:00
|
|
|
StmtResult Catch = Actions.ActOnObjCAtCatchStmt(AtCatchFinallyLoc,
|
2010-04-24 06:50:49 +08:00
|
|
|
RParenLoc,
|
|
|
|
FirstPart,
|
2010-08-24 07:25:46 +08:00
|
|
|
CatchBody.take());
|
2010-04-24 06:50:49 +08:00
|
|
|
if (!Catch.isInvalid())
|
|
|
|
CatchStmts.push_back(Catch.release());
|
|
|
|
|
2008-02-06 05:27:35 +08:00
|
|
|
} else {
|
2008-11-18 15:48:38 +08:00
|
|
|
Diag(AtCatchFinallyLoc, diag::err_expected_lparen_after)
|
|
|
|
<< "@catch clause";
|
2008-12-12 04:12:42 +08:00
|
|
|
return StmtError();
|
2007-09-20 03:14:32 +08:00
|
|
|
}
|
|
|
|
catch_or_finally_seen = true;
|
2008-03-10 14:06:04 +08:00
|
|
|
} else {
|
|
|
|
assert(Tok.isObjCAtKeyword(tok::objc_finally) && "Lookahead confused?");
|
2008-02-06 05:27:35 +08:00
|
|
|
ConsumeToken(); // consume finally
|
2008-12-10 14:34:36 +08:00
|
|
|
ParseScope FinallyScope(this, Scope::DeclScope);
|
2008-12-09 21:15:23 +08:00
|
|
|
|
2010-08-24 14:29:42 +08:00
|
|
|
StmtResult FinallyBody(true);
|
2008-02-15 03:27:54 +08:00
|
|
|
if (Tok.is(tok::l_brace))
|
|
|
|
FinallyBody = ParseCompoundStatementBody();
|
|
|
|
else
|
|
|
|
Diag(Tok, diag::err_expected_lbrace);
|
2008-12-09 21:15:23 +08:00
|
|
|
if (FinallyBody.isInvalid())
|
2007-11-02 08:18:53 +08:00
|
|
|
FinallyBody = Actions.ActOnNullStmt(Tok.getLocation());
|
2008-12-09 21:15:23 +08:00
|
|
|
FinallyStmt = Actions.ActOnObjCAtFinallyStmt(AtCatchFinallyLoc,
|
2010-08-24 07:25:46 +08:00
|
|
|
FinallyBody.take());
|
2007-09-20 03:14:32 +08:00
|
|
|
catch_or_finally_seen = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2007-11-02 23:39:31 +08:00
|
|
|
if (!catch_or_finally_seen) {
|
2007-09-20 03:14:32 +08:00
|
|
|
Diag(atLoc, diag::err_missing_catch_finally);
|
2008-12-12 04:12:42 +08:00
|
|
|
return StmtError();
|
2007-11-02 23:39:31 +08:00
|
|
|
}
|
2010-04-24 06:50:49 +08:00
|
|
|
|
2010-08-24 07:25:46 +08:00
|
|
|
return Actions.ActOnObjCAtTryStmt(atLoc, TryBody.take(),
|
2010-04-24 06:50:49 +08:00
|
|
|
move_arg(CatchStmts),
|
2010-08-24 07:25:46 +08:00
|
|
|
FinallyStmt.take());
|
2007-09-20 03:14:32 +08:00
|
|
|
}
|
2007-09-01 08:26:16 +08:00
|
|
|
|
2007-09-07 05:24:23 +08:00
|
|
|
/// objc-method-def: objc-method-proto ';'[opt] '{' body '}'
|
2007-09-01 08:26:16 +08:00
|
|
|
///
|
2010-08-21 17:40:31 +08:00
|
|
|
Decl *Parser::ParseObjCMethodDefinition() {
|
|
|
|
Decl *MDecl = ParseObjCMethodPrototype(ObjCImpDecl);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-08-27 07:41:50 +08:00
|
|
|
PrettyDeclStackTraceEntry CrashInfo(Actions, MDecl, Tok.getLocation(),
|
|
|
|
"parsing Objective-C method");
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-09-01 08:26:16 +08:00
|
|
|
// parse optional ';'
|
2009-10-21 00:39:13 +08:00
|
|
|
if (Tok.is(tok::semi)) {
|
2009-11-11 06:55:49 +08:00
|
|
|
if (ObjCImpDecl) {
|
|
|
|
Diag(Tok, diag::warn_semicolon_before_method_body)
|
2010-04-01 01:46:05 +08:00
|
|
|
<< FixItHint::CreateRemoval(Tok.getLocation());
|
2009-11-11 06:55:49 +08:00
|
|
|
}
|
2007-09-01 08:26:16 +08:00
|
|
|
ConsumeToken();
|
2009-10-21 00:39:13 +08:00
|
|
|
}
|
2007-09-01 08:26:16 +08:00
|
|
|
|
2007-11-12 03:54:21 +08:00
|
|
|
// We should have an opening brace now.
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::l_brace)) {
|
2008-03-01 05:48:07 +08:00
|
|
|
Diag(Tok, diag::err_expected_method_body);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-11-12 03:54:21 +08:00
|
|
|
// Skip over garbage, until we get to '{'. Don't eat the '{'.
|
|
|
|
SkipUntil(tok::l_brace, true, true);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-11-12 03:54:21 +08:00
|
|
|
// If we didn't find the '{', bail out.
|
|
|
|
if (Tok.isNot(tok::l_brace))
|
2010-08-21 17:40:31 +08:00
|
|
|
return 0;
|
2007-09-01 08:26:16 +08:00
|
|
|
}
|
2007-11-12 03:54:21 +08:00
|
|
|
SourceLocation BraceLoc = Tok.getLocation();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-11-12 03:54:21 +08:00
|
|
|
// Enter a scope for the method body.
|
2010-04-12 13:38:43 +08:00
|
|
|
ParseScope BodyScope(this,
|
|
|
|
Scope::ObjCMethodScope|Scope::FnScope|Scope::DeclScope);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-11-12 03:54:21 +08:00
|
|
|
// Tell the actions module that we have entered a method definition with the
|
2008-07-26 01:57:26 +08:00
|
|
|
// specified Declarator for the method.
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.ActOnStartOfObjCMethodDef(getCurScope(), MDecl);
|
2008-12-12 03:30:53 +08:00
|
|
|
|
2011-01-04 08:27:27 +08:00
|
|
|
if (PP.isCodeCompletionEnabled())
|
|
|
|
if (trySkippingFunctionBodyForCodeCompletion())
|
|
|
|
return Actions.ActOnFinishFunctionBody(MDecl, 0);
|
2011-01-04 06:33:06 +08:00
|
|
|
|
2010-08-24 14:29:42 +08:00
|
|
|
StmtResult FnBody(ParseCompoundStatementBody());
|
2008-12-12 03:30:53 +08:00
|
|
|
|
2007-11-12 03:54:21 +08:00
|
|
|
// If the function body could not be parsed, make a bogus compoundstmt.
|
2008-12-09 21:15:23 +08:00
|
|
|
if (FnBody.isInvalid())
|
2008-12-21 20:04:03 +08:00
|
|
|
FnBody = Actions.ActOnCompoundStmt(BraceLoc, BraceLoc,
|
|
|
|
MultiStmtArg(Actions), false);
|
2008-12-14 00:23:55 +08:00
|
|
|
|
2009-03-03 06:00:56 +08:00
|
|
|
// TODO: Pass argument information.
|
2010-08-24 07:25:46 +08:00
|
|
|
Actions.ActOnFinishFunctionBody(MDecl, FnBody.take());
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-11-12 03:54:21 +08:00
|
|
|
// Leave the function body scope.
|
2008-12-10 14:34:36 +08:00
|
|
|
BodyScope.Exit();
|
2008-12-14 00:23:55 +08:00
|
|
|
|
2007-11-14 07:01:27 +08:00
|
|
|
return MDecl;
|
2006-11-05 10:08:13 +08:00
|
|
|
}
|
2007-08-22 01:43:55 +08:00
|
|
|
|
2010-08-24 14:29:42 +08:00
|
|
|
StmtResult Parser::ParseObjCAtStatement(SourceLocation AtLoc) {
|
2009-12-07 17:51:25 +08:00
|
|
|
if (Tok.is(tok::code_completion)) {
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.CodeCompleteObjCAtStatement(getCurScope());
|
2010-05-25 13:58:43 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
2009-12-07 17:51:25 +08:00
|
|
|
return StmtError();
|
2009-12-08 00:33:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (Tok.isObjCAtKeyword(tok::objc_try))
|
2008-03-10 14:06:04 +08:00
|
|
|
return ParseObjCTryStmt(AtLoc);
|
2009-12-08 00:33:19 +08:00
|
|
|
|
|
|
|
if (Tok.isObjCAtKeyword(tok::objc_throw))
|
2008-02-06 05:27:35 +08:00
|
|
|
return ParseObjCThrowStmt(AtLoc);
|
2009-12-08 00:33:19 +08:00
|
|
|
|
|
|
|
if (Tok.isObjCAtKeyword(tok::objc_synchronized))
|
2008-02-06 05:27:35 +08:00
|
|
|
return ParseObjCSynchronizedStmt(AtLoc);
|
2009-12-08 00:33:19 +08:00
|
|
|
|
2010-08-24 14:29:42 +08:00
|
|
|
ExprResult Res(ParseExpressionWithLeadingAt(AtLoc));
|
2008-12-09 21:15:23 +08:00
|
|
|
if (Res.isInvalid()) {
|
2008-02-06 05:27:35 +08:00
|
|
|
// If the expression is invalid, skip ahead to the next semicolon. Not
|
|
|
|
// doing this opens us up to the possibility of infinite loops if
|
|
|
|
// ParseExpression does not consume any tokens.
|
|
|
|
SkipUntil(tok::semi);
|
2008-12-12 04:12:42 +08:00
|
|
|
return StmtError();
|
2008-02-06 05:27:35 +08:00
|
|
|
}
|
2009-12-08 00:33:19 +08:00
|
|
|
|
2008-02-06 05:27:35 +08:00
|
|
|
// Otherwise, eat the semicolon.
|
2010-09-07 23:23:11 +08:00
|
|
|
ExpectAndConsumeSemi(diag::err_expected_semi_after_expr);
|
2010-08-24 07:25:46 +08:00
|
|
|
return Actions.ActOnExprStmt(Actions.MakeFullExpr(Res.take()));
|
2008-02-06 05:27:35 +08:00
|
|
|
}
|
|
|
|
|
2010-08-24 14:29:42 +08:00
|
|
|
ExprResult Parser::ParseObjCAtExpression(SourceLocation AtLoc) {
|
2007-08-22 01:43:55 +08:00
|
|
|
switch (Tok.getKind()) {
|
2009-12-07 17:51:25 +08:00
|
|
|
case tok::code_completion:
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.CodeCompleteObjCAtExpression(getCurScope());
|
2010-05-25 13:58:43 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
2009-12-07 17:51:25 +08:00
|
|
|
return ExprError();
|
|
|
|
|
2007-12-12 09:04:12 +08:00
|
|
|
case tok::string_literal: // primary-expression: string-literal
|
|
|
|
case tok::wide_string_literal:
|
2008-12-13 23:32:12 +08:00
|
|
|
return ParsePostfixExpressionSuffix(ParseObjCStringLiteral(AtLoc));
|
2007-12-12 09:04:12 +08:00
|
|
|
default:
|
2008-08-05 14:19:09 +08:00
|
|
|
if (Tok.getIdentifierInfo() == 0)
|
2008-12-13 23:32:12 +08:00
|
|
|
return ExprError(Diag(AtLoc, diag::err_unexpected_at));
|
2008-12-12 05:36:32 +08:00
|
|
|
|
2008-08-05 14:19:09 +08:00
|
|
|
switch (Tok.getIdentifierInfo()->getObjCKeywordID()) {
|
|
|
|
case tok::objc_encode:
|
2008-12-13 23:32:12 +08:00
|
|
|
return ParsePostfixExpressionSuffix(ParseObjCEncodeExpression(AtLoc));
|
2008-08-05 14:19:09 +08:00
|
|
|
case tok::objc_protocol:
|
2008-12-13 23:32:12 +08:00
|
|
|
return ParsePostfixExpressionSuffix(ParseObjCProtocolExpression(AtLoc));
|
2008-08-05 14:19:09 +08:00
|
|
|
case tok::objc_selector:
|
2008-12-13 23:32:12 +08:00
|
|
|
return ParsePostfixExpressionSuffix(ParseObjCSelectorExpression(AtLoc));
|
2008-08-05 14:19:09 +08:00
|
|
|
default:
|
2008-12-13 23:32:12 +08:00
|
|
|
return ExprError(Diag(AtLoc, diag::err_unexpected_at));
|
2008-08-05 14:19:09 +08:00
|
|
|
}
|
2007-08-22 01:43:55 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-04-22 06:36:40 +08:00
|
|
|
/// \brirg Parse the receiver of an Objective-C++ message send.
|
|
|
|
///
|
|
|
|
/// This routine parses the receiver of a message send in
|
|
|
|
/// Objective-C++ either as a type or as an expression. Note that this
|
|
|
|
/// routine must not be called to parse a send to 'super', since it
|
|
|
|
/// has no way to return such a result.
|
|
|
|
///
|
|
|
|
/// \param IsExpr Whether the receiver was parsed as an expression.
|
|
|
|
///
|
|
|
|
/// \param TypeOrExpr If the receiver was parsed as an expression (\c
|
|
|
|
/// IsExpr is true), the parsed expression. If the receiver was parsed
|
|
|
|
/// as a type (\c IsExpr is false), the parsed type.
|
|
|
|
///
|
|
|
|
/// \returns True if an error occurred during parsing or semantic
|
|
|
|
/// analysis, in which case the arguments do not have valid
|
|
|
|
/// values. Otherwise, returns false for a successful parse.
|
|
|
|
///
|
|
|
|
/// objc-receiver: [C++]
|
|
|
|
/// 'super' [not parsed here]
|
|
|
|
/// expression
|
|
|
|
/// simple-type-specifier
|
|
|
|
/// typename-specifier
|
|
|
|
bool Parser::ParseObjCXXMessageReceiver(bool &IsExpr, void *&TypeOrExpr) {
|
2010-09-15 22:51:05 +08:00
|
|
|
InMessageExpressionRAIIObject InMessage(*this, true);
|
|
|
|
|
2010-04-22 06:36:40 +08:00
|
|
|
if (Tok.is(tok::identifier) || Tok.is(tok::coloncolon) ||
|
|
|
|
Tok.is(tok::kw_typename) || Tok.is(tok::annot_cxxscope))
|
|
|
|
TryAnnotateTypeOrScopeToken();
|
|
|
|
|
|
|
|
if (!isCXXSimpleTypeSpecifier()) {
|
|
|
|
// objc-receiver:
|
|
|
|
// expression
|
2010-08-24 14:29:42 +08:00
|
|
|
ExprResult Receiver = ParseExpression();
|
2010-04-22 06:36:40 +08:00
|
|
|
if (Receiver.isInvalid())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
IsExpr = true;
|
|
|
|
TypeOrExpr = Receiver.take();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// objc-receiver:
|
|
|
|
// typename-specifier
|
|
|
|
// simple-type-specifier
|
|
|
|
// expression (that starts with one of the above)
|
|
|
|
DeclSpec DS;
|
|
|
|
ParseCXXSimpleTypeSpecifier(DS);
|
|
|
|
|
|
|
|
if (Tok.is(tok::l_paren)) {
|
|
|
|
// If we see an opening parentheses at this point, we are
|
|
|
|
// actually parsing an expression that starts with a
|
|
|
|
// function-style cast, e.g.,
|
|
|
|
//
|
|
|
|
// postfix-expression:
|
|
|
|
// simple-type-specifier ( expression-list [opt] )
|
|
|
|
// typename-specifier ( expression-list [opt] )
|
|
|
|
//
|
|
|
|
// Parse the remainder of this case, then the (optional)
|
|
|
|
// postfix-expression suffix, followed by the (optional)
|
|
|
|
// right-hand side of the binary expression. We have an
|
|
|
|
// instance method.
|
2010-08-24 14:29:42 +08:00
|
|
|
ExprResult Receiver = ParseCXXTypeConstructExpression(DS);
|
2010-04-22 06:36:40 +08:00
|
|
|
if (!Receiver.isInvalid())
|
2010-08-24 07:25:46 +08:00
|
|
|
Receiver = ParsePostfixExpressionSuffix(Receiver.take());
|
2010-04-22 06:36:40 +08:00
|
|
|
if (!Receiver.isInvalid())
|
2010-08-24 07:25:46 +08:00
|
|
|
Receiver = ParseRHSOfBinaryExpression(Receiver.take(), prec::Comma);
|
2010-04-22 06:36:40 +08:00
|
|
|
if (Receiver.isInvalid())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
IsExpr = true;
|
|
|
|
TypeOrExpr = Receiver.take();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We have a class message. Turn the simple-type-specifier or
|
|
|
|
// typename-specifier we parsed into a type and parse the
|
|
|
|
// remainder of the class message.
|
|
|
|
Declarator DeclaratorInfo(DS, Declarator::TypeNameContext);
|
2010-07-03 01:43:08 +08:00
|
|
|
TypeResult Type = Actions.ActOnTypeName(getCurScope(), DeclaratorInfo);
|
2010-04-22 06:36:40 +08:00
|
|
|
if (Type.isInvalid())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
IsExpr = false;
|
2010-08-24 13:47:05 +08:00
|
|
|
TypeOrExpr = Type.get().getAsOpaquePtr();
|
2010-04-22 06:36:40 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2010-05-31 22:40:22 +08:00
|
|
|
/// \brief Determine whether the parser is currently referring to a an
|
|
|
|
/// Objective-C message send, using a simplified heuristic to avoid overhead.
|
|
|
|
///
|
|
|
|
/// This routine will only return true for a subset of valid message-send
|
|
|
|
/// expressions.
|
|
|
|
bool Parser::isSimpleObjCMessageExpression() {
|
2010-06-01 02:18:22 +08:00
|
|
|
assert(Tok.is(tok::l_square) && getLang().ObjC1 &&
|
2010-05-31 22:40:22 +08:00
|
|
|
"Incorrect start for isSimpleObjCMessageExpression");
|
|
|
|
return GetLookAheadToken(1).is(tok::identifier) &&
|
|
|
|
GetLookAheadToken(2).is(tok::identifier);
|
|
|
|
}
|
|
|
|
|
2010-09-16 09:51:54 +08:00
|
|
|
bool Parser::isStartOfObjCClassMessageMissingOpenBracket() {
|
|
|
|
if (!getLang().ObjC1 || !NextToken().is(tok::identifier) ||
|
|
|
|
InMessageExpression)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
|
|
ParsedType Type;
|
|
|
|
|
|
|
|
if (Tok.is(tok::annot_typename))
|
|
|
|
Type = getTypeAnnotation(Tok);
|
|
|
|
else if (Tok.is(tok::identifier))
|
|
|
|
Type = Actions.getTypeName(*Tok.getIdentifierInfo(), Tok.getLocation(),
|
|
|
|
getCurScope());
|
|
|
|
else
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (!Type.get().isNull() && Type.get()->isObjCObjectOrInterfaceType()) {
|
|
|
|
const Token &AfterNext = GetLookAheadToken(2);
|
|
|
|
if (AfterNext.is(tok::colon) || AfterNext.is(tok::r_square)) {
|
|
|
|
if (Tok.is(tok::identifier))
|
|
|
|
TryAnnotateTypeOrScopeToken();
|
|
|
|
|
|
|
|
return Tok.is(tok::annot_typename);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
/// objc-message-expr:
|
2007-09-06 03:52:07 +08:00
|
|
|
/// '[' objc-receiver objc-message-args ']'
|
|
|
|
///
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
/// objc-receiver: [C]
|
2010-04-11 16:28:14 +08:00
|
|
|
/// 'super'
|
2007-09-06 03:52:07 +08:00
|
|
|
/// expression
|
|
|
|
/// class-name
|
|
|
|
/// type-name
|
2010-04-22 06:36:40 +08:00
|
|
|
///
|
2010-08-24 14:29:42 +08:00
|
|
|
ExprResult Parser::ParseObjCMessageExpression() {
|
2008-01-26 02:59:06 +08:00
|
|
|
assert(Tok.is(tok::l_square) && "'[' expected");
|
|
|
|
SourceLocation LBracLoc = ConsumeBracket(); // consume '['
|
|
|
|
|
2010-05-28 07:06:34 +08:00
|
|
|
if (Tok.is(tok::code_completion)) {
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.CodeCompleteObjCMessageReceiver(getCurScope());
|
2010-05-28 07:06:34 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
|
|
|
SkipUntil(tok::r_square);
|
|
|
|
return ExprError();
|
|
|
|
}
|
|
|
|
|
2010-09-15 22:51:05 +08:00
|
|
|
InMessageExpressionRAIIObject InMessage(*this, true);
|
|
|
|
|
2010-04-22 06:36:40 +08:00
|
|
|
if (getLang().CPlusPlus) {
|
|
|
|
// We completely separate the C and C++ cases because C++ requires
|
|
|
|
// more complicated (read: slower) parsing.
|
|
|
|
|
|
|
|
// Handle send to super.
|
|
|
|
// FIXME: This doesn't benefit from the same typo-correction we
|
|
|
|
// get in Objective-C.
|
|
|
|
if (Tok.is(tok::identifier) && Tok.getIdentifierInfo() == Ident_super &&
|
2010-07-03 01:43:08 +08:00
|
|
|
NextToken().isNot(tok::period) && getCurScope()->isInObjcMethodScope())
|
2010-08-24 13:47:05 +08:00
|
|
|
return ParseObjCMessageExpressionBody(LBracLoc, ConsumeToken(),
|
|
|
|
ParsedType(), 0);
|
2010-04-22 06:36:40 +08:00
|
|
|
|
|
|
|
// Parse the receiver, which is either a type or an expression.
|
|
|
|
bool IsExpr;
|
2010-09-16 02:35:19 +08:00
|
|
|
void *TypeOrExpr = NULL;
|
2010-04-22 06:36:40 +08:00
|
|
|
if (ParseObjCXXMessageReceiver(IsExpr, TypeOrExpr)) {
|
|
|
|
SkipUntil(tok::r_square);
|
|
|
|
return ExprError();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (IsExpr)
|
2010-08-24 13:47:05 +08:00
|
|
|
return ParseObjCMessageExpressionBody(LBracLoc, SourceLocation(),
|
|
|
|
ParsedType(),
|
2010-08-24 07:25:46 +08:00
|
|
|
static_cast<Expr*>(TypeOrExpr));
|
2010-04-22 06:36:40 +08:00
|
|
|
|
|
|
|
return ParseObjCMessageExpressionBody(LBracLoc, SourceLocation(),
|
2010-08-24 13:47:05 +08:00
|
|
|
ParsedType::getFromOpaquePtr(TypeOrExpr),
|
|
|
|
0);
|
2010-06-01 02:18:22 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (Tok.is(tok::identifier)) {
|
2010-04-14 10:22:16 +08:00
|
|
|
IdentifierInfo *Name = Tok.getIdentifierInfo();
|
|
|
|
SourceLocation NameLoc = Tok.getLocation();
|
2010-08-24 13:47:05 +08:00
|
|
|
ParsedType ReceiverType;
|
2010-07-03 01:43:08 +08:00
|
|
|
switch (Actions.getObjCMessageKind(getCurScope(), Name, NameLoc,
|
2010-04-14 10:22:16 +08:00
|
|
|
Name == Ident_super,
|
2010-04-22 04:38:13 +08:00
|
|
|
NextToken().is(tok::period),
|
|
|
|
ReceiverType)) {
|
2010-08-27 07:41:50 +08:00
|
|
|
case Sema::ObjCSuperMessage:
|
2010-08-24 13:47:05 +08:00
|
|
|
return ParseObjCMessageExpressionBody(LBracLoc, ConsumeToken(),
|
|
|
|
ParsedType(), 0);
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
|
2010-08-27 07:41:50 +08:00
|
|
|
case Sema::ObjCClassMessage:
|
2010-04-22 04:38:13 +08:00
|
|
|
if (!ReceiverType) {
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
SkipUntil(tok::r_square);
|
|
|
|
return ExprError();
|
|
|
|
}
|
|
|
|
|
2010-04-22 04:38:13 +08:00
|
|
|
ConsumeToken(); // the type name
|
|
|
|
|
|
|
|
return ParseObjCMessageExpressionBody(LBracLoc, SourceLocation(),
|
2010-08-24 07:25:46 +08:00
|
|
|
ReceiverType, 0);
|
2010-04-14 10:22:16 +08:00
|
|
|
|
2010-08-27 07:41:50 +08:00
|
|
|
case Sema::ObjCInstanceMessage:
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
// Fall through to parse an expression.
|
2010-04-14 10:22:16 +08:00
|
|
|
break;
|
2009-04-09 03:50:10 +08:00
|
|
|
}
|
2008-01-26 02:59:06 +08:00
|
|
|
}
|
2010-04-11 16:28:14 +08:00
|
|
|
|
|
|
|
// Otherwise, an arbitrary expression can be the receiver of a send.
|
2010-08-24 14:29:42 +08:00
|
|
|
ExprResult Res(ParseExpression());
|
2008-12-09 21:15:23 +08:00
|
|
|
if (Res.isInvalid()) {
|
2008-01-26 03:43:26 +08:00
|
|
|
SkipUntil(tok::r_square);
|
2008-12-13 23:32:12 +08:00
|
|
|
return move(Res);
|
2008-01-26 02:59:06 +08:00
|
|
|
}
|
2008-12-13 23:32:12 +08:00
|
|
|
|
2010-08-24 13:47:05 +08:00
|
|
|
return ParseObjCMessageExpressionBody(LBracLoc, SourceLocation(),
|
|
|
|
ParsedType(), Res.take());
|
2008-01-26 02:59:06 +08:00
|
|
|
}
|
2008-12-13 23:32:12 +08:00
|
|
|
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
/// \brief Parse the remainder of an Objective-C message following the
|
|
|
|
/// '[' objc-receiver.
|
|
|
|
///
|
|
|
|
/// This routine handles sends to super, class messages (sent to a
|
|
|
|
/// class name), and instance messages (sent to an object), and the
|
|
|
|
/// target is represented by \p SuperLoc, \p ReceiverType, or \p
|
|
|
|
/// ReceiverExpr, respectively. Only one of these parameters may have
|
|
|
|
/// a valid value.
|
|
|
|
///
|
|
|
|
/// \param LBracLoc The location of the opening '['.
|
|
|
|
///
|
|
|
|
/// \param SuperLoc If this is a send to 'super', the location of the
|
|
|
|
/// 'super' keyword that indicates a send to the superclass.
|
|
|
|
///
|
|
|
|
/// \param ReceiverType If this is a class message, the type of the
|
|
|
|
/// class we are sending a message to.
|
|
|
|
///
|
|
|
|
/// \param ReceiverExpr If this is an instance message, the expression
|
|
|
|
/// used to compute the receiver object.
|
2009-09-09 23:08:12 +08:00
|
|
|
///
|
2007-09-06 03:52:07 +08:00
|
|
|
/// objc-message-args:
|
|
|
|
/// objc-selector
|
|
|
|
/// objc-keywordarg-list
|
|
|
|
///
|
|
|
|
/// objc-keywordarg-list:
|
|
|
|
/// objc-keywordarg
|
|
|
|
/// objc-keywordarg-list objc-keywordarg
|
|
|
|
///
|
2009-09-09 23:08:12 +08:00
|
|
|
/// objc-keywordarg:
|
2007-09-06 03:52:07 +08:00
|
|
|
/// selector-name[opt] ':' objc-keywordexpr
|
|
|
|
///
|
|
|
|
/// objc-keywordexpr:
|
|
|
|
/// nonempty-expr-list
|
|
|
|
///
|
|
|
|
/// nonempty-expr-list:
|
|
|
|
/// assignment-expression
|
|
|
|
/// nonempty-expr-list , assignment-expression
|
2008-12-13 23:32:12 +08:00
|
|
|
///
|
2010-08-24 14:29:42 +08:00
|
|
|
ExprResult
|
2008-01-26 02:59:06 +08:00
|
|
|
Parser::ParseObjCMessageExpressionBody(SourceLocation LBracLoc,
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
SourceLocation SuperLoc,
|
2010-08-24 13:47:05 +08:00
|
|
|
ParsedType ReceiverType,
|
2008-12-13 23:32:12 +08:00
|
|
|
ExprArg ReceiverExpr) {
|
2010-09-15 22:51:05 +08:00
|
|
|
InMessageExpressionRAIIObject InMessage(*this, true);
|
|
|
|
|
2009-11-07 10:08:14 +08:00
|
|
|
if (Tok.is(tok::code_completion)) {
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
if (SuperLoc.isValid())
|
2010-09-21 07:34:21 +08:00
|
|
|
Actions.CodeCompleteObjCSuperMessage(getCurScope(), SuperLoc, 0, 0,
|
|
|
|
false);
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
else if (ReceiverType)
|
2010-09-21 07:34:21 +08:00
|
|
|
Actions.CodeCompleteObjCClassMessage(getCurScope(), ReceiverType, 0, 0,
|
|
|
|
false);
|
2009-11-07 10:08:14 +08:00
|
|
|
else
|
2010-08-24 07:25:46 +08:00
|
|
|
Actions.CodeCompleteObjCInstanceMessage(getCurScope(), ReceiverExpr,
|
2010-09-21 07:34:21 +08:00
|
|
|
0, 0, false);
|
2010-05-25 13:58:43 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
2009-11-07 10:08:14 +08:00
|
|
|
}
|
2009-11-19 09:08:35 +08:00
|
|
|
|
2007-09-06 07:08:20 +08:00
|
|
|
// Parse objc-selector
|
2007-10-11 08:55:41 +08:00
|
|
|
SourceLocation Loc;
|
2009-04-12 02:13:45 +08:00
|
|
|
IdentifierInfo *selIdent = ParseObjCSelectorPiece(Loc);
|
Add SelectorInfo (similar in spirit to IdentifierInfo). The key difference is SelectorInfo is not string-oriented, it is a unique aggregate of IdentifierInfo's (using a folding set). SelectorInfo also has a richer API that simplifies the parser/action interface. 3 noteworthy benefits:
#1: It is cleaner. I never "liked" storing keyword selectors (i.e. foo:bar:baz) in the IdentifierTable.
#2: It is more space efficient. Since Cocoa keyword selectors can be quite long, this technique is space saving. For Cocoa.h, pulling the keyword selectors out saves ~180k. The cost of the SelectorInfo data is ~100k. Saves ~80k, or 43%.
#3: It results in many API simplifications. Here are some highlights:
- Removed 3 actions (ActOnKeywordMessage, ActOnUnaryMessage, & one flavor of ObjcBuildMethodDeclaration that was specific to unary messages).
- Removed 3 funky structs from DeclSpec.h (ObjcKeywordMessage, ObjcKeywordDecl, and ObjcKeywordInfo).
- Removed 2 ivars and 2 constructors from ObjCMessageExpr (fyi, this space savings has not been measured).
I am happy with the way it turned out (though it took a bit more hacking than I expected). Given the central role of selectors in ObjC, making sure this is "right" will pay dividends later.
Thanks to Chris for talking this through with me and suggesting this approach.
llvm-svn: 42395
2007-09-27 22:38:14 +08:00
|
|
|
|
2009-02-15 02:21:46 +08:00
|
|
|
SourceLocation SelectorLoc = Loc;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
Add SelectorInfo (similar in spirit to IdentifierInfo). The key difference is SelectorInfo is not string-oriented, it is a unique aggregate of IdentifierInfo's (using a folding set). SelectorInfo also has a richer API that simplifies the parser/action interface. 3 noteworthy benefits:
#1: It is cleaner. I never "liked" storing keyword selectors (i.e. foo:bar:baz) in the IdentifierTable.
#2: It is more space efficient. Since Cocoa keyword selectors can be quite long, this technique is space saving. For Cocoa.h, pulling the keyword selectors out saves ~180k. The cost of the SelectorInfo data is ~100k. Saves ~80k, or 43%.
#3: It results in many API simplifications. Here are some highlights:
- Removed 3 actions (ActOnKeywordMessage, ActOnUnaryMessage, & one flavor of ObjcBuildMethodDeclaration that was specific to unary messages).
- Removed 3 funky structs from DeclSpec.h (ObjcKeywordMessage, ObjcKeywordDecl, and ObjcKeywordInfo).
- Removed 2 ivars and 2 constructors from ObjCMessageExpr (fyi, this space savings has not been measured).
I am happy with the way it turned out (though it took a bit more hacking than I expected). Given the central role of selectors in ObjC, making sure this is "right" will pay dividends later.
Thanks to Chris for talking this through with me and suggesting this approach.
llvm-svn: 42395
2007-09-27 22:38:14 +08:00
|
|
|
llvm::SmallVector<IdentifierInfo *, 12> KeyIdents;
|
2008-11-26 06:21:31 +08:00
|
|
|
ExprVector KeyExprs(Actions);
|
Add SelectorInfo (similar in spirit to IdentifierInfo). The key difference is SelectorInfo is not string-oriented, it is a unique aggregate of IdentifierInfo's (using a folding set). SelectorInfo also has a richer API that simplifies the parser/action interface. 3 noteworthy benefits:
#1: It is cleaner. I never "liked" storing keyword selectors (i.e. foo:bar:baz) in the IdentifierTable.
#2: It is more space efficient. Since Cocoa keyword selectors can be quite long, this technique is space saving. For Cocoa.h, pulling the keyword selectors out saves ~180k. The cost of the SelectorInfo data is ~100k. Saves ~80k, or 43%.
#3: It results in many API simplifications. Here are some highlights:
- Removed 3 actions (ActOnKeywordMessage, ActOnUnaryMessage, & one flavor of ObjcBuildMethodDeclaration that was specific to unary messages).
- Removed 3 funky structs from DeclSpec.h (ObjcKeywordMessage, ObjcKeywordDecl, and ObjcKeywordInfo).
- Removed 2 ivars and 2 constructors from ObjCMessageExpr (fyi, this space savings has not been measured).
I am happy with the way it turned out (though it took a bit more hacking than I expected). Given the central role of selectors in ObjC, making sure this is "right" will pay dividends later.
Thanks to Chris for talking this through with me and suggesting this approach.
llvm-svn: 42395
2007-09-27 22:38:14 +08:00
|
|
|
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.is(tok::colon)) {
|
2007-09-06 07:08:20 +08:00
|
|
|
while (1) {
|
|
|
|
// Each iteration parses a single keyword argument.
|
Add SelectorInfo (similar in spirit to IdentifierInfo). The key difference is SelectorInfo is not string-oriented, it is a unique aggregate of IdentifierInfo's (using a folding set). SelectorInfo also has a richer API that simplifies the parser/action interface. 3 noteworthy benefits:
#1: It is cleaner. I never "liked" storing keyword selectors (i.e. foo:bar:baz) in the IdentifierTable.
#2: It is more space efficient. Since Cocoa keyword selectors can be quite long, this technique is space saving. For Cocoa.h, pulling the keyword selectors out saves ~180k. The cost of the SelectorInfo data is ~100k. Saves ~80k, or 43%.
#3: It results in many API simplifications. Here are some highlights:
- Removed 3 actions (ActOnKeywordMessage, ActOnUnaryMessage, & one flavor of ObjcBuildMethodDeclaration that was specific to unary messages).
- Removed 3 funky structs from DeclSpec.h (ObjcKeywordMessage, ObjcKeywordDecl, and ObjcKeywordInfo).
- Removed 2 ivars and 2 constructors from ObjCMessageExpr (fyi, this space savings has not been measured).
I am happy with the way it turned out (though it took a bit more hacking than I expected). Given the central role of selectors in ObjC, making sure this is "right" will pay dividends later.
Thanks to Chris for talking this through with me and suggesting this approach.
llvm-svn: 42395
2007-09-27 22:38:14 +08:00
|
|
|
KeyIdents.push_back(selIdent);
|
2007-09-18 04:25:27 +08:00
|
|
|
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::colon)) {
|
2007-09-06 07:08:20 +08:00
|
|
|
Diag(Tok, diag::err_expected_colon);
|
2008-08-05 14:19:09 +08:00
|
|
|
// We must manually skip to a ']', otherwise the expression skipper will
|
|
|
|
// stop at the ']' when it skips to the ';'. We want it to skip beyond
|
|
|
|
// the enclosing expression.
|
|
|
|
SkipUntil(tok::r_square);
|
2008-12-13 23:32:12 +08:00
|
|
|
return ExprError();
|
2007-09-06 07:08:20 +08:00
|
|
|
}
|
2008-12-13 23:32:12 +08:00
|
|
|
|
Add SelectorInfo (similar in spirit to IdentifierInfo). The key difference is SelectorInfo is not string-oriented, it is a unique aggregate of IdentifierInfo's (using a folding set). SelectorInfo also has a richer API that simplifies the parser/action interface. 3 noteworthy benefits:
#1: It is cleaner. I never "liked" storing keyword selectors (i.e. foo:bar:baz) in the IdentifierTable.
#2: It is more space efficient. Since Cocoa keyword selectors can be quite long, this technique is space saving. For Cocoa.h, pulling the keyword selectors out saves ~180k. The cost of the SelectorInfo data is ~100k. Saves ~80k, or 43%.
#3: It results in many API simplifications. Here are some highlights:
- Removed 3 actions (ActOnKeywordMessage, ActOnUnaryMessage, & one flavor of ObjcBuildMethodDeclaration that was specific to unary messages).
- Removed 3 funky structs from DeclSpec.h (ObjcKeywordMessage, ObjcKeywordDecl, and ObjcKeywordInfo).
- Removed 2 ivars and 2 constructors from ObjCMessageExpr (fyi, this space savings has not been measured).
I am happy with the way it turned out (though it took a bit more hacking than I expected). Given the central role of selectors in ObjC, making sure this is "right" will pay dividends later.
Thanks to Chris for talking this through with me and suggesting this approach.
llvm-svn: 42395
2007-09-27 22:38:14 +08:00
|
|
|
ConsumeToken(); // Eat the ':'.
|
2009-09-09 23:08:12 +08:00
|
|
|
/// Parse the expression after ':'
|
2010-09-21 07:34:21 +08:00
|
|
|
|
|
|
|
if (Tok.is(tok::code_completion)) {
|
|
|
|
if (SuperLoc.isValid())
|
|
|
|
Actions.CodeCompleteObjCSuperMessage(getCurScope(), SuperLoc,
|
|
|
|
KeyIdents.data(),
|
|
|
|
KeyIdents.size(),
|
|
|
|
/*AtArgumentEpression=*/true);
|
|
|
|
else if (ReceiverType)
|
|
|
|
Actions.CodeCompleteObjCClassMessage(getCurScope(), ReceiverType,
|
|
|
|
KeyIdents.data(),
|
|
|
|
KeyIdents.size(),
|
|
|
|
/*AtArgumentEpression=*/true);
|
|
|
|
else
|
|
|
|
Actions.CodeCompleteObjCInstanceMessage(getCurScope(), ReceiverExpr,
|
|
|
|
KeyIdents.data(),
|
|
|
|
KeyIdents.size(),
|
|
|
|
/*AtArgumentEpression=*/true);
|
|
|
|
|
|
|
|
ConsumeCodeCompletionToken();
|
|
|
|
SkipUntil(tok::r_square);
|
|
|
|
return ExprError();
|
|
|
|
}
|
|
|
|
|
2010-08-24 14:29:42 +08:00
|
|
|
ExprResult Res(ParseAssignmentExpression());
|
2008-12-09 21:15:23 +08:00
|
|
|
if (Res.isInvalid()) {
|
2008-08-05 14:19:09 +08:00
|
|
|
// We must manually skip to a ']', otherwise the expression skipper will
|
|
|
|
// stop at the ']' when it skips to the ';'. We want it to skip beyond
|
|
|
|
// the enclosing expression.
|
|
|
|
SkipUntil(tok::r_square);
|
2008-12-13 23:32:12 +08:00
|
|
|
return move(Res);
|
2007-09-18 04:25:27 +08:00
|
|
|
}
|
2008-12-13 23:32:12 +08:00
|
|
|
|
2007-09-18 04:25:27 +08:00
|
|
|
// We have a valid expression.
|
2008-12-10 08:02:53 +08:00
|
|
|
KeyExprs.push_back(Res.release());
|
2008-12-13 23:32:12 +08:00
|
|
|
|
2009-11-19 09:08:35 +08:00
|
|
|
// Code completion after each argument.
|
|
|
|
if (Tok.is(tok::code_completion)) {
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
if (SuperLoc.isValid())
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.CodeCompleteObjCSuperMessage(getCurScope(), SuperLoc,
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
KeyIdents.data(),
|
2010-09-21 07:34:21 +08:00
|
|
|
KeyIdents.size(),
|
|
|
|
/*AtArgumentEpression=*/false);
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
else if (ReceiverType)
|
2010-07-03 01:43:08 +08:00
|
|
|
Actions.CodeCompleteObjCClassMessage(getCurScope(), ReceiverType,
|
2009-11-19 09:08:35 +08:00
|
|
|
KeyIdents.data(),
|
2010-09-21 07:34:21 +08:00
|
|
|
KeyIdents.size(),
|
|
|
|
/*AtArgumentEpression=*/false);
|
2009-11-19 09:08:35 +08:00
|
|
|
else
|
2010-08-24 07:25:46 +08:00
|
|
|
Actions.CodeCompleteObjCInstanceMessage(getCurScope(), ReceiverExpr,
|
2009-11-19 09:08:35 +08:00
|
|
|
KeyIdents.data(),
|
2010-09-21 07:34:21 +08:00
|
|
|
KeyIdents.size(),
|
|
|
|
/*AtArgumentEpression=*/false);
|
2010-05-25 13:58:43 +08:00
|
|
|
ConsumeCodeCompletionToken();
|
2010-09-21 07:34:21 +08:00
|
|
|
SkipUntil(tok::r_square);
|
|
|
|
return ExprError();
|
2009-11-19 09:08:35 +08:00
|
|
|
}
|
|
|
|
|
2007-09-18 04:25:27 +08:00
|
|
|
// Check for another keyword selector.
|
2009-04-12 02:13:45 +08:00
|
|
|
selIdent = ParseObjCSelectorPiece(Loc);
|
2007-10-10 01:51:17 +08:00
|
|
|
if (!selIdent && Tok.isNot(tok::colon))
|
2007-09-06 07:08:20 +08:00
|
|
|
break;
|
|
|
|
// We have a selector or a colon, continue parsing.
|
|
|
|
}
|
|
|
|
// Parse the, optional, argument list, comma separated.
|
2007-10-10 01:51:17 +08:00
|
|
|
while (Tok.is(tok::comma)) {
|
2007-11-15 21:05:42 +08:00
|
|
|
ConsumeToken(); // Eat the ','.
|
2009-09-09 23:08:12 +08:00
|
|
|
/// Parse the expression after ','
|
2010-08-24 14:29:42 +08:00
|
|
|
ExprResult Res(ParseAssignmentExpression());
|
2008-12-09 21:15:23 +08:00
|
|
|
if (Res.isInvalid()) {
|
2008-08-05 14:19:09 +08:00
|
|
|
// We must manually skip to a ']', otherwise the expression skipper will
|
|
|
|
// stop at the ']' when it skips to the ';'. We want it to skip beyond
|
|
|
|
// the enclosing expression.
|
|
|
|
SkipUntil(tok::r_square);
|
2008-12-13 23:32:12 +08:00
|
|
|
return move(Res);
|
2007-11-15 21:05:42 +08:00
|
|
|
}
|
2008-12-09 21:15:23 +08:00
|
|
|
|
2007-11-15 21:05:42 +08:00
|
|
|
// We have a valid expression.
|
2008-12-10 08:02:53 +08:00
|
|
|
KeyExprs.push_back(Res.release());
|
2007-09-06 07:08:20 +08:00
|
|
|
}
|
|
|
|
} else if (!selIdent) {
|
|
|
|
Diag(Tok, diag::err_expected_ident); // missing selector name.
|
2008-12-13 23:32:12 +08:00
|
|
|
|
2008-08-05 14:19:09 +08:00
|
|
|
// We must manually skip to a ']', otherwise the expression skipper will
|
|
|
|
// stop at the ']' when it skips to the ';'. We want it to skip beyond
|
|
|
|
// the enclosing expression.
|
|
|
|
SkipUntil(tok::r_square);
|
2008-12-13 23:32:12 +08:00
|
|
|
return ExprError();
|
2007-09-06 07:08:20 +08:00
|
|
|
}
|
2010-04-01 04:22:35 +08:00
|
|
|
|
2007-10-10 01:51:17 +08:00
|
|
|
if (Tok.isNot(tok::r_square)) {
|
2010-04-01 04:22:35 +08:00
|
|
|
if (Tok.is(tok::identifier))
|
|
|
|
Diag(Tok, diag::err_expected_colon);
|
|
|
|
else
|
|
|
|
Diag(Tok, diag::err_expected_rsquare);
|
2008-08-05 14:19:09 +08:00
|
|
|
// We must manually skip to a ']', otherwise the expression skipper will
|
|
|
|
// stop at the ']' when it skips to the ';'. We want it to skip beyond
|
|
|
|
// the enclosing expression.
|
|
|
|
SkipUntil(tok::r_square);
|
2008-12-13 23:32:12 +08:00
|
|
|
return ExprError();
|
2007-09-06 07:08:20 +08:00
|
|
|
}
|
2008-12-13 23:32:12 +08:00
|
|
|
|
2008-01-26 02:59:06 +08:00
|
|
|
SourceLocation RBracLoc = ConsumeBracket(); // consume ']'
|
2008-12-13 23:32:12 +08:00
|
|
|
|
2007-10-06 02:42:47 +08:00
|
|
|
unsigned nKeys = KeyIdents.size();
|
2007-10-07 10:00:24 +08:00
|
|
|
if (nKeys == 0)
|
|
|
|
KeyIdents.push_back(selIdent);
|
|
|
|
Selector Sel = PP.getSelectorTable().getSelector(nKeys, &KeyIdents[0]);
|
2008-12-13 23:32:12 +08:00
|
|
|
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
if (SuperLoc.isValid())
|
2010-07-03 01:43:08 +08:00
|
|
|
return Actions.ActOnSuperMessage(getCurScope(), SuperLoc, Sel,
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
LBracLoc, SelectorLoc, RBracLoc,
|
2010-08-27 07:41:50 +08:00
|
|
|
MultiExprArg(Actions,
|
|
|
|
KeyExprs.take(),
|
|
|
|
KeyExprs.size()));
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
else if (ReceiverType)
|
2010-07-03 01:43:08 +08:00
|
|
|
return Actions.ActOnClassMessage(getCurScope(), ReceiverType, Sel,
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
LBracLoc, SelectorLoc, RBracLoc,
|
2010-08-27 07:41:50 +08:00
|
|
|
MultiExprArg(Actions,
|
|
|
|
KeyExprs.take(),
|
|
|
|
KeyExprs.size()));
|
2010-08-24 07:25:46 +08:00
|
|
|
return Actions.ActOnInstanceMessage(getCurScope(), ReceiverExpr, Sel,
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
LBracLoc, SelectorLoc, RBracLoc,
|
2010-08-27 07:41:50 +08:00
|
|
|
MultiExprArg(Actions,
|
|
|
|
KeyExprs.take(),
|
|
|
|
KeyExprs.size()));
|
2007-09-06 03:52:07 +08:00
|
|
|
}
|
|
|
|
|
2010-08-24 14:29:42 +08:00
|
|
|
ExprResult Parser::ParseObjCStringLiteral(SourceLocation AtLoc) {
|
|
|
|
ExprResult Res(ParseStringLiteralExpression());
|
2008-12-13 23:32:12 +08:00
|
|
|
if (Res.isInvalid()) return move(Res);
|
|
|
|
|
2007-12-12 09:04:12 +08:00
|
|
|
// @"foo" @"bar" is a valid concatenated string. Eat any subsequent string
|
|
|
|
// expressions. At this point, we know that the only valid thing that starts
|
|
|
|
// with '@' is an @"".
|
|
|
|
llvm::SmallVector<SourceLocation, 4> AtLocs;
|
2008-11-26 06:21:31 +08:00
|
|
|
ExprVector AtStrings(Actions);
|
2007-12-12 09:04:12 +08:00
|
|
|
AtLocs.push_back(AtLoc);
|
2008-12-10 08:02:53 +08:00
|
|
|
AtStrings.push_back(Res.release());
|
2008-12-09 21:15:23 +08:00
|
|
|
|
2007-12-12 09:04:12 +08:00
|
|
|
while (Tok.is(tok::at)) {
|
|
|
|
AtLocs.push_back(ConsumeToken()); // eat the @.
|
|
|
|
|
2008-12-10 04:22:58 +08:00
|
|
|
// Invalid unless there is a string literal.
|
2009-02-18 13:56:09 +08:00
|
|
|
if (!isTokenStringLiteral())
|
|
|
|
return ExprError(Diag(Tok, diag::err_objc_concat_string));
|
2007-12-12 09:04:12 +08:00
|
|
|
|
2010-08-24 14:29:42 +08:00
|
|
|
ExprResult Lit(ParseStringLiteralExpression());
|
2008-12-09 21:15:23 +08:00
|
|
|
if (Lit.isInvalid())
|
2008-12-13 23:32:12 +08:00
|
|
|
return move(Lit);
|
2008-12-09 21:15:23 +08:00
|
|
|
|
2008-12-10 08:02:53 +08:00
|
|
|
AtStrings.push_back(Lit.release());
|
2007-12-12 09:04:12 +08:00
|
|
|
}
|
2008-12-13 23:32:12 +08:00
|
|
|
|
|
|
|
return Owned(Actions.ParseObjCStringLiteral(&AtLocs[0], AtStrings.take(),
|
|
|
|
AtStrings.size()));
|
2007-08-22 01:43:55 +08:00
|
|
|
}
|
2007-08-22 23:14:15 +08:00
|
|
|
|
|
|
|
/// objc-encode-expression:
|
|
|
|
/// @encode ( type-name )
|
2010-08-24 14:29:42 +08:00
|
|
|
ExprResult
|
2008-12-13 23:32:12 +08:00
|
|
|
Parser::ParseObjCEncodeExpression(SourceLocation AtLoc) {
|
2007-08-24 02:16:40 +08:00
|
|
|
assert(Tok.isObjCAtKeyword(tok::objc_encode) && "Not an @encode expression!");
|
2008-12-13 23:32:12 +08:00
|
|
|
|
2007-08-22 23:14:15 +08:00
|
|
|
SourceLocation EncLoc = ConsumeToken();
|
2008-12-13 23:32:12 +08:00
|
|
|
|
2008-08-05 14:19:09 +08:00
|
|
|
if (Tok.isNot(tok::l_paren))
|
2008-12-13 23:32:12 +08:00
|
|
|
return ExprError(Diag(Tok, diag::err_expected_lparen_after) << "@encode");
|
|
|
|
|
2007-08-22 23:14:15 +08:00
|
|
|
SourceLocation LParenLoc = ConsumeParen();
|
2008-12-13 23:32:12 +08:00
|
|
|
|
2009-02-19 01:45:20 +08:00
|
|
|
TypeResult Ty = ParseTypeName();
|
2008-12-13 23:32:12 +08:00
|
|
|
|
2007-08-23 23:31:37 +08:00
|
|
|
SourceLocation RParenLoc = MatchRHSPunctuation(tok::r_paren, LParenLoc);
|
2008-12-13 23:32:12 +08:00
|
|
|
|
2009-02-19 01:45:20 +08:00
|
|
|
if (Ty.isInvalid())
|
|
|
|
return ExprError();
|
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
return Owned(Actions.ParseObjCEncodeExpression(AtLoc, EncLoc, LParenLoc,
|
2009-02-19 01:45:20 +08:00
|
|
|
Ty.get(), RParenLoc));
|
2007-08-22 23:14:15 +08:00
|
|
|
}
|
2007-08-23 23:25:28 +08:00
|
|
|
|
|
|
|
/// objc-protocol-expression
|
|
|
|
/// @protocol ( protocol-name )
|
2010-08-24 14:29:42 +08:00
|
|
|
ExprResult
|
2008-12-13 23:32:12 +08:00
|
|
|
Parser::ParseObjCProtocolExpression(SourceLocation AtLoc) {
|
2007-08-23 23:25:28 +08:00
|
|
|
SourceLocation ProtoLoc = ConsumeToken();
|
2008-12-13 23:32:12 +08:00
|
|
|
|
2008-08-05 14:19:09 +08:00
|
|
|
if (Tok.isNot(tok::l_paren))
|
2008-12-13 23:32:12 +08:00
|
|
|
return ExprError(Diag(Tok, diag::err_expected_lparen_after) << "@protocol");
|
|
|
|
|
2007-08-23 23:25:28 +08:00
|
|
|
SourceLocation LParenLoc = ConsumeParen();
|
2008-12-13 23:32:12 +08:00
|
|
|
|
2008-08-05 14:19:09 +08:00
|
|
|
if (Tok.isNot(tok::identifier))
|
2008-12-13 23:32:12 +08:00
|
|
|
return ExprError(Diag(Tok, diag::err_expected_ident));
|
|
|
|
|
2007-10-18 00:58:11 +08:00
|
|
|
IdentifierInfo *protocolId = Tok.getIdentifierInfo();
|
2007-08-23 23:25:28 +08:00
|
|
|
ConsumeToken();
|
2008-12-13 23:32:12 +08:00
|
|
|
|
2007-08-23 23:31:37 +08:00
|
|
|
SourceLocation RParenLoc = MatchRHSPunctuation(tok::r_paren, LParenLoc);
|
2007-08-23 23:25:28 +08:00
|
|
|
|
2008-12-13 23:32:12 +08:00
|
|
|
return Owned(Actions.ParseObjCProtocolExpression(protocolId, AtLoc, ProtoLoc,
|
|
|
|
LParenLoc, RParenLoc));
|
2007-08-23 23:25:28 +08:00
|
|
|
}
|
2007-10-16 07:39:13 +08:00
|
|
|
|
|
|
|
/// objc-selector-expression
|
|
|
|
/// @selector '(' objc-keyword-selector ')'
|
2010-08-24 14:29:42 +08:00
|
|
|
ExprResult Parser::ParseObjCSelectorExpression(SourceLocation AtLoc) {
|
2007-10-16 07:39:13 +08:00
|
|
|
SourceLocation SelectorLoc = ConsumeToken();
|
2008-12-13 23:32:12 +08:00
|
|
|
|
2008-08-05 14:19:09 +08:00
|
|
|
if (Tok.isNot(tok::l_paren))
|
2008-12-13 23:32:12 +08:00
|
|
|
return ExprError(Diag(Tok, diag::err_expected_lparen_after) << "@selector");
|
|
|
|
|
2007-10-17 04:40:23 +08:00
|
|
|
llvm::SmallVector<IdentifierInfo *, 12> KeyIdents;
|
2007-10-16 07:39:13 +08:00
|
|
|
SourceLocation LParenLoc = ConsumeParen();
|
|
|
|
SourceLocation sLoc;
|
2010-08-26 23:07:07 +08:00
|
|
|
|
|
|
|
if (Tok.is(tok::code_completion)) {
|
|
|
|
Actions.CodeCompleteObjCSelector(getCurScope(), KeyIdents.data(),
|
|
|
|
KeyIdents.size());
|
|
|
|
ConsumeCodeCompletionToken();
|
|
|
|
MatchRHSPunctuation(tok::r_paren, LParenLoc);
|
|
|
|
return ExprError();
|
|
|
|
}
|
|
|
|
|
2009-04-12 02:13:45 +08:00
|
|
|
IdentifierInfo *SelIdent = ParseObjCSelectorPiece(sLoc);
|
2010-08-28 06:32:41 +08:00
|
|
|
if (!SelIdent && // missing selector name.
|
|
|
|
Tok.isNot(tok::colon) && Tok.isNot(tok::coloncolon))
|
2008-12-13 23:32:12 +08:00
|
|
|
return ExprError(Diag(Tok, diag::err_expected_ident));
|
|
|
|
|
2007-10-17 04:40:23 +08:00
|
|
|
KeyIdents.push_back(SelIdent);
|
2007-12-06 06:21:29 +08:00
|
|
|
unsigned nColons = 0;
|
|
|
|
if (Tok.isNot(tok::r_paren)) {
|
2007-10-16 07:39:13 +08:00
|
|
|
while (1) {
|
2010-08-28 06:32:41 +08:00
|
|
|
if (Tok.is(tok::coloncolon)) { // Handle :: in C++.
|
|
|
|
++nColons;
|
|
|
|
KeyIdents.push_back(0);
|
|
|
|
} else if (Tok.isNot(tok::colon))
|
2008-12-13 23:32:12 +08:00
|
|
|
return ExprError(Diag(Tok, diag::err_expected_colon));
|
|
|
|
|
2010-08-28 06:32:41 +08:00
|
|
|
++nColons;
|
2007-10-16 07:39:13 +08:00
|
|
|
ConsumeToken(); // Eat the ':'.
|
|
|
|
if (Tok.is(tok::r_paren))
|
|
|
|
break;
|
2010-08-26 23:07:07 +08:00
|
|
|
|
|
|
|
if (Tok.is(tok::code_completion)) {
|
|
|
|
Actions.CodeCompleteObjCSelector(getCurScope(), KeyIdents.data(),
|
|
|
|
KeyIdents.size());
|
|
|
|
ConsumeCodeCompletionToken();
|
|
|
|
MatchRHSPunctuation(tok::r_paren, LParenLoc);
|
|
|
|
return ExprError();
|
|
|
|
}
|
|
|
|
|
2007-10-16 07:39:13 +08:00
|
|
|
// Check for another keyword selector.
|
|
|
|
SourceLocation Loc;
|
2009-04-12 02:13:45 +08:00
|
|
|
SelIdent = ParseObjCSelectorPiece(Loc);
|
2007-10-17 04:40:23 +08:00
|
|
|
KeyIdents.push_back(SelIdent);
|
2007-10-16 07:39:13 +08:00
|
|
|
if (!SelIdent && Tok.isNot(tok::colon))
|
|
|
|
break;
|
|
|
|
}
|
2007-12-06 06:21:29 +08:00
|
|
|
}
|
2007-10-16 07:39:13 +08:00
|
|
|
SourceLocation RParenLoc = MatchRHSPunctuation(tok::r_paren, LParenLoc);
|
2007-12-06 06:21:29 +08:00
|
|
|
Selector Sel = PP.getSelectorTable().getSelector(nColons, &KeyIdents[0]);
|
2008-12-13 23:32:12 +08:00
|
|
|
return Owned(Actions.ParseObjCSelectorExpression(Sel, AtLoc, SelectorLoc,
|
|
|
|
LParenLoc, RParenLoc));
|
2007-10-19 23:38:32 +08:00
|
|
|
}
|