1418 lines
37 KiB
Plaintext

/**
* This Objective-C grammar was copied and adapted from the Objective-C-CPD-Language Github project. (https://github.com/jkennedy1980/Objective-C-CPD-Language)
*
* Original author:
* @author Michael Hall - Based on Parnel SableCC ObjectiveC grammar and javacc C grammar
*
* Original source file:
* https://github.com/jkennedy1980/Objective-C-CPD-Language/blob/master/src/main/javacc/ObjC2.0.jj (commit 5b07b5cc424522e435929bbff7589c690100edf6)
*/
options {
JAVA_UNICODE_ESCAPE = true;
BUILD_PARSER=false;
CACHE_TOKENS=true;
STATIC=false;
}
PARSER_BEGIN(ObjectiveCParser)
package net.sourceforge.pmd.lang.objectivec.ast;
import java.io.*;
import java.util.*;
import net.sourceforge.pmd.lang.ast.CharStream;
import net.sourceforge.pmd.lang.ast.TokenMgrError;
/**
* Grammar to parse ObjectiveC 2.0
* @author Michael Hall - Based on Parnel SableCC ObjectiveC grammar and javacc C grammar
*
* Josh Kennedy - added CSTRING_LITERAL so parser doesn't die on CStrings.
* Paul Taykalo - added support for 0x literals, @compatibility_alias, and the $ character identifier
* Paul Taykalo - vertical tab (\v) and alert (\a) characters are added
*/
public class ObjectiveCParser
{
// Simple logging flag
private static final boolean verbose = false;
// HashSet for omitted test cases
private static final Set omissions = new HashSet();
static {
/* Foundation */
omissions.add("Foundation.h");
/* AppKit */
omissions.add("AppKit.h"); /* All #import's */
omissions.add("AppKitDefines.h"); /* All preprocessor */
omissions.add("NSSpellServer.h");
omissions.add("NSNibDeclarations.h");
/* CoreData */
omissions.add("CoreData.h"); /* All #import's */
/* CoreFoundation */
omissions.add("CoreFoundation.h"); /* All #import's */
/* CoreLocation */
omissions.add("CoreLocation.h"); /* All #import's */
/* Quartz Core */
omissions.add("CoreAnimation.h"); /* All #import's */
omissions.add("CoreImage.h"); /* All #import's */
omissions.add("CoreVideo.h"); /* All #import's */
omissions.add("CVBase.h"); /* All #import's */
omissions.add("CVBuffer.h"); /* All #import's */
omissions.add("CVDisplayLink.h"); /* All #import's */
omissions.add("CVHostTime.h"); /* All #import's */
omissions.add("CVImageBuffer.h"); /* All #import's */
omissions.add("CVOpenGLBuffer.h"); /* All #import's */
omissions.add("CVOpenGLBufferPool.h"); /* All #import's */
omissions.add("CVOpenGLTexture.h"); /* All #import's */
omissions.add("CVOpenGLTextureCache.h"); /* All #import's */
omissions.add("CVPixelBuffer.h"); /* All #import's */
omissions.add("CVPixelBufferPool.h"); /* All #import's */
omissions.add("CVPixelFormatDescription.h"); /* All #import's */
omissions.add("CVReturn.h"); /* All #import's */
omissions.add("QuartzCore.h"); /* All #import's */
/* WebKit */
omissions.add("DOM.h"); /* All #import's */
omissions.add("DOMCore.h"); /* All #import's */
omissions.add("DOMEvents.h"); /* All #import's */
omissions.add("DOMHTML.h"); /* All #import's */
omissions.add("DOMRanges.h"); /* All #import's */
omissions.add("DOMStylesheets.h"); /* All #import's */
omissions.add("DOMTraversal.h"); /* All #import's */
omissions.add("DOMViews.h"); /* All #import's */
omissions.add("DOMXPath.h"); /* All #import's */
omissions.add("WebKit.h"); /* All #import's */
/* Preference Panes */
omissions.add("PreferencePanes.h"); /* All #import's */
}
// HashSet for storing typedef types
private static Set types = new HashSet();
// Stack for determining when the parser
// is parsing a typedef definition.
private static Stack typedefParsingStack = new Stack();
// Returns true if the given string is
// a typedef type.
private static boolean isType(String type){
return types.contains(type);
}
// Returns true if the given string is
// in the test case omission list
private static boolean isOmitted(String fileName) {
return omissions.contains(fileName);
}
// Add a typedef type to those already defined
private static void addType(String type){
types.add(type);
}
// Prints out all the types used in parsing the c source
private static void printTypes(){
for (Iterator i = types.iterator(); i.hasNext();) {
System.out.println(i.next());
}
}
public ObjectiveCParser(String fileName)
{
this(System.in);
try { ReInit(new FileInputStream(new File(fileName))); }
catch(Exception e) { e.printStackTrace(); }
}
public static void main(String args[]) {
ObjectiveCParser parser = null;
String ps = System.getProperty("path.separator");
// Hack to include type "special types"
types.add("__signed__");
types.add("__const");
types.add("__inline__");
types.add("__signed");
if (args.length == 0) {
System.out.println("ObjectiveC 2.0 Parser Version 1.0: Reading from standard input . . .");
parser = new ObjectiveCParser(System.in);
} else if (args.length == 1) {
if (new File(args[0]).isDirectory()) {
String[] files = new File(args[0]).list();
for (int i=0;i<files.length;i++)
if (!isOmitted(files[i]))
processFile(new StringBuffer(args[0]).append(File.separator).append(files[i]).toString());
}
else processFile(args[0]);
} else {
System.out.println("ObjectiveC 2.0 Parser Version 1.0: Usage is one of:");
System.out.println(" java ObjectiveCParser < inputfile");
System.out.println("OR");
System.out.println(" java ObjectiveCParser inputfile");
return;
}
}
private static void processFile(String fileName) {
ObjectiveCParser parser = null;
System.out.println("ObjectiveC 2.0 Parser Version 1.0: Reading from file " + fileName + " . . .");
try {
parser = new ObjectiveCParser(new java.io.FileInputStream(fileName));
} catch (java.io.FileNotFoundException e) {
System.out.println("ObjectiveC 2.0 Parser Version 1.0: File " + fileName + " not found.");
return;
}
try {
parser.TranslationUnit();
System.out.println("ObjectiveC 2.0 Parser Version 1.0: ObjectiveC program parsed successfully.");
} catch (ParseException e) {
System.out.println(e.getMessage());
System.out.println("ObjectiveC 2.0 Parser Version 1.0: Encountered errors during parse.");
}
}
}
PARSER_END(ObjectiveCParser)
SKIP : {
" "
| "\t"
| "\n"
| "\r"
| "\f"
| "\\\n" /* Special issue for long strings concatenation */
| "\\\r\n" /* Special issue for long strings concatenation */
| "__weak"
| "__strong"
| "__attribute__"
| "__objc_exception__"
| "#" : PREPROCESSOR_OUTPUT
| "extern \"C\" {"
| "CF_EXPORT"
| "CF_INLINE"
| "CF_EXTERN_C_BEGIN"
| "CF_EXTERN_C_END"
| "CA_EXTERN_C_BEGIN"
| "CA_EXTERN_C_END"
| "CA_EXTERN"
| "CA_DEPRECATED"
| "COREDATA_EXTERN"
| "NS_INLINE"
| "APPKIT_EXTERN"
| "FOUNDATION_EXPORT" : PREPROCESSOR_OUTPUT /* mjh - Cocoa specific not currently of interest, ignore like preprocessor stuff */
| "DEPRECATED_IN_MAC_OS_X_VERSION_10_0_AND_LATER" /* mjh - not currently of interest */
| "DEPRECATED_IN_MAC_OS_X_VERSION_10_1_AND_LATER" /* mjh - not currently of interest */
| "DEPRECATED_IN_MAC_OS_X_VERSION_10_2_AND_LATER" /* mjh - not currently of interest */
| "DEPRECATED_IN_MAC_OS_X_VERSION_10_3_AND_LATER" /* mjh - not currently of interest */
| "DEPRECATED_IN_MAC_OS_X_VERSION_10_4_AND_LATER" /* mjh - not currently of interest */
| "DEPRECATED_IN_MAC_OS_X_VERSION_10_5_AND_LATER" /* mjh - not currently of interest */
| "DEPRECATED_IN_MAC_OS_X_VERSION_10_6_AND_LATER" /* mjh - not currently of interest */
| "AVAILABLE_MAC_OS_X_VERSION_10_0_AND_LATER" /* mjh - not currently of interest */
| "AVAILABLE_MAC_OS_X_VERSION_10_1_AND_LATER" /* mjh - not currently of interest */
| "AVAILABLE_MAC_OS_X_VERSION_10_2_AND_LATER" /* mjh - not currently of interest */
| "AVAILABLE_MAC_OS_X_VERSION_10_3_AND_LATER" /* mjh - not currently of interest */
| "AVAILABLE_MAC_OS_X_VERSION_10_4_AND_LATER" /* mjh - not currently of interest */
| "AVAILABLE_MAC_OS_X_VERSION_10_5_AND_LATER" /* mjh - not currently of interest */
| "AVAILABLE_MAC_OS_X_VERSION_10_6_AND_LATER" /* mjh - not currently of interest */
| "AVAILABLE_MAC_OS_X_VERSION_10_6_AND_LATER_BUT_DEPRECATED" /* mjh - not currently of interest */
| "AVAILABLE_MAC_OS_X_VERSION_10_0_AND_LATER_BUT_DEPRECATED_IN_MAC_OS_X_VERSION_10_6"
| "AVAILABLE_MAC_OS_X_VERSION_10_0_AND_LATER_BUT_DEPRECATED_IN_MAC_OS_X_VERSION_10_4"
| "AVAILABLE_MAC_OS_X_VERSION_10_1_AND_LATER_BUT_DEPRECATED_IN_MAC_OS_X_VERSION_10_4"
| "AVAILABLE_MAC_OS_X_VERSION_10_1_AND_LATER_BUT_DEPRECATED_IN_MAC_OS_X_VERSION_10_5"
| "AVAILABLE_MAC_OS_X_VERSION_10_2_AND_LATER_BUT_DEPRECATED_IN_MAC_OS_X_VERSION_10_5"
| "AVAILABLE_MAC_OS_X_VERSION_10_3_AND_LATER_BUT_DEPRECATED_IN_MAC_OS_X_VERSION_10_4"
| "AVAILABLE_MAC_OS_X_VERSION_10_3_AND_LATER_BUT_DEPRECATED_IN_MAC_OS_X_VERSION_10_5"
| "AVAILABLE_MAC_OS_X_VERSION_10_4_AND_LATER_BUT_DEPRECATED_IN_MAC_OS_X_VERSION_10_5"
| "AVAILABLE_MAC_OS_X_VERSION_10_4_AND_LATER_BUT_DEPRECATED_IN_MAC_OS_X_VERSION_10_6"
| "AVAILABLE_WEBKIT_VERSION_1_3_AND_LATER_BUT_DEPRECATED_IN_WEBKIT_VERSION_3_0"
| "AVAILABLE_WEBKIT_VERSION_1_0_AND_LATER_BUT_DEPRECATED_IN_WEBKIT_VERSION_4_0"
| "AVAILABLE_WEBKIT_VERSION_3_0_AND_LATER" /* mjh - not currently of interest */
| "AVAILABLE_WEBKIT_VERSION_3_0_AND_LATER_BUT_DEPRECATED" /* mjh - not currently of interest */
| "AVAILABLE_MAC_OS_X_VERSION_10_5_AND_LATER_BUT_DEPRECATED" /* mjh - not currently of interest */
| "AVAILABLE_IN_WEBKIT_VERSION_4_0" /* mjh - not currently of interest */
| "NP_LOADDS" /* mjh - not currently of interest */
| "NS_REQUIRES_NIL_TERMINATION"
| "UNAVAILABLE_ATTRIBUTE"
| <"," "NS_NONATOMIC_IPHONEONLY"> /* mjh - not currently of interest */
| "NS_FORMAT_FUNCTION" : FORMAT_FUNC
| "CF_FORMAT_ARGUMENT" : FORMAT_FUNC
| "__OSX_AVAILABLE_STARTING" : FORMAT_FUNC /* mjh - same sort of thing */
| "WEBKIT_OBJC_METHOD_ANNOTATION" : FORMAT_FUNC /* mjh - same sort of thing */
| "(void (^)" : VOID_BLOCK
}
/* mjh - we try to just ignore the parameter list */
<VOID_BLOCK> SKIP :
{
"))" : DEFAULT
}
<VOID_BLOCK> MORE:
{
< ~[")"] >
}
<PREPROCESSOR_OUTPUT> SKIP:
{
"\n" : DEFAULT
}
<PREPROCESSOR_OUTPUT> MORE:
{
"\\\n"
|
"\\\r\n"
|
< ~[] >
}
/* mjh - we try to just ignore the "(" whatever ")" */
<FORMAT_FUNC> SKIP :
{
")" : DEFAULT
}
<FOAMAT_FUNC> SPECIAL_TOKEN :
{
<FUNC: "(" (~[")"])* ")">
}
<FORMAT_FUNC> MORE :
{
"\\\n"
|
"\\\r\n"
|
< ~[")"] >
}
/* COMMENTS */
MORE :
{
"/*" : IN_MULTI_LINE_COMMENT
}
SPECIAL_TOKEN :
{
<SINGLE_LINE_COMMENT: "//" (~["\n", "\r"])* ("\n" | "\r" | "\r\n")?>
}
<IN_MULTI_LINE_COMMENT>
SPECIAL_TOKEN :
{
<MULTI_LINE_COMMENT: "*/" > : DEFAULT
}
<IN_MULTI_LINE_COMMENT>
MORE :
{
< ~[] >
}
/* OPERATORS */
TOKEN :
{
< ASSIGN: "=" >
| < LT: "<" >
| < GT: ">" >
| < BANG: "!" >
| < TILDE: "~" >
| < HOOK: "?" >
| < COLON: ":" >
| < EQ: "==" >
| < LE: "<=" >
| < GE: ">=" >
| < NE: "!=" >
| < SC_OR: "||" >
| < SC_AND: "&&" >
| < INCR: "++" >
| < DECR: "--" >
| < PLUS: "+" >
| < MINUS: "-" >
| < STAR: "*" >
| < SLASH: "/" >
| < BIT_AND: "&" >
| < BIT_OR: "|" >
| < XOR: "^" >
| < REM: "%" >
| < LSHIFT: "<<" >
| < PLUSASSIGN: "+=" >
| < MINUSASSIGN: "-=" >
| < STARASSIGN: "*=" >
| < SLASHASSIGN: "/=" >
| < ANDASSIGN: "&=" >
| < ORASSIGN: "|=" >
| < XORASSIGN: "^=" >
| < REMASSIGN: "%=" >
| < LSHIFTASSIGN: "<<=" >
| < RSIGNEDSHIFTASSIGN: ">>=" >
| < RUNSIGNEDSHIFTASSIGN: ">>>=" >
| < ELLIPSIS: "..." >
}
TOKEN :
{
< ESCAPE_SEQUENCE :
"\\" ["'","\"","?","\\","a","b","f","n","r","t","v"] /* SIMPLE_ESCAPE_SEQUENCE */
| "\\" <OCTAL_DIGIT> (<OCTAL_DIGIT> (<OCTAL_DIGIT>)?)? /* OCTAL_ESCAPE_SEQUENCE */
| "\\" "x" (<HEX_DIGIT>)+ /* HEX_ESCAPE_SEQUENCE */
>
}
TOKEN : {
<INTEGER_LITERAL: <DECIMAL_LITERAL> (["l","L"] | ("ul" | "UL") )? | <HEX_LITERAL> (["l","L","U"])? | <OCTAL_LITERAL> (["l","L"])?>
| <#DECIMAL_LITERAL: ["1"-"9"] (["0"-"9"])* | (["0"])+>
| <#HEX_LITERAL: "0" ["x","X"] (["0"-"9","a"-"f","A"-"F"])+>
| <#OCTAL_LITERAL: "0" (["0"-"7"])*>
| <FLOATING_POINT_LITERAL: (["0"-"9"])+ "." (["0"-"9"])* (<EXPONENT>)? (["f","F","d","D"])? | "." (["0"-"9"])+ (<EXPONENT>)? (["f","F","d","D"])? | (["0"-"9"])+ <EXPONENT> (["f","F","d","D"])? | (["0"-"9"])+ (<EXPONENT>)? ["f","F","d","D"]>
| <#EXPONENT: ["e","E"] (["+","-"])? (["0"-"9"])+>
| <CHARACTER_LITERAL: "\'" (~["\'","\\","\n","\r"] | "\\" (["n","t","b","r","f","a","v","\\","\'","\""] | ["0"-"7"] (["0"-"7"])? | ["0"-"3"] ["0"-"7"] ["0"-"7"])) "\'">
| <STRING_LITERAL: [ "@" ] "\"" ( ~["\"","\\","\n","\r"] | "\\" ["x","X"] (["0"-"9","a"-"f","A"-"F"]) | "\\" ( ["n","t","b","r","f","a","v","\\","\'","\""] | ["0"-"7"] (["0"-"7"])? | ["0"-"3"] ["0"-"7"] ["0"-"7"] | ( ["\n","\r"] | "\r\n")))* "\"">
| <CSTRING_LITERAL: "\"" ( ~["\"","\\","\n","\r"] | "\\" ["x","X"] (["0"-"9","a"-"f","A"-"F"]) | "\\" ( ["n","t","b","r","f","a","v","\\","\'","\""] | ["0"-"7"] (["0"-"7"])? | ["0"-"3"] ["0"-"7"] ["0"-"7"] | ( ["\n","\r"] | "\r\n")))* "\"">
}
TOKEN :
{
< DIGIT: ["0"-"9"] >
|
< NONZERO_DIGIT : ["1"-"9"] >
|
< OCTAL_DIGIT : ["1"-"7"] >
|
< #HEX_DIGIT : <DIGIT> | ["a"-"f"] | ["A"-"F"] >
|
< #HEX_QUAD : <HEX_DIGIT> <HEX_DIGIT> <HEX_DIGIT> <HEX_DIGIT> >
|
< #UNSIGNED_SUFFIX : ["u","U"] | ("UL" | "ul") >
|
< #LONG_SUFFIX : ["l","L"] | ("LL" | "ll") >
|
< #INTEGER_SUFFIX : <UNSIGNED_SUFFIX> (<LONG_SUFFIX>)? | <LONG_SUFFIX> (<UNSIGNED_SUFFIX>)? >
|
< DECIMAL_CONSTANT : <NONZERO_DIGIT> (<DIGIT>)* >
|
< OCTAL_CONSTANT : "0" | <OCTAL_DIGIT> >
|
< HEX_PREFIX : "0" ["x","X"] >
|
< HEX_CONSTANT : <HEX_PREFIX> (<HEX_DIGIT>)+ >
}
TOKEN :
{
< INTEGER_CONSTANT :
<DECIMAL_CONSTANT> (<INTEGER_SUFFIX>)?
| <OCTAL_CONSTANT> (<INTEGER_SUFFIX>)?
| <HEX_CONSTANT> (<INTEGER_SUFFIX>)?
>
}
TOKEN : {
<FIXED_NUM_TYPE : "short" | "int" | ("long")+ >
}
TOKEN : {
<CONTINUE: "continue"> |
<VOLATILE: "volatile"> |
<REGISTER: "register"> |
<UNSIGNED: "unsigned"> |
<UCHAR: "unsigned char"> |
<TYPEDEF: "typedef"> |
<DFLT: "default"> |
<DOUBLE: "double"> |
<BOOL: "BOOL" > |
<BOOLEAN: "Boolean" > |
<SIZEOF: "sizeof"> |
<SWITCH: "switch"> |
<RETURN: "return"> |
<EXTERN: "extern"> |
<STRUCT: "struct"> |
<STATIC: "static"> |
<SIGNED: "signed"> |
<SCHAR: "signed char"> |
<SINT32: "SInt32"> |
<INT32_T: "int32_t"> |
<WHILE: "while"> |
<BREAK: "break"> |
<UNION: "union"> |
<CONST: "const"> |
<FLOAT: "float"> |
<ELSE: "else"> |
<CASE: "case"> |
<ENUM: "enum"> |
<AUTO: "auto"> |
<VOID: "void"> |
<CHAR: "char"> |
<GOTO: "goto"> |
<FOR: "for"> |
<IF: "if"> |
<DO: "do">
}
TOKEN : {
<IBACTION: "IBAction">
| <IBOUTLET: "IBOutlet">
| <ID: "id">
| <SEL: "SEL">
| <IMP: "IMP">
| <CLASS: "Class">
| <NIL: "nil">
}
/* mjh - see ProtocolQualifier
TOKEN :
{
<IN_KEYWORD : "in">
}
*/
TOKEN :
{
<PROTOCOL_QUALIFIER_EXCEPT_IN : // 'in' must be separate token
"out"
| "inout"
| "bycopy"
| "byref"
| "oneway"
>
}
TOKEN :
{
<LEFT_PAREN: "(">
| <RIGHT_PAREN: ")">
| <LEFT_SQUARE: "[">
| <RIGHT_SQUARE: "]">
| <LEFT_BRACE: "{">
| <RIGHT_BRACE: "}">
| <SEMICOLON: ";">
}
TOKEN :
{
<AT_IMPLEMENTATION: "@implementation">
| <AT_INTERFACE: "@interface">
| <AT_PROTOCOL: "@protocol">
| <AT_ENCODE: "@encode">
| <AT_SYNCHRONIZED: "@synchronized">
| <AT_SELECTOR: "@selector">
| <AT_END: "@end">
| <AT_DEFS: "@defs">
| <AT_CLASS: "@class">
|
<AT_TRY: "@try">
| <AT_THROW: "@throw">
| <AT_CATCH: "@catch">
| <AT_FINALLY: "@finally">
|
<AT_PRIVATE: "@private">
| <AT_PACKAGE: "@package">
| <AT_PUBLIC: "@public">
| <AT_PROTECTED: "@protected">
|
<AT_PROPERTY: "@property">
| <AT_SYNTHESIZE: "@synthesize">
| <AT_DYNAMIC: "@dynamic">
|
<AT_OPTIONAL: "@optional">
| <AT_REQUIRED: "@required">
| <AT_AUTORELEASEPOOL: "@autoreleasepool">
| <AT_COMPATIBILITY_ALIAS: "@compatibility_alias">
| <AT: "@">
}
TOKEN :
{
<CGFLOAT: "CGFloat">
| <NSINTEGER: "NSInteger">
| <NSRECT: "NSRect">
| <NSSIZE: "NSSize">
| <NSUINTEGER: "NSUInteger">
}
TOKEN : {
<POSSIBLE_COCOA_TYPE: "NS" (["a"-"z"] | ["A"-"Z"] | <DIGIT> | "_")*>
| <POSSIBLE_CORE_TYPE: "CF" (["a"-"z"] | ["A"-"Z"] | <DIGIT> | "_")*>
}
TOKEN : { <EXPORTED_CALLBACK : "EXPORTED_CALLBACK"> }
TOKEN :
{
< IDENT : <IDENT_NONDIGIT> ( <IDENT_NONDIGIT> | <DIGIT>)* >
| <#IDENT_NONDIGIT : <NONDIGIT> | <UNIVERSAL_CHARACTER_NAME> > // "may include other implementation-defined characters"
| <#NONDIGIT : ["a"-"z"] | ["A"-"Z"] | "_" | "$" | <NONDIGIT_UNICODE> >
| <#NONDIGIT_UNICODE : [ /* source: http://stackoverflow.com/questions/30933785 */
"\u0024",
"\u0041"-"\u005a",
"\u005f",
"\u0061"-"\u007a",
"\u00c0"-"\u00d6",
"\u00d8"-"\u00f6",
"\u00f8"-"\u00ff",
"\u0100"-"\u1fff"
] >
| <#UNIVERSAL_CHARACTER_NAME : ("\\u" <HEX_QUAD>) | ("\\U" <HEX_QUAD> <HEX_QUAD>) >
}
TOKEN : { <BLOCK: "(^"> }
/***********************************************
* THE OBJECTIVEC LANGUAGE GRAMMAR STARTS HERE *
***********************************************/
void TranslationUnit() : {}
{
(ExternalDeclaration())+
}
void ExternalDeclaration() : {}
{
( LOOKAHEAD( FunctionDefinition() ) FunctionDefinition()
| LOOKAHEAD(3) StrippedParens() /* mjh if we skip all but parens for something like __attribute__((__objc_exception__)) */
| Declaration()
| LOOKAHEAD(3) ClassInterface()
| LOOKAHEAD(3) ClassImplementation()
| CategoryInterface()
| CategoryImplementation()
| ProtocolDeclaration()
| ClassDeclarationList() )
}
void StrippedParens() : {} /* See ExternalDeclaration above */
{
<LEFT_PAREN> [ StrippedParens() ] <RIGHT_PAREN>
}
void FunctionDefinition() : {}
{
[LOOKAHEAD(DeclarationSpecifiers()) DeclarationSpecifiers()] Declarator() [ DeclarationList() ]
CompoundStatement()
}
void Declaration() : {}
{
DeclarationSpecifiers() [ InitDeclaratorList() ] [ <EXPORTED_CALLBACK> "(" ParameterTypeList() ")" "(" ParameterTypeList() ")" ] /* ugly hack for single header typedef */ ";"
}
void ClassInterface() : {}
{
<AT_INTERFACE> ClassName() (ColonSuperClassName())? (ProtocolReferenceList())?
[ LOOKAHEAD(2) ["{"] <AT_INTERFACE> ClassName() (ColonSuperClassName())? (ProtocolReferenceList())? ] /* dup for preprocessor #else */
(InstanceVariables())? (InterfaceDeclaration())* <AT_END>
}
void ClassImplementation() : {}
{
<AT_IMPLEMENTATION> ClassName() (LOOKAHEAD(2) ColonSuperClassName())? (InstanceVariables())? (ImplementationDefinition())* <AT_END>
}
void CategoryInterface() : {}
{
<AT_INTERFACE> ClassName() <LEFT_PAREN> (CategoryName())? <RIGHT_PAREN> (ProtocolReferenceList())? (InterfaceDeclaration())* <AT_END>
}
void CategoryImplementation() : {}
{
<AT_IMPLEMENTATION> ClassName() <LEFT_PAREN> CategoryName() <RIGHT_PAREN> (ImplementationDefinition())* <AT_END>
}
void ProtocolDeclaration() : {}
{
<AT_PROTOCOL> ProtocolList() (ProtocolReferenceList())? ProtocolInterfaceDeclaration() ( <AT_END> | ";" )
}
void ClassDeclarationList() : {}
{
<AT_CLASS> ClassList() ";"
}
void ClassList() : {}
{
ClassName() ["," ClassList()]
}
void ProtocolReferenceList() : {}
{
<LT> ProtocolList() <GT> [ <LT> ProtocolList() <GT> /* if preproc #else, see AppKit NSWindow */ ]
}
void ProtocolList() : {}
{
ProtocolName() ["," ProtocolList()]
}
Token ObjCIDENT() : { Token t; }
{
t = <POSSIBLE_COCOA_TYPE> { return t; } | t = <POSSIBLE_CORE_TYPE> { return t; } | t = <IDENT> { return t; }
}
void ClassName() : {}
{
ObjCIDENT()
}
void SuperClassName() : {}
{
ObjCIDENT()
}
void ColonSuperClassName() : {}
{
<COLON> SuperClassName()
}
void CategoryName() : {}
{
<POSSIBLE_COCOA_TYPE> | <POSSIBLE_CORE_TYPE> | <IDENT>
}
void ProtocolName() : {}
{
<POSSIBLE_COCOA_TYPE> | <POSSIBLE_CORE_TYPE> | <IDENT>
}
void InstanceVariables() : {}
{
<LEFT_BRACE> (InstanceVariableDeclaration())* <RIGHT_BRACE> [ ";" ]
}
void InstanceVariableDeclaration() : {}
{
VisibilitySpecification()
| LOOKAHEAD(3) InstanceVariableDeclarator()
| StructDeclaration()
}
void InstanceVariableDeclarator() : {}
{
( LOOKAHEAD(3) [ IBOutlet() ]
(LOOKAHEAD(2) ClassName() "*" | <ID> [ ProtocolReferenceList() ]) <IDENT> ";"
| Declaration() )
}
void VisibilitySpecification() : {}
{
<AT_PRIVATE> [ ";" ]
| <AT_PROTECTED>
| <AT_PACKAGE>
| <AT_PUBLIC>
}
void IBOutlet() : {} /* mjh */
{
<IBOUTLET>
}
void ProtocolInterfaceDeclaration() : {}
{
(InterfaceDeclaration())* (QualifiedProtocolInterfaceDeclaration())*
}
void QualifiedProtocolInterfaceDeclaration() : {}
{
<AT_OPTIONAL> (InterfaceDeclaration())*
| <AT_REQUIRED> (InterfaceDeclaration())*
}
void InterfaceDeclaration() : {}
{
Declaration()
| PropertyDeclaration()
| MethodDeclaration()
}
void PropertyDeclaration() : {}
{
<AT_PROPERTY> (PropertyAttributesDeclaration())? ( LOOKAHEAD(3) StructDeclaration() | <ID> (ProtocolReferenceList())? <IDENT> ";" )
}
void PropertyAttributesDeclaration() : {}
{
<LEFT_PAREN> PropertyAttributesList() <RIGHT_PAREN>
}
void PropertyAttributesList() : {}
{
PropertyAttribute() ["," PropertyAttributesList()]
}
void PropertyAttribute() : {}
{
LOOKAHEAD(3) <IDENT> <ASSIGN> /* prop */ <IDENT> [ <COLON> ] // setter
// | LOOKAHEAD(2) <IDENT> <ASSIGN> /* getter or ivar */ <IDENT> // getter ivar
| <IDENT> // nonatomic, readwrite, readonly, retain, assign, copy
| <POSSIBLE_COCOA_TYPE> /* NS_NONATOMIC_IPHONEONLY */
}
void MethodDeclaration() : {}
{
ClassMethodDeclaration()
| InstanceMethodDeclaration()
}
void ClassMethodDeclaration() : {}
{
<PLUS> (MethodType())? MethodSelector() <SEMICOLON>
}
void InstanceMethodDeclaration() : {}
{
<MINUS> (MethodType())? MethodSelector() <SEMICOLON>
}
void ImplementationDefinition() : {}
{
( LOOKAHEAD( FunctionDefinition() ) FunctionDefinition()
| Declaration()
| PropertyImplementation()
| MethodDefinition() )
}
void PropertyImplementation() : {}
{
<AT_SYNTHESIZE> PropertySynthesizeList() <SEMICOLON>
| <AT_DYNAMIC> PropertySynthesizeList() <SEMICOLON>
}
void PropertySynthesizeList() : {}
{
PropertySynthesizeItem() ["," PropertySynthesizeList()]
/*
PropertySynthesizeList() "," PropertySynthesizeItem()
| PropertySynthesizeItem()
*/
}
void PropertySynthesizeItem() : {}
{
LOOKAHEAD(2) <IDENT>
| <IDENT> <EQ> <IDENT>
}
void MethodDefinition() : {}
{
ClassMethodDefinition()
| InstanceMethodDefinition()
}
void ClassMethodDefinition() : {}
{
<PLUS> (MethodType())? MethodSelectorNoList() (Declaration())* (<SEMICOLON>)? CompoundStatement()
}
void InstanceMethodDefinition() : {}
{
<MINUS> (MethodType())? MethodSelectorNoList() (Declaration())* (<SEMICOLON>)? CompoundStatement()
}
void MethodSelectorNoList() : {}
{
LOOKAHEAD(3) KeywordSelector() [LOOKAHEAD(2) (",..." | "," "...")]
| UnarySelector()
}
void MethodSelector() : {}
{
LOOKAHEAD(3) MethodSelectorNoList()
| KeywordSelector() "," ParameterTypeList() // this is correct according to objcbook, but causes conflict if followed by declaration*
}
void UnarySelector() : {}
{
Selector()
}
void KeywordSelector() : {}
{
(LOOKAHEAD(2) KeywordDeclarator())+
}
void KeywordDeclarator() : {}
{
(Selector())? <COLON> (MethodType())? <IDENT>
}
void Selector() : {}
{
<IDENT>
}
void MethodType() : {}
{
<LEFT_PAREN> (LOOKAHEAD(3) Block() | TypeNameWithUnknownType()) <RIGHT_PAREN>
}
void SelectorExpression() : {}
{
<AT_SELECTOR> <LEFT_PAREN> SelectorName() <RIGHT_PAREN>
}
void SelectorName() : {}
{
LOOKAHEAD(2) Selector()
| (KeywordName())+
}
void KeywordName() : {}
{
(Selector())? <COLON>
}
void ProtocolExpression() : {}
{
<AT_PROTOCOL> <LEFT_PAREN> ProtocolName() <RIGHT_PAREN>
}
void EncodeExpression() : {}
{
<AT_ENCODE> <LEFT_PAREN> TypeNameWithUnknownType() <RIGHT_PAREN>
}
void DeclarationList() : {}
{
( LOOKAHEAD(Declaration()) Declaration() )+
}
void DeclarationSpecifiers() : {}
{
/**
* We enumerate the possibilities (closure?) like SpecifierQualifierList
* instead of the standard C grammar recursive definition
* Although more complex it might actually end up cleaner?
* (For one thing you don't keep recursively matching <ObjCIDENT> types
* in TypeSpecifier(), this can muddle things or require semantic lookahead
* workarounds, see the sample javacc C implementation of this nonterminal)
*
* Assumes none actually should be duplicated
*/
LOOKAHEAD(3) TypeSpecifier() [ StorageClassSpecifier() ] [ TypeQualifier() ]
| TypeSpecifier() [ TypeQualifier() ] [ StorageClassSpecifier() ]
| LOOKAHEAD(3) StorageClassSpecifier() [ TypeQualifier() ] [ LOOKAHEAD(2) TypeSpecifier() ]
| LOOKAHEAD(3) StorageClassSpecifier() [ LOOKAHEAD(2) TypeSpecifier() ] [ TypeQualifier() ]
| LOOKAHEAD(3) TypeQualifier() [ StorageClassSpecifier() ] [ LOOKAHEAD(2) TypeSpecifier() ]
| LOOKAHEAD(2) TypeQualifier() [ LOOKAHEAD(2) TypeSpecifier() ] [ StorageClassSpecifier() ]
}
void StorageClassSpecifier() : {}
{
( <AUTO> | <REGISTER> | <STATIC> | <EXTERN> | <TYPEDEF>
{
typedefParsingStack.push(Boolean.TRUE);
} )
}
void TypeSpecifier() : {}
{
( GreedyFixedNumType() | <VOID> | <CHAR> | <FLOAT> | <DOUBLE> | <BOOL> |
<IBACTION> /* mjh - same as void */ | <ID> | /* Cocoa additions */
<INT32_T> |
<CGFLOAT> | <NSRECT> | <NSUINTEGER> | <NSINTEGER> | <NSSIZE> | <SEL> | <CLASS> | <IMP> | /* mjh - more cocoa additions */
<SINT32> | <BOOLEAN> | /* CoreFoundation additions */
<SCHAR> | <UCHAR> | StructOrUnionSpecifier() | EnumSpecifier() | LOOKAHEAD( { isType(getToken(1).image) } )TypedefName() |
PossibleCocoaType() | PossibleCoreType() | PossibleUnknownType()
)
}
void GreedyFixedNumType() : {}
{
LOOKAHEAD(2) [ <SIGNED> | <UNSIGNED> ] <FIXED_NUM_TYPE> [ LOOKAHEAD(2) <FIXED_NUM_TYPE> ] /* long long */ |
<SIGNED> |
<UNSIGNED>
}
void PossibleCocoaType() : { Token t; }
{
t = <POSSIBLE_COCOA_TYPE>
{
if (verbose)
System.out.println("WARNING: Adding typedef on possible Cocoa type " + t.image + " at line number " + t.beginLine + ", column number " + t.beginColumn);
addType(t.image);
}
}
void PossibleCoreType() : { Token t; }
{
t = <POSSIBLE_CORE_TYPE>
{
if (verbose)
System.out.println("WARNING: Adding typedef on possible CoreFoundation type " + t.image + " at line number " + t.beginLine + ", column number " + t.beginColumn);
addType(t.image);
}
}
void PossibleUnknownType() : { Token t; }
{
t = <IDENT>
{
if (verbose)
System.out.println("WARNING: Adding typedef on unknown identifier " + t.image + " at line number " + t.beginLine + ", column number " + t.beginColumn);
addType(t.image);
}
}
void TypeSpecifierWithUnknownType() : {}
{
LOOKAHEAD(2) (ObjCIDENT() | <ID>) (ProtocolReferenceList())?
| TypeSpecifier()
}
void TypeQualifier() : {}
{
( <CONST> | <VOLATILE> )
}
void StructOrUnionSpecifier() : {}
{
{
typedefParsingStack.push(Boolean.FALSE);
}
StructOrUnion() ( LOOKAHEAD(3) [ LOOKAHEAD(2) ObjCIDENT() ] ( "{" StructDeclarationList() "}" | ObjCIDENT() ) | <IDENT> )
{
typedefParsingStack.pop();
}
}
void StructOrUnion() : {}
{
( <STRUCT> | <UNION> )
}
void StructDeclarationList() : {}
{
(StructDeclaration())+
}
void InitDeclaratorList() : {}
{
InitDeclarator() ("," InitDeclarator())*
{
// Finished with a typedefDeclaration??
if(!(typedefParsingStack.empty()) && ((Boolean)typedefParsingStack.peek()).booleanValue()){
typedefParsingStack.pop();
}
}
}
void InitDeclarator() : {}
{
Declarator() [ "=" Initializer() ]
}
void ProtocolQualifier() : {}
{
<PROTOCOL_QUALIFIER_EXCEPT_IN>
| /* <IN_KEYWORD> */ In() /* Making it a IN_KEYWORD token disallows other uses */
}
void In() : {}
{
LOOKAHEAD( { getToken(0).image.equals("in") } ) <IDENT>
}
void StructDeclaration() : {}
{
SpecifierQualifierList() ( StructDeclaratorList() | "[" ConstantExpression() "]" ) ";"
}
void SpecifierQualifierWithUnknownType() : {}
{
LOOKAHEAD(2) TypeSpecifierWithUnknownType()
| TypeQualifier()
| ProtocolQualifier()
}
void SpecifierQualifierList() : {}
{
/**
* Since we don't know what all possible typedef's might be and end up matching any <IDENT> as a possible type
* we could recursively match both tokens in something like...
* CFIndex version;
* to PossibleUnknownType() in TypeSpecifier().
* So we don't follow the elegant C recursive definition but enumeratte the two possible combinations
*/
TypeSpecifier() [ TypeQualifier() ]
| TypeQualifier() TypeSpecifier()
}
void StructDeclaratorList() : {}
{
StructDeclarator() ( "," StructDeclarator() )*
}
void StructDeclarator() : {}
{
Declarator()
}
void EnumSpecifier() : {}
{
<ENUM> ( LOOKAHEAD(3) [ ObjCIDENT() ] "{" EnumeratorList() "}" | <IDENT> )
}
void EnumeratorList() : {}
{
Enumerator() ("," [ Enumerator() ])*
}
void Enumerator() : {}
{
ObjCIDENT() [ "=" ConstantExpression() ]
}
void Declarator() : {}
{
LOOKAHEAD(3) Block()
| LOOKAHEAD(3) [ Pointer() ] DirectDeclarator() [ ":" ConstantExpression() ]
| ":" ConstantExpression()
| LOOKAHEAD(2) TypeSpecifier() // trying to handle 'typedef type-spec type-spec' special case, a little ugly
}
void Block() : {}
{
( LOOKAHEAD(3) <VOID> | <BOOL> | <ID> | ObjCIDENT() [ "*" ])? <BLOCK> [ ObjCIDENT() ] <RIGHT_PAREN> <LEFT_PAREN> ParameterList() <RIGHT_PAREN>
}
void DirectDeclarator() : { Token t;}
{
( t = ObjCIDENT()
{ if(!(typedefParsingStack.empty()) && ((Boolean)typedefParsingStack.peek()).booleanValue()){
addType(t.image);
}
}
| LOOKAHEAD(3) "(" Declarator() ")" )
{ typedefParsingStack.push( Boolean.FALSE ); }
( "[" [ ConstantExpression() ] "]" |
LOOKAHEAD(3) "(" ParameterTypeList() ")" |
"(" [ IdentifierList() ] ")" )*
{ typedefParsingStack.pop(); }
}
void Pointer() : {}
{
"*" [ TypeQualifierList() ] [ Pointer() ]
}
void TypeQualifierList() : {}
{
(TypeQualifier())+
}
void ParameterTypeList() : {}
{
ParameterList() ["," "..." ]
}
void ParameterList() : {}
{
ParameterDeclaration() (LOOKAHEAD(2) "," ParameterDeclaration())*
}
void ParameterDeclaration() : {}
{
DeclarationSpecifiers() ( LOOKAHEAD(Declarator()) Declarator() | [ AbstractDeclarator() ] )
}
void IdentifierList() : {}
{
<IDENT> ("," <IDENT>)*
}
void Initializer() : {}
{
( AssignmentExpression() |
"{" InitializerList() [","] "}" )
}
void InitializerList() : {}
{
Initializer() (LOOKAHEAD(2) "," Initializer())*
}
void TypeName() : {}
{
SpecifierQualifierList() [ AbstractDeclarator() ]
}
void TypeNameWithUnknownType() : {}
{
(SpecifierQualifierWithUnknownType())+ (AbstractDeclarator())?
}
void AbstractDeclarator() : {}
{
( LOOKAHEAD(3) Pointer() |
[Pointer()] DirectAbstractDeclarator() )
}
void DirectAbstractDeclarator() : {}
{
( LOOKAHEAD(2) "(" AbstractDeclarator() ")" |
"[" [ConstantExpression()] "]" |
"(" [ParameterTypeList()] ")" )
( "[" [ ConstantExpression() ] "]" | "(" [ ParameterTypeList() ] ")" )*
}
void TypedefName() : {}
{
ObjCIDENT()
}
void Statement() : {}
{
( LOOKAHEAD(2) LabeledStatement() |
LOOKAHEAD(3) InitStatement() |
ExpressionStatement() |
CompoundStatement() |
SelectionStatement() |
IterationStatement() |
JumpStatement() )
}
void LabeledStatement() : {}
{
( <IDENT> ":" Statement() |
<CASE> ConstantExpression() ":" Statement() |
<DFLT> ":" Statement() )
}
void InitStatement() : {} /* mjh not part of normal C grammar but works */
{
[ LOOKAHEAD(2) TypeSpecifier() ] ["*"] <IDENT> ("," <IDENT>)*
("=" ( LOOKAHEAD(3) MessageExpression() | PostfixExpression() ) ( ";" | "," InitStatement() )
| ";") /* catch embedded variable declarations */
}
void ExpressionStatement() : {}
{
[ Expression() ] ";"
}
void CompoundStatement() : {}
{
"{" ( LOOKAHEAD(DeclarationList()) DeclarationList() | StatementList() )* "}"
}
void StatementList() : {}
{
(LOOKAHEAD(2) Statement())+
}
void SelectionStatement() : {}
{
( <IF> "(" [ LOOKAHEAD(2) NestedLogicalExpression() ] Expression() ")" Statement() [ LOOKAHEAD(2) <ELSE> Statement() ] |
<SWITCH> "(" Expression() ")" Statement() )
}
void NestedLogicalExpression() : {} /* mjh - introduced as part of nesting handling */
{
"(" [ LOOKAHEAD(2) NestedLogicalExpression() ] Expression() ")" LogicalOperator()
}
void LogicalOperator() : {} /* mjh - introduced as part of nesting handling */
{
<XOR> | <EQ> | <SC_OR> | <SC_AND>
}
void IterationStatement() : {}
{
( <WHILE> "(" Expression() ")" Statement() |
<DO> Statement() <WHILE> "(" Expression() ")" ";" |
<FOR> "("
( LOOKAHEAD(3) TypeSpecifier() <IDENT> In() PrimaryExpression() |
[ LOOKAHEAD(NumberTypeInit()) NumberTypeInit() ] [ Expression() ] ";" [ Expression() ] ";" [ Expression() ] )
")" Statement()
)
}
void NumberTypeInit() : {} /* mjh - allow number type initializers, e.g. on 'for' statement inits */
{
( <FIXED_NUM_TYPE> | <FLOAT> | <DOUBLE> )
}
void JumpStatement() : {}
{
( <GOTO> <IDENT> ";" |
<CONTINUE> ";" |
<BREAK> ";" |
<RETURN> [ Expression() ] ";" )
}
void Expression() : {}
{
AssignmentExpression() ( "," AssignmentExpression() )*
}
void AssignmentExpression() : {}
{
LOOKAHEAD(UnaryExpression() AssignmentOperator()) UnaryExpression() AssignmentOperator() AssignmentExpression() |
LOOKAHEAD(3) ConditionalExpression()
}
void AssignmentOperator() : {}
{
( "=" | "*=" | "/=" | "%=" | "+=" | "-=" | "<<=" | ">>=" | "&=" | "^=" | "|=" )
}
void ConditionalExpression() : {}
{
LogicalORExpression() [ "?" Expression() ":" ConditionalExpression() ]
}
void ConstantExpression() : {}
{
ConditionalExpression()
}
void LogicalORExpression() : {}
{
LogicalANDExpression() [ "||" (LOOKAHEAD(3) LogicalORExpression() | NestedLogicalExpression()) ]
}
void LogicalANDExpression() : {}
{
InclusiveORExpression() [ "&&" (LOOKAHEAD(3) LogicalANDExpression() | NestedLogicalExpression()) ]
}
void InclusiveORExpression() : {}
{
ExclusiveORExpression() [ "|" InclusiveORExpression() ]
}
void ExclusiveORExpression() : {}
{
ANDExpression() [ "^" ExclusiveORExpression() ]
}
void ANDExpression() : {}
{
EqualityExpression() [ "&" ANDExpression() ]
}
void EqualityExpression() : {}
{
RelationalExpression() [ ( "==" | "!=" ) EqualityExpression() ]
}
void RelationalExpression() : {}
{
ShiftExpression() [ ( "<" | ">" | "<=" | ">=" ) RelationalExpression() ]
}
void ShiftExpression() : {}
{
AdditiveExpression() [ ( "<<" | ">>" ) ShiftExpression() ]
}
void AdditiveExpression() : {}
{
MultiplicativeExpression() [ ( "+" | "-" ) AdditiveExpression() ]
}
void MultiplicativeExpression() : {}
{
CastExpression() [ ( "*" | "/" | "%" ) MultiplicativeExpression() ]
}
void CastExpression() : {}
{
LOOKAHEAD("(" TypeName() ")" CastExpression() ) "(" TypeName() ")" CastExpression() |
// LOOKAHEAD(3) "(" TypeSpecifier() ")" CastExpression() |
UnaryExpression()
}
void UnaryExpression() : {}
{
( LOOKAHEAD(3) PostfixExpression() |
"++" UnaryExpression() |
"--" UnaryExpression() |
UnaryOperator() CastExpression() |
<SIZEOF> ( LOOKAHEAD(UnaryExpression() ) UnaryExpression() | "(" TypeName() ")" ) )
}
void UnaryOperator() : {}
{
( "&" | "*" | "+" | "-" | "~" | "!" )
}
void PostfixExpression() : {}
{
<NIL> | /* mjh */
PrimaryExpression() ( "[" Expression() "]" |
"(" [ LOOKAHEAD(ArgumentExpressionList() ) ArgumentExpressionList() ] ")" |
"." <IDENT> |
"->" <IDENT> |
"++" |
"--" )*
}
void PrimaryExpression() : {}
{
( ObjCIDENT() |
Constant() |
"(" Expression() ")" |
MessageExpression() |
SelectorExpression() |
ProtocolExpression() |
EncodeExpression() )
}
void MessageExpression() : {}
{
<LEFT_SQUARE> Receiver() MessageSelector() <RIGHT_SQUARE>
}
void Receiver() : {}
{
LOOKAHEAD(2) Expression() // catches 'self' and 'super' as ident
| TypeDefedIDENT()
}
void TypeDefedIDENT() : {}
{
<BOOL> | <IMP> | <SEL> | <CLASS> | <ID>
}
void MessageSelector() : {}
{
LOOKAHEAD(2) (KeywordArgument())+
| <IDENT>
}
void KeywordArgument() : {}
{
LOOKAHEAD(3) <IDENT> <COLON> Expression()
| <IDENT> <COLON> <NIL>
| LOOKAHEAD(2) <COLON> <NIL>
| <COLON> Expression()
}
void ArgumentExpressionList() : {}
{
AssignmentExpression() ( "," AssignmentExpression() )*
}
void Constant() : {}
{
<INTEGER_LITERAL> | <FLOATING_POINT_LITERAL> | <CHARACTER_LITERAL> | <STRING_LITERAL> | <INTEGER_CONSTANT> | <CSTRING_LITERAL>
}