| %{ |
| #include "aidl_language.h" |
| #include "aidl_language_y.h" |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <string.h> |
| |
| int yylex(yy::parser::semantic_type *, yy::parser::location_type *, void *); |
| |
| #define lex_scanner ps->Scanner() |
| |
| %} |
| |
| %parse-param { Parser* ps } |
| %lex-param { void *lex_scanner } |
| |
| %pure-parser |
| %skeleton "glr.cc" |
| |
| %error-verbose |
| |
| %union { |
| AidlToken* token; |
| int integer; |
| std::string *str; |
| AidlType::Annotation annotation; |
| AidlType::Annotation annotation_list; |
| AidlType* type; |
| AidlType* unannotated_type; |
| AidlArgument* arg; |
| AidlArgument::Direction direction; |
| std::vector<std::unique_ptr<AidlArgument>>* arg_list; |
| AidlMethod* method; |
| AidlMember* constant; |
| std::vector<std::unique_ptr<AidlMember>>* members; |
| AidlQualifiedName* qname; |
| AidlInterface* interface_obj; |
| AidlParcelable* parcelable; |
| AidlDocument* parcelable_list; |
| } |
| |
| %token<token> IDENTIFIER INTERFACE ONEWAY C_STR HEXVALUE |
| %token<integer> INTVALUE |
| |
| %token '(' ')' ',' '=' '[' ']' '<' '>' '.' '{' '}' ';' |
| %token UNKNOWN "unrecognized character" |
| %token IN OUT INOUT PACKAGE IMPORT PARCELABLE CPP_HEADER CONST INT STRING |
| %token ANNOTATION_NULLABLE ANNOTATION_UTF8 ANNOTATION_UTF8_CPP |
| |
| %type<parcelable_list> parcelable_decls |
| %type<parcelable> parcelable_decl |
| %type<members> members |
| %type<interface_obj> interface_decl |
| %type<method> method_decl |
| %type<constant> constant_decl |
| %type<annotation> annotation |
| %type<annotation_list>annotation_list |
| %type<type> type |
| %type<unannotated_type> unannotated_type |
| %type<arg_list> arg_list |
| %type<arg> arg |
| %type<direction> direction |
| %type<str> generic_list |
| %type<qname> qualified_name |
| |
| %type<token> identifier error |
| %% |
| document |
| : package imports parcelable_decls |
| { ps->SetDocument($3); } |
| | package imports interface_decl |
| { ps->SetDocument(new AidlDocument($3)); }; |
| |
| /* A couple of tokens that are keywords elsewhere are identifiers when |
| * occurring in the identifier position. Therefore identifier is a |
| * non-terminal, which is either an IDENTIFIER token, or one of the |
| * aforementioned keyword tokens. |
| */ |
| identifier |
| : IDENTIFIER |
| { $$ = $1; } |
| | CPP_HEADER |
| { $$ = new AidlToken("cpp_header", ""); } |
| | INT |
| { $$ = new AidlToken("int", ""); } |
| | STRING |
| { $$ = new AidlToken("String", ""); } |
| ; |
| |
| package |
| : {} |
| | PACKAGE qualified_name ';' |
| { ps->SetPackage($2); }; |
| |
| imports |
| : {} |
| | import imports {}; |
| |
| import |
| : IMPORT qualified_name ';' |
| { ps->AddImport($2, @1.begin.line); }; |
| |
| qualified_name |
| : identifier { |
| $$ = new AidlQualifiedName($1->GetText(), $1->GetComments()); |
| delete $1; |
| } |
| | qualified_name '.' identifier |
| { $$ = $1; |
| $$->AddTerm($3->GetText()); |
| delete $3; |
| }; |
| |
| parcelable_decls |
| : |
| { $$ = new AidlDocument(); } |
| | parcelable_decls parcelable_decl { |
| $$ = $1; |
| $$->AddParcelable($2); |
| } |
| | parcelable_decls error { |
| ps->AddError(); |
| $$ = $1; |
| }; |
| |
| parcelable_decl |
| : PARCELABLE qualified_name ';' { |
| $$ = new AidlParcelable($2, @2.begin.line, ps->Package()); |
| } |
| | PARCELABLE qualified_name CPP_HEADER C_STR ';' { |
| $$ = new AidlParcelable($2, @2.begin.line, ps->Package(), $4->GetText()); |
| } |
| | PARCELABLE error ';' { |
| ps->AddError(); |
| $$ = NULL; |
| }; |
| |
| interface_decl |
| : annotation_list INTERFACE identifier '{' members '}' { |
| $$ = new AidlInterface($3->GetText(), @2.begin.line, $2->GetComments(), |
| false, $5, ps->Package()); |
| $$->Annotate($1); |
| delete $2; |
| delete $3; |
| } |
| | annotation_list ONEWAY INTERFACE identifier '{' members '}' { |
| $$ = new AidlInterface($4->GetText(), @4.begin.line, $2->GetComments(), |
| true, $6, ps->Package()); |
| $$->Annotate($1); |
| delete $2; |
| delete $3; |
| delete $4; |
| } |
| | annotation_list INTERFACE error '{' members '}' { |
| ps->AddError(); |
| $$ = NULL; |
| delete $2; |
| delete $3; |
| delete $5; |
| }; |
| |
| members |
| : |
| { $$ = new std::vector<std::unique_ptr<AidlMember>>(); } |
| | members method_decl |
| { $1->push_back(std::unique_ptr<AidlMember>($2)); } |
| | members constant_decl |
| { $1->push_back(std::unique_ptr<AidlMember>($2)); } |
| | members error ';' { |
| ps->AddError(); |
| $$ = $1; |
| }; |
| |
| constant_decl |
| : CONST INT identifier '=' INTVALUE ';' { |
| $$ = new AidlIntConstant($3->GetText(), $5); |
| delete $3; |
| } |
| | CONST INT identifier '=' HEXVALUE ';' { |
| $$ = new AidlIntConstant($3->GetText(), $5->GetText(), @5.begin.line); |
| delete $3; |
| } |
| | CONST STRING identifier '=' C_STR ';' { |
| $$ = new AidlStringConstant($3->GetText(), $5->GetText(), @5.begin.line); |
| delete $3; |
| delete $5; |
| } |
| ; |
| |
| method_decl |
| : type identifier '(' arg_list ')' ';' { |
| $$ = new AidlMethod(false, $1, $2->GetText(), $4, @2.begin.line, |
| $1->GetComments()); |
| delete $2; |
| } |
| | ONEWAY type identifier '(' arg_list ')' ';' { |
| $$ = new AidlMethod(true, $2, $3->GetText(), $5, @3.begin.line, |
| $1->GetComments()); |
| delete $1; |
| delete $3; |
| } |
| | type identifier '(' arg_list ')' '=' INTVALUE ';' { |
| $$ = new AidlMethod(false, $1, $2->GetText(), $4, @2.begin.line, |
| $1->GetComments(), $7); |
| delete $2; |
| } |
| | ONEWAY type identifier '(' arg_list ')' '=' INTVALUE ';' { |
| $$ = new AidlMethod(true, $2, $3->GetText(), $5, @3.begin.line, |
| $1->GetComments(), $8); |
| delete $1; |
| delete $3; |
| }; |
| |
| arg_list |
| : |
| { $$ = new std::vector<std::unique_ptr<AidlArgument>>(); } |
| | arg { |
| $$ = new std::vector<std::unique_ptr<AidlArgument>>(); |
| $$->push_back(std::unique_ptr<AidlArgument>($1)); |
| } |
| | arg_list ',' arg { |
| $$ = $1; |
| $$->push_back(std::unique_ptr<AidlArgument>($3)); |
| }; |
| |
| arg |
| : direction type identifier { |
| $$ = new AidlArgument($1, $2, $3->GetText(), @3.begin.line); |
| delete $3; |
| } |
| | type identifier { |
| $$ = new AidlArgument($1, $2->GetText(), @2.begin.line); |
| delete $2; |
| } |
| | error { |
| ps->AddError(); |
| }; |
| |
| unannotated_type |
| : qualified_name { |
| $$ = new AidlType($1->GetDotName(), @1.begin.line, $1->GetComments(), false); |
| delete $1; |
| } |
| | qualified_name '[' ']' { |
| $$ = new AidlType($1->GetDotName(), @1.begin.line, $1->GetComments(), |
| true); |
| delete $1; |
| } |
| | qualified_name '<' generic_list '>' { |
| $$ = new AidlType($1->GetDotName() + "<" + *$3 + ">", @1.begin.line, |
| $1->GetComments(), false); |
| delete $1; |
| delete $3; |
| }; |
| |
| type |
| : annotation_list unannotated_type { |
| $$ = $2; |
| $2->Annotate($1); |
| }; |
| |
| generic_list |
| : qualified_name { |
| $$ = new std::string($1->GetDotName()); |
| delete $1; |
| } |
| | generic_list ',' qualified_name { |
| $$ = new std::string(*$1 + "," + $3->GetDotName()); |
| delete $1; |
| delete $3; |
| }; |
| |
| annotation_list |
| : |
| { $$ = AidlType::AnnotationNone; } |
| | annotation_list annotation |
| { $$ = static_cast<AidlType::Annotation>($1 | $2); }; |
| |
| annotation |
| : ANNOTATION_NULLABLE |
| { $$ = AidlType::AnnotationNullable; } |
| | ANNOTATION_UTF8 |
| { $$ = AidlType::AnnotationUtf8; } |
| | ANNOTATION_UTF8_CPP |
| { $$ = AidlType::AnnotationUtf8InCpp; }; |
| |
| direction |
| : IN |
| { $$ = AidlArgument::IN_DIR; } |
| | OUT |
| { $$ = AidlArgument::OUT_DIR; } |
| | INOUT |
| { $$ = AidlArgument::INOUT_DIR; }; |
| |
| %% |
| |
| #include <ctype.h> |
| #include <stdio.h> |
| |
| void yy::parser::error(const yy::parser::location_type& l, |
| const std::string& errstr) { |
| std::cerr << ps->FileName() << ":" << l << ": " << errstr << std::endl; |
| } |