blob: da5c562f9d7adf87b21e4c3ae30fbb9657e0c8b3 [file] [log] [blame]
Mark Slee31985722006-05-24 21:45:31 +00001%{
David Reissea2cba82009-03-30 21:35:00 +00002/*
3 * Licensed to the Apache Software Foundation (ASF) under one
4 * or more contributor license agreements. See the NOTICE file
5 * distributed with this work for additional information
6 * regarding copyright ownership. The ASF licenses this file
7 * to you under the Apache License, Version 2.0 (the
8 * "License"); you may not use this file except in compliance
9 * with the License. You may obtain a copy of the License at
10 *
11 * http://www.apache.org/licenses/LICENSE-2.0
12 *
13 * Unless required by applicable law or agreed to in writing,
14 * software distributed under the License is distributed on an
15 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16 * KIND, either express or implied. See the License for the
17 * specific language governing permissions and limitations
18 * under the License.
19 */
Mark Slee31985722006-05-24 21:45:31 +000020
21/**
22 * Thrift parser.
23 *
24 * This parser is used on a thrift definition file.
25 *
Mark Slee31985722006-05-24 21:45:31 +000026 */
27
David Reissf1454162008-06-30 20:45:47 +000028#define __STDC_LIMIT_MACROS
29#define __STDC_FORMAT_MACROS
Mark Slee31985722006-05-24 21:45:31 +000030#include <stdio.h>
Roger Meier12d70532011-12-14 23:35:28 +000031#ifndef _MSC_VER
David Reissf1454162008-06-30 20:45:47 +000032#include <inttypes.h>
Roger Meier12d70532011-12-14 23:35:28 +000033#else
34#include <stdint.h>
35#endif
David Reiss400a5432008-07-25 19:48:39 +000036#include <limits.h>
Ben Craige9576752013-10-11 08:19:16 -050037#ifdef _MSC_VER
38#include "windows/config.h"
39#endif
Mark Slee31985722006-05-24 21:45:31 +000040#include "main.h"
41#include "globals.h"
42#include "parse/t_program.h"
Mark Sleef0712dc2006-10-25 19:03:57 +000043#include "parse/t_scope.h"
Mark Slee31985722006-05-24 21:45:31 +000044
Ben Craige9576752013-10-11 08:19:16 -050045#ifdef _MSC_VER
46//warning C4065: switch statement contains 'default' but no 'case' labels
47#pragma warning(disable:4065)
48#endif
49
Mark Sleef5377b32006-10-10 01:42:59 +000050/**
51 * This global variable is used for automatic numbering of field indices etc.
52 * when parsing the members of a struct. Field values are automatically
53 * assigned starting from -1 and working their way down.
54 */
Mark Slee9cb7c612006-09-01 22:17:45 +000055int y_field_val = -1;
Mark Slee78165722007-09-10 22:08:49 +000056int g_arglist = 0;
Bryan Duxburyab3666e2009-09-01 23:03:47 +000057const int struct_is_struct = 0;
58const int struct_is_union = 1;
Mark Slee31985722006-05-24 21:45:31 +000059
60%}
61
Mark Sleef5377b32006-10-10 01:42:59 +000062/**
63 * This structure is used by the parser to hold the data types associated with
64 * various parse nodes.
65 */
Mark Slee31985722006-05-24 21:45:31 +000066%union {
Mark Slee30152872006-11-28 01:24:07 +000067 char* id;
David Reissf1454162008-06-30 20:45:47 +000068 int64_t iconst;
Mark Slee30152872006-11-28 01:24:07 +000069 double dconst;
70 bool tbool;
David Reisscdffe262007-08-14 17:12:31 +000071 t_doc* tdoc;
Mark Slee30152872006-11-28 01:24:07 +000072 t_type* ttype;
Mark Slee6a47fed2007-02-07 02:40:59 +000073 t_base_type* tbase;
Mark Slee30152872006-11-28 01:24:07 +000074 t_typedef* ttypedef;
75 t_enum* tenum;
76 t_enum_value* tenumv;
77 t_const* tconst;
78 t_const_value* tconstv;
79 t_struct* tstruct;
80 t_service* tservice;
81 t_function* tfunction;
82 t_field* tfield;
David Reisscdffe262007-08-14 17:12:31 +000083 char* dtext;
David Reiss8320a922007-08-14 19:59:26 +000084 t_field::e_req ereq;
David Reissa2309992008-12-10 01:52:48 +000085 t_annotation* tannot;
Bryan Duxburyc7206a42011-08-17 23:17:04 +000086 t_field_id tfieldid;
Mark Slee31985722006-05-24 21:45:31 +000087}
88
Mark Sleef5377b32006-10-10 01:42:59 +000089/**
90 * Strings identifier
91 */
Mark Slee31985722006-05-24 21:45:31 +000092%token<id> tok_identifier
Mark Sleef0712dc2006-10-25 19:03:57 +000093%token<id> tok_literal
David Reisscdffe262007-08-14 17:12:31 +000094%token<dtext> tok_doctext
Mark Sleebd588222007-11-21 08:43:35 +000095%token<id> tok_st_identifier
Mark Sleef5377b32006-10-10 01:42:59 +000096
97/**
Mark Slee30152872006-11-28 01:24:07 +000098 * Constant values
Mark Sleef5377b32006-10-10 01:42:59 +000099 */
Mark Slee31985722006-05-24 21:45:31 +0000100%token<iconst> tok_int_constant
Mark Slee30152872006-11-28 01:24:07 +0000101%token<dconst> tok_dub_constant
Mark Slee31985722006-05-24 21:45:31 +0000102
Mark Sleef5377b32006-10-10 01:42:59 +0000103/**
David Reiss399442b2008-02-20 02:28:05 +0000104 * Header keywords
Mark Sleef5377b32006-10-10 01:42:59 +0000105 */
Mark Sleef0712dc2006-10-25 19:03:57 +0000106%token tok_include
Mark Slee9cb7c612006-09-01 22:17:45 +0000107%token tok_namespace
Mark Sleef0712dc2006-10-25 19:03:57 +0000108%token tok_cpp_namespace
109%token tok_cpp_include
110%token tok_cpp_type
Mark Sleee888b372007-01-12 01:06:24 +0000111%token tok_php_namespace
David Reissc6fc3292007-08-30 00:58:43 +0000112%token tok_py_module
Mark Slee27ed6ec2007-08-16 01:26:31 +0000113%token tok_perl_package
Mark Sleef0712dc2006-10-25 19:03:57 +0000114%token tok_java_package
Mark Slee782abbb2007-01-19 00:17:02 +0000115%token tok_xsd_all
Mark Slee36bfa2e2007-01-19 20:09:51 +0000116%token tok_xsd_optional
Mark Slee7df0e2a2007-02-06 21:03:18 +0000117%token tok_xsd_nillable
Mark Slee0d9199e2007-01-31 02:08:30 +0000118%token tok_xsd_namespace
Mark Slee21135c32007-02-05 21:52:08 +0000119%token tok_xsd_attrs
Mark Slee58dfb4f2007-07-06 02:45:25 +0000120%token tok_ruby_namespace
Mark Sleebd588222007-11-21 08:43:35 +0000121%token tok_smalltalk_category
David Reiss15457c92007-12-14 07:03:03 +0000122%token tok_smalltalk_prefix
Mark Slee7e9eea42007-09-10 21:00:23 +0000123%token tok_cocoa_prefix
David Reiss7f42bcf2008-01-11 20:59:12 +0000124%token tok_csharp_namespace
Jake Farrell7ae13e12011-10-18 14:35:26 +0000125%token tok_delphi_namespace
Mark Slee9cb7c612006-09-01 22:17:45 +0000126
Mark Sleef5377b32006-10-10 01:42:59 +0000127/**
128 * Base datatype keywords
129 */
130%token tok_void
Mark Slee78f58e22006-09-02 04:17:07 +0000131%token tok_bool
Mark Slee31985722006-05-24 21:45:31 +0000132%token tok_byte
133%token tok_string
Mark Slee8d725a22007-04-13 01:57:12 +0000134%token tok_binary
Mark Sleeb6200d82007-01-19 19:14:36 +0000135%token tok_slist
Mark Slee6a47fed2007-02-07 02:40:59 +0000136%token tok_senum
Mark Slee9cb7c612006-09-01 22:17:45 +0000137%token tok_i16
Mark Slee31985722006-05-24 21:45:31 +0000138%token tok_i32
Mark Slee31985722006-05-24 21:45:31 +0000139%token tok_i64
Mark Slee9cb7c612006-09-01 22:17:45 +0000140%token tok_double
Mark Slee31985722006-05-24 21:45:31 +0000141
Mark Sleef5377b32006-10-10 01:42:59 +0000142/**
143 * Complex type keywords
144 */
Mark Slee31985722006-05-24 21:45:31 +0000145%token tok_map
146%token tok_list
147%token tok_set
148
Mark Sleef5377b32006-10-10 01:42:59 +0000149/**
150 * Function modifiers
Mark Slee27ed6ec2007-08-16 01:26:31 +0000151 */
David Reiss6985a422009-03-24 20:00:47 +0000152%token tok_oneway
Mark Slee31985722006-05-24 21:45:31 +0000153
Mark Sleef5377b32006-10-10 01:42:59 +0000154/**
155 * Thrift language keywords
156 */
Mark Slee31985722006-05-24 21:45:31 +0000157%token tok_typedef
158%token tok_struct
Mark Slee9cb7c612006-09-01 22:17:45 +0000159%token tok_xception
160%token tok_throws
Mark Sleef0712dc2006-10-25 19:03:57 +0000161%token tok_extends
Mark Slee31985722006-05-24 21:45:31 +0000162%token tok_service
163%token tok_enum
Mark Slee30152872006-11-28 01:24:07 +0000164%token tok_const
David Reiss8320a922007-08-14 19:59:26 +0000165%token tok_required
166%token tok_optional
Bryan Duxburyab3666e2009-09-01 23:03:47 +0000167%token tok_union
Jens Geyer885c6792014-05-02 21:31:55 +0200168%token tok_reference
Mark Slee31985722006-05-24 21:45:31 +0000169
Mark Sleef5377b32006-10-10 01:42:59 +0000170/**
171 * Grammar nodes
172 */
173
Mark Slee31985722006-05-24 21:45:31 +0000174%type<ttype> BaseType
David Reissc8e30052009-07-27 17:02:42 +0000175%type<ttype> SimpleBaseType
Mark Sleee8540632006-05-30 09:24:40 +0000176%type<ttype> ContainerType
David Reissa2309992008-12-10 01:52:48 +0000177%type<ttype> SimpleContainerType
Mark Sleee8540632006-05-30 09:24:40 +0000178%type<ttype> MapType
179%type<ttype> SetType
180%type<ttype> ListType
Mark Slee31985722006-05-24 21:45:31 +0000181
David Reisscdffe262007-08-14 17:12:31 +0000182%type<tdoc> Definition
Mark Sleef0712dc2006-10-25 19:03:57 +0000183%type<ttype> TypeDefinition
184
Mark Slee31985722006-05-24 21:45:31 +0000185%type<ttypedef> Typedef
Mark Slee31985722006-05-24 21:45:31 +0000186
David Reissa2309992008-12-10 01:52:48 +0000187%type<ttype> TypeAnnotations
188%type<ttype> TypeAnnotationList
189%type<tannot> TypeAnnotation
Jens Geyer89847df2014-05-02 23:50:04 +0200190%type<id> TypeAnnotationValue
David Reissa2309992008-12-10 01:52:48 +0000191
Mark Slee31985722006-05-24 21:45:31 +0000192%type<tfield> Field
Bryan Duxburyc7206a42011-08-17 23:17:04 +0000193%type<tfieldid> FieldIdentifier
David Reiss8320a922007-08-14 19:59:26 +0000194%type<ereq> FieldRequiredness
Mark Slee31985722006-05-24 21:45:31 +0000195%type<ttype> FieldType
Mark Slee7ff32452007-02-01 05:26:18 +0000196%type<tconstv> FieldValue
Mark Sleee8540632006-05-30 09:24:40 +0000197%type<tstruct> FieldList
Jens Geyer885c6792014-05-02 21:31:55 +0200198%type<tbool> FieldReference
Mark Slee31985722006-05-24 21:45:31 +0000199
200%type<tenum> Enum
201%type<tenum> EnumDefList
Mark Slee30152872006-11-28 01:24:07 +0000202%type<tenumv> EnumDef
203
Mark Slee6a47fed2007-02-07 02:40:59 +0000204%type<ttypedef> Senum
205%type<tbase> SenumDefList
206%type<id> SenumDef
207
Mark Slee30152872006-11-28 01:24:07 +0000208%type<tconst> Const
209%type<tconstv> ConstValue
210%type<tconstv> ConstList
211%type<tconstv> ConstListContents
212%type<tconstv> ConstMap
213%type<tconstv> ConstMapContents
Mark Slee31985722006-05-24 21:45:31 +0000214
Bryan Duxburyab3666e2009-09-01 23:03:47 +0000215%type<iconst> StructHead
Mark Slee31985722006-05-24 21:45:31 +0000216%type<tstruct> Struct
Mark Slee9cb7c612006-09-01 22:17:45 +0000217%type<tstruct> Xception
Mark Slee31985722006-05-24 21:45:31 +0000218%type<tservice> Service
219
220%type<tfunction> Function
Mark Slee31985722006-05-24 21:45:31 +0000221%type<ttype> FunctionType
222%type<tservice> FunctionList
223
Mark Slee36bfa2e2007-01-19 20:09:51 +0000224%type<tstruct> Throws
225%type<tservice> Extends
David Reiss6985a422009-03-24 20:00:47 +0000226%type<tbool> Oneway
Mark Slee36bfa2e2007-01-19 20:09:51 +0000227%type<tbool> XsdAll
228%type<tbool> XsdOptional
Mark Slee7df0e2a2007-02-06 21:03:18 +0000229%type<tbool> XsdNillable
Mark Slee748d83f2007-02-07 01:20:08 +0000230%type<tstruct> XsdAttributes
Mark Slee36bfa2e2007-01-19 20:09:51 +0000231%type<id> CppType
Mark Slee52f643d2006-08-09 00:03:43 +0000232
David Reisscbd4bac2007-08-14 17:12:33 +0000233%type<dtext> CaptureDocText
ccheeverf53b5cf2007-02-05 20:33:11 +0000234
Mark Slee31985722006-05-24 21:45:31 +0000235%%
236
Mark Sleef5377b32006-10-10 01:42:59 +0000237/**
238 * Thrift Grammar Implementation.
239 *
240 * For the most part this source file works its way top down from what you
241 * might expect to find in a typical .thrift file, i.e. type definitions and
242 * namespaces up top followed by service definitions using those types.
243 */
Mark Slee31985722006-05-24 21:45:31 +0000244
245Program:
David Reisscbd4bac2007-08-14 17:12:33 +0000246 HeaderList DefinitionList
Mark Sleef0712dc2006-10-25 19:03:57 +0000247 {
248 pdebug("Program -> Headers DefinitionList");
Jens Geyere8379b52014-01-25 00:59:45 +0100249 if((g_program_doctext_candidate != NULL) && (g_program_doctext_status != ALREADY_PROCESSED))
250 {
251 g_program->set_doc(g_program_doctext_candidate);
252 g_program_doctext_status = ALREADY_PROCESSED;
David Reissc2532a92007-07-30 23:46:11 +0000253 }
David Reisscbd4bac2007-08-14 17:12:33 +0000254 clear_doctext();
Mark Sleef0712dc2006-10-25 19:03:57 +0000255 }
256
David Reisscbd4bac2007-08-14 17:12:33 +0000257CaptureDocText:
258 {
259 if (g_parse_mode == PROGRAM) {
Mark Sleebd588222007-11-21 08:43:35 +0000260 $$ = g_doctext;
David Reisscbd4bac2007-08-14 17:12:33 +0000261 g_doctext = NULL;
Mark Slee27ed6ec2007-08-16 01:26:31 +0000262 } else {
David Reisscbd4bac2007-08-14 17:12:33 +0000263 $$ = NULL;
264 }
265 }
266
267/* TODO(dreiss): Try to DestroyDocText in all sorts or random places. */
268DestroyDocText:
269 {
270 if (g_parse_mode == PROGRAM) {
271 clear_doctext();
272 }
273 }
274
275/* We have to DestroyDocText here, otherwise it catches the doctext
276 on the first real element. */
Mark Sleef0712dc2006-10-25 19:03:57 +0000277HeaderList:
David Reisscbd4bac2007-08-14 17:12:33 +0000278 HeaderList DestroyDocText Header
Mark Sleef0712dc2006-10-25 19:03:57 +0000279 {
280 pdebug("HeaderList -> HeaderList Header");
281 }
282|
283 {
284 pdebug("HeaderList -> ");
285 }
286
287Header:
288 Include
289 {
290 pdebug("Header -> Include");
291 }
David Reiss79eca142008-02-27 01:55:13 +0000292| tok_namespace tok_identifier tok_identifier
Mark Sleef0712dc2006-10-25 19:03:57 +0000293 {
David Reiss79eca142008-02-27 01:55:13 +0000294 pdebug("Header -> tok_namespace tok_identifier tok_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100295 declare_valid_program_doctext();
Mark Sleef0712dc2006-10-25 19:03:57 +0000296 if (g_parse_mode == PROGRAM) {
David Reiss79eca142008-02-27 01:55:13 +0000297 g_program->set_namespace($2, $3);
Mark Sleef0712dc2006-10-25 19:03:57 +0000298 }
299 }
David Reissfb790d72010-09-02 16:41:45 +0000300| tok_namespace '*' tok_identifier
301 {
302 pdebug("Header -> tok_namespace * tok_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100303 declare_valid_program_doctext();
David Reissfb790d72010-09-02 16:41:45 +0000304 if (g_parse_mode == PROGRAM) {
305 g_program->set_namespace("*", $3);
306 }
307 }
David Reiss9a08dc62008-02-27 01:55:17 +0000308/* TODO(dreiss): Get rid of this once everyone is using the new hotness. */
Mark Sleef0712dc2006-10-25 19:03:57 +0000309| tok_cpp_namespace tok_identifier
310 {
David Reiss9a08dc62008-02-27 01:55:17 +0000311 pwarning(1, "'cpp_namespace' is deprecated. Use 'namespace cpp' instead");
Mark Sleef0712dc2006-10-25 19:03:57 +0000312 pdebug("Header -> tok_cpp_namespace tok_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100313 declare_valid_program_doctext();
Mark Sleef0712dc2006-10-25 19:03:57 +0000314 if (g_parse_mode == PROGRAM) {
David Reiss9a08dc62008-02-27 01:55:17 +0000315 g_program->set_namespace("cpp", $2);
Mark Sleef0712dc2006-10-25 19:03:57 +0000316 }
317 }
318| tok_cpp_include tok_literal
319 {
320 pdebug("Header -> tok_cpp_include tok_literal");
Jens Geyere8379b52014-01-25 00:59:45 +0100321 declare_valid_program_doctext();
Mark Sleef0712dc2006-10-25 19:03:57 +0000322 if (g_parse_mode == PROGRAM) {
323 g_program->add_cpp_include($2);
324 }
325 }
Mark Sleee888b372007-01-12 01:06:24 +0000326| tok_php_namespace tok_identifier
327 {
David Reiss554ea6f2009-02-17 20:28:37 +0000328 pwarning(1, "'php_namespace' is deprecated. Use 'namespace php' instead");
Mark Sleee888b372007-01-12 01:06:24 +0000329 pdebug("Header -> tok_php_namespace tok_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100330 declare_valid_program_doctext();
Mark Sleee888b372007-01-12 01:06:24 +0000331 if (g_parse_mode == PROGRAM) {
David Reiss554ea6f2009-02-17 20:28:37 +0000332 g_program->set_namespace("php", $2);
Mark Sleee888b372007-01-12 01:06:24 +0000333 }
334 }
David Reiss320e45c2008-03-27 21:41:54 +0000335/* TODO(dreiss): Get rid of this once everyone is using the new hotness. */
David Reissc6fc3292007-08-30 00:58:43 +0000336| tok_py_module tok_identifier
337 {
David Reiss320e45c2008-03-27 21:41:54 +0000338 pwarning(1, "'py_module' is deprecated. Use 'namespace py' instead");
David Reissc6fc3292007-08-30 00:58:43 +0000339 pdebug("Header -> tok_py_module tok_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100340 declare_valid_program_doctext();
David Reissc6fc3292007-08-30 00:58:43 +0000341 if (g_parse_mode == PROGRAM) {
David Reiss320e45c2008-03-27 21:41:54 +0000342 g_program->set_namespace("py", $2);
David Reissc6fc3292007-08-30 00:58:43 +0000343 }
344 }
David Reiss07ef3a92008-03-27 21:42:39 +0000345/* TODO(dreiss): Get rid of this once everyone is using the new hotness. */
Mark Slee27ed6ec2007-08-16 01:26:31 +0000346| tok_perl_package tok_identifier
347 {
David Reiss07ef3a92008-03-27 21:42:39 +0000348 pwarning(1, "'perl_package' is deprecated. Use 'namespace perl' instead");
Mark Slee27ed6ec2007-08-16 01:26:31 +0000349 pdebug("Header -> tok_perl_namespace tok_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100350 declare_valid_program_doctext();
Mark Slee27ed6ec2007-08-16 01:26:31 +0000351 if (g_parse_mode == PROGRAM) {
David Reiss07ef3a92008-03-27 21:42:39 +0000352 g_program->set_namespace("perl", $2);
Mark Slee27ed6ec2007-08-16 01:26:31 +0000353 }
354 }
David Reiss6a4b82c2008-03-27 21:42:16 +0000355/* TODO(dreiss): Get rid of this once everyone is using the new hotness. */
Mark Slee58dfb4f2007-07-06 02:45:25 +0000356| tok_ruby_namespace tok_identifier
357 {
David Reiss6a4b82c2008-03-27 21:42:16 +0000358 pwarning(1, "'ruby_namespace' is deprecated. Use 'namespace rb' instead");
Mark Slee58dfb4f2007-07-06 02:45:25 +0000359 pdebug("Header -> tok_ruby_namespace tok_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100360 declare_valid_program_doctext();
Mark Slee58dfb4f2007-07-06 02:45:25 +0000361 if (g_parse_mode == PROGRAM) {
David Reiss6a4b82c2008-03-27 21:42:16 +0000362 g_program->set_namespace("rb", $2);
Mark Slee58dfb4f2007-07-06 02:45:25 +0000363 }
364 }
David Reiss3b455012008-03-27 21:40:46 +0000365/* TODO(dreiss): Get rid of this once everyone is using the new hotness. */
Mark Sleebd588222007-11-21 08:43:35 +0000366| tok_smalltalk_category tok_st_identifier
367 {
David Reiss3b455012008-03-27 21:40:46 +0000368 pwarning(1, "'smalltalk_category' is deprecated. Use 'namespace smalltalk.category' instead");
Mark Sleebd588222007-11-21 08:43:35 +0000369 pdebug("Header -> tok_smalltalk_category tok_st_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100370 declare_valid_program_doctext();
Mark Sleebd588222007-11-21 08:43:35 +0000371 if (g_parse_mode == PROGRAM) {
David Reiss3b455012008-03-27 21:40:46 +0000372 g_program->set_namespace("smalltalk.category", $2);
Mark Sleebd588222007-11-21 08:43:35 +0000373 }
374 }
David Reiss3b455012008-03-27 21:40:46 +0000375/* TODO(dreiss): Get rid of this once everyone is using the new hotness. */
David Reiss15457c92007-12-14 07:03:03 +0000376| tok_smalltalk_prefix tok_identifier
377 {
David Reiss3b455012008-03-27 21:40:46 +0000378 pwarning(1, "'smalltalk_prefix' is deprecated. Use 'namespace smalltalk.prefix' instead");
David Reiss15457c92007-12-14 07:03:03 +0000379 pdebug("Header -> tok_smalltalk_prefix tok_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100380 declare_valid_program_doctext();
David Reiss15457c92007-12-14 07:03:03 +0000381 if (g_parse_mode == PROGRAM) {
David Reiss3b455012008-03-27 21:40:46 +0000382 g_program->set_namespace("smalltalk.prefix", $2);
David Reiss15457c92007-12-14 07:03:03 +0000383 }
384 }
David Reiss771f8c72008-02-27 01:55:25 +0000385/* TODO(dreiss): Get rid of this once everyone is using the new hotness. */
Mark Sleef0712dc2006-10-25 19:03:57 +0000386| tok_java_package tok_identifier
387 {
David Reiss9f646152008-03-02 21:59:48 +0000388 pwarning(1, "'java_package' is deprecated. Use 'namespace java' instead");
Mark Sleef0712dc2006-10-25 19:03:57 +0000389 pdebug("Header -> tok_java_package tok_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100390 declare_valid_program_doctext();
Mark Sleef0712dc2006-10-25 19:03:57 +0000391 if (g_parse_mode == PROGRAM) {
David Reiss771f8c72008-02-27 01:55:25 +0000392 g_program->set_namespace("java", $2);
Mark Sleef0712dc2006-10-25 19:03:57 +0000393 }
394 }
David Reiss54b602b2008-03-27 21:41:06 +0000395/* TODO(dreiss): Get rid of this once everyone is using the new hotness. */
Mark Slee7e9eea42007-09-10 21:00:23 +0000396| tok_cocoa_prefix tok_identifier
397 {
David Reiss54b602b2008-03-27 21:41:06 +0000398 pwarning(1, "'cocoa_prefix' is deprecated. Use 'namespace cocoa' instead");
Mark Slee7e9eea42007-09-10 21:00:23 +0000399 pdebug("Header -> tok_cocoa_prefix tok_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100400 declare_valid_program_doctext();
Mark Slee7e9eea42007-09-10 21:00:23 +0000401 if (g_parse_mode == PROGRAM) {
David Reiss54b602b2008-03-27 21:41:06 +0000402 g_program->set_namespace("cocoa", $2);
Mark Slee7e9eea42007-09-10 21:00:23 +0000403 }
404 }
David Reiss92e10d82009-02-17 20:28:19 +0000405/* TODO(dreiss): Get rid of this once everyone is using the new hotness. */
Mark Slee0d9199e2007-01-31 02:08:30 +0000406| tok_xsd_namespace tok_literal
407 {
David Reiss92e10d82009-02-17 20:28:19 +0000408 pwarning(1, "'xsd_namespace' is deprecated. Use 'namespace xsd' instead");
Mark Slee0d9199e2007-01-31 02:08:30 +0000409 pdebug("Header -> tok_xsd_namespace tok_literal");
Jens Geyere8379b52014-01-25 00:59:45 +0100410 declare_valid_program_doctext();
Mark Slee0d9199e2007-01-31 02:08:30 +0000411 if (g_parse_mode == PROGRAM) {
David Reiss92e10d82009-02-17 20:28:19 +0000412 g_program->set_namespace("cocoa", $2);
Mark Slee0d9199e2007-01-31 02:08:30 +0000413 }
414 }
David Reiss9d65bf02008-03-27 21:41:37 +0000415/* TODO(dreiss): Get rid of this once everyone is using the new hotness. */
David Reiss7f42bcf2008-01-11 20:59:12 +0000416| tok_csharp_namespace tok_identifier
417 {
David Reiss9d65bf02008-03-27 21:41:37 +0000418 pwarning(1, "'csharp_namespace' is deprecated. Use 'namespace csharp' instead");
David Reiss919ae802008-03-27 21:41:11 +0000419 pdebug("Header -> tok_csharp_namespace tok_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100420 declare_valid_program_doctext();
David Reiss7f42bcf2008-01-11 20:59:12 +0000421 if (g_parse_mode == PROGRAM) {
David Reiss9d65bf02008-03-27 21:41:37 +0000422 g_program->set_namespace("csharp", $2);
David Reiss7f42bcf2008-01-11 20:59:12 +0000423 }
424 }
Jake Farrell7ae13e12011-10-18 14:35:26 +0000425/* TODO(dreiss): Get rid of this once everyone is using the new hotness. */
426| tok_delphi_namespace tok_identifier
427 {
428 pwarning(1, "'delphi_namespace' is deprecated. Use 'namespace delphi' instead");
429 pdebug("Header -> tok_delphi_namespace tok_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100430 declare_valid_program_doctext();
Jake Farrell7ae13e12011-10-18 14:35:26 +0000431 if (g_parse_mode == PROGRAM) {
432 g_program->set_namespace("delphi", $2);
433 }
434 }
Mark Sleef0712dc2006-10-25 19:03:57 +0000435
436Include:
437 tok_include tok_literal
438 {
Mark Slee27ed6ec2007-08-16 01:26:31 +0000439 pdebug("Include -> tok_include tok_literal");
Jens Geyere8379b52014-01-25 00:59:45 +0100440 declare_valid_program_doctext();
Mark Sleef0712dc2006-10-25 19:03:57 +0000441 if (g_parse_mode == INCLUDES) {
442 std::string path = include_file(std::string($2));
443 if (!path.empty()) {
kholst76f2c882008-01-16 02:47:41 +0000444 g_program->add_include(path, std::string($2));
Mark Sleef0712dc2006-10-25 19:03:57 +0000445 }
446 }
447 }
Mark Slee31985722006-05-24 21:45:31 +0000448
449DefinitionList:
David Reisscbd4bac2007-08-14 17:12:33 +0000450 DefinitionList CaptureDocText Definition
Mark Slee31985722006-05-24 21:45:31 +0000451 {
452 pdebug("DefinitionList -> DefinitionList Definition");
David Reisscdffe262007-08-14 17:12:31 +0000453 if ($2 != NULL && $3 != NULL) {
454 $3->set_doc($2);
455 }
Mark Slee31985722006-05-24 21:45:31 +0000456 }
457|
458 {
459 pdebug("DefinitionList -> ");
460 }
461
462Definition:
Mark Slee30152872006-11-28 01:24:07 +0000463 Const
464 {
465 pdebug("Definition -> Const");
466 if (g_parse_mode == PROGRAM) {
467 g_program->add_const($1);
Mark Sleebd588222007-11-21 08:43:35 +0000468 }
David Reisscdffe262007-08-14 17:12:31 +0000469 $$ = $1;
Mark Slee30152872006-11-28 01:24:07 +0000470 }
471| TypeDefinition
Mark Slee9cb7c612006-09-01 22:17:45 +0000472 {
Mark Sleef0712dc2006-10-25 19:03:57 +0000473 pdebug("Definition -> TypeDefinition");
474 if (g_parse_mode == PROGRAM) {
475 g_scope->add_type($1->get_name(), $1);
476 if (g_parent_scope != NULL) {
477 g_parent_scope->add_type(g_parent_prefix + $1->get_name(), $1);
478 }
Roger Meierba406d32013-07-15 22:41:34 +0200479 if (! g_program->is_unique_typename($1)) {
480 yyerror("Type \"%s\" is already defined.", $1->get_name().c_str());
481 exit(1);
482 }
Mark Sleef0712dc2006-10-25 19:03:57 +0000483 }
David Reisscdffe262007-08-14 17:12:31 +0000484 $$ = $1;
Mark Slee9cb7c612006-09-01 22:17:45 +0000485 }
Mark Slee31985722006-05-24 21:45:31 +0000486| Service
487 {
488 pdebug("Definition -> Service");
Mark Sleef0712dc2006-10-25 19:03:57 +0000489 if (g_parse_mode == PROGRAM) {
490 g_scope->add_service($1->get_name(), $1);
491 if (g_parent_scope != NULL) {
492 g_parent_scope->add_service(g_parent_prefix + $1->get_name(), $1);
493 }
494 g_program->add_service($1);
Roger Meierba406d32013-07-15 22:41:34 +0200495 if (! g_program->is_unique_typename($1)) {
496 yyerror("Type \"%s\" is already defined.", $1->get_name().c_str());
497 exit(1);
498 }
Mark Sleef0712dc2006-10-25 19:03:57 +0000499 }
David Reisscdffe262007-08-14 17:12:31 +0000500 $$ = $1;
Mark Slee9cb7c612006-09-01 22:17:45 +0000501 }
502
Mark Sleef0712dc2006-10-25 19:03:57 +0000503TypeDefinition:
504 Typedef
Mark Slee9cb7c612006-09-01 22:17:45 +0000505 {
Mark Sleef0712dc2006-10-25 19:03:57 +0000506 pdebug("TypeDefinition -> Typedef");
507 if (g_parse_mode == PROGRAM) {
508 g_program->add_typedef($1);
509 }
510 }
511| Enum
512 {
513 pdebug("TypeDefinition -> Enum");
514 if (g_parse_mode == PROGRAM) {
515 g_program->add_enum($1);
516 }
517 }
Mark Slee6a47fed2007-02-07 02:40:59 +0000518| Senum
519 {
520 pdebug("TypeDefinition -> Senum");
521 if (g_parse_mode == PROGRAM) {
522 g_program->add_typedef($1);
523 }
524 }
Mark Sleef0712dc2006-10-25 19:03:57 +0000525| Struct
526 {
527 pdebug("TypeDefinition -> Struct");
528 if (g_parse_mode == PROGRAM) {
529 g_program->add_struct($1);
530 }
531 }
532| Xception
Mark Slee27ed6ec2007-08-16 01:26:31 +0000533 {
Mark Sleef0712dc2006-10-25 19:03:57 +0000534 pdebug("TypeDefinition -> Xception");
535 if (g_parse_mode == PROGRAM) {
536 g_program->add_xception($1);
537 }
Mark Slee9cb7c612006-09-01 22:17:45 +0000538 }
Mark Slee31985722006-05-24 21:45:31 +0000539
540Typedef:
Roger Meier30877382012-09-17 21:18:05 +0000541 tok_typedef FieldType tok_identifier TypeAnnotations
Mark Slee31985722006-05-24 21:45:31 +0000542 {
David Reiss4dd78012010-03-09 05:19:08 +0000543 pdebug("TypeDef -> tok_typedef FieldType tok_identifier");
Jens Geyer12c09f42013-08-25 14:16:27 +0200544 validate_simple_identifier( $3);
David Reisscdffe262007-08-14 17:12:31 +0000545 t_typedef *td = new t_typedef(g_program, $2, $3);
Mark Slee31985722006-05-24 21:45:31 +0000546 $$ = td;
Roger Meier30877382012-09-17 21:18:05 +0000547 if ($4 != NULL) {
548 $$->annotations_ = $4->annotations_;
549 delete $4;
550 }
Mark Slee31985722006-05-24 21:45:31 +0000551 }
552
Mark Slee6a47fed2007-02-07 02:40:59 +0000553CommaOrSemicolonOptional:
554 ','
555 {}
556| ';'
557 {}
558|
559 {}
ccheeverf53b5cf2007-02-05 20:33:11 +0000560
Mark Slee31985722006-05-24 21:45:31 +0000561Enum:
Roger Meier30877382012-09-17 21:18:05 +0000562 tok_enum tok_identifier '{' EnumDefList '}' TypeAnnotations
Mark Slee31985722006-05-24 21:45:31 +0000563 {
564 pdebug("Enum -> tok_enum tok_identifier { EnumDefList }");
David Reisscdffe262007-08-14 17:12:31 +0000565 $$ = $4;
Jens Geyer12c09f42013-08-25 14:16:27 +0200566 validate_simple_identifier( $2);
David Reisscdffe262007-08-14 17:12:31 +0000567 $$->set_name($2);
Roger Meier30877382012-09-17 21:18:05 +0000568 if ($6 != NULL) {
569 $$->annotations_ = $6->annotations_;
570 delete $6;
571 }
Bryan Duxbury2d804702009-12-18 19:41:11 +0000572 $$->resolve_values();
Bryan Duxbury9f0a7862010-09-12 14:38:36 +0000573 // make constants for all the enum values
574 if (g_parse_mode == PROGRAM) {
575 const std::vector<t_enum_value*>& enum_values = $$->get_constants();
576 std::vector<t_enum_value*>::const_iterator c_iter;
577 for (c_iter = enum_values.begin(); c_iter != enum_values.end(); ++c_iter) {
578 std::string const_name = $$->get_name() + "." + (*c_iter)->get_name();
579 t_const_value* const_val = new t_const_value((*c_iter)->get_value());
580 const_val->set_enum($$);
581 g_scope->add_constant(const_name, new t_const(g_type_i32, (*c_iter)->get_name(), const_val));
582 if (g_parent_scope != NULL) {
583 g_parent_scope->add_constant(g_parent_prefix + const_name, new t_const(g_type_i32, (*c_iter)->get_name(), const_val));
584 }
585 }
586 }
Mark Slee31985722006-05-24 21:45:31 +0000587 }
588
589EnumDefList:
Mark Slee207cb462006-11-02 18:43:12 +0000590 EnumDefList EnumDef
Mark Slee31985722006-05-24 21:45:31 +0000591 {
592 pdebug("EnumDefList -> EnumDefList EnumDef");
593 $$ = $1;
Mark Slee207cb462006-11-02 18:43:12 +0000594 $$->append($2);
Mark Slee31985722006-05-24 21:45:31 +0000595 }
596|
597 {
598 pdebug("EnumDefList -> ");
Mark Sleef0712dc2006-10-25 19:03:57 +0000599 $$ = new t_enum(g_program);
Mark Slee31985722006-05-24 21:45:31 +0000600 }
601
602EnumDef:
Roger Meier30877382012-09-17 21:18:05 +0000603 CaptureDocText tok_identifier '=' tok_int_constant TypeAnnotations CommaOrSemicolonOptional
Mark Slee31985722006-05-24 21:45:31 +0000604 {
Mark Slee30152872006-11-28 01:24:07 +0000605 pdebug("EnumDef -> tok_identifier = tok_int_constant");
ccheeverf53b5cf2007-02-05 20:33:11 +0000606 if ($4 < 0) {
607 pwarning(1, "Negative value supplied for enum %s.\n", $2);
Mark Slee31985722006-05-24 21:45:31 +0000608 }
David Reissf1454162008-06-30 20:45:47 +0000609 if ($4 > INT_MAX) {
610 pwarning(1, "64-bit value supplied for enum %s.\n", $2);
611 }
Jens Geyer12c09f42013-08-25 14:16:27 +0200612 validate_simple_identifier( $2);
Ben Craige9576752013-10-11 08:19:16 -0500613 $$ = new t_enum_value($2, static_cast<int>($4));
ccheeverf53b5cf2007-02-05 20:33:11 +0000614 if ($1 != NULL) {
615 $$->set_doc($1);
616 }
Roger Meier30877382012-09-17 21:18:05 +0000617 if ($5 != NULL) {
618 $$->annotations_ = $5->annotations_;
619 delete $5;
620 }
Mark Slee31985722006-05-24 21:45:31 +0000621 }
622|
Roger Meier30877382012-09-17 21:18:05 +0000623 CaptureDocText tok_identifier TypeAnnotations CommaOrSemicolonOptional
Mark Slee31985722006-05-24 21:45:31 +0000624 {
Mark Slee30152872006-11-28 01:24:07 +0000625 pdebug("EnumDef -> tok_identifier");
Jens Geyer12c09f42013-08-25 14:16:27 +0200626 validate_simple_identifier( $2);
ccheeverf53b5cf2007-02-05 20:33:11 +0000627 $$ = new t_enum_value($2);
628 if ($1 != NULL) {
629 $$->set_doc($1);
630 }
Roger Meier30877382012-09-17 21:18:05 +0000631 if ($3 != NULL) {
632 $$->annotations_ = $3->annotations_;
633 delete $3;
634 }
Mark Slee30152872006-11-28 01:24:07 +0000635 }
636
Mark Slee6a47fed2007-02-07 02:40:59 +0000637Senum:
Roger Meier30877382012-09-17 21:18:05 +0000638 tok_senum tok_identifier '{' SenumDefList '}' TypeAnnotations
Mark Slee6a47fed2007-02-07 02:40:59 +0000639 {
640 pdebug("Senum -> tok_senum tok_identifier { SenumDefList }");
Jens Geyer12c09f42013-08-25 14:16:27 +0200641 validate_simple_identifier( $2);
David Reisscdffe262007-08-14 17:12:31 +0000642 $$ = new t_typedef(g_program, $4, $2);
Roger Meier30877382012-09-17 21:18:05 +0000643 if ($6 != NULL) {
644 $$->annotations_ = $6->annotations_;
645 delete $6;
646 }
Mark Slee6a47fed2007-02-07 02:40:59 +0000647 }
648
649SenumDefList:
650 SenumDefList SenumDef
651 {
652 pdebug("SenumDefList -> SenumDefList SenumDef");
653 $$ = $1;
654 $$->add_string_enum_val($2);
655 }
656|
657 {
658 pdebug("SenumDefList -> ");
659 $$ = new t_base_type("string", t_base_type::TYPE_STRING);
660 $$->set_string_enum(true);
661 }
662
663SenumDef:
664 tok_literal CommaOrSemicolonOptional
665 {
666 pdebug("SenumDef -> tok_literal");
667 $$ = $1;
668 }
669
Mark Slee30152872006-11-28 01:24:07 +0000670Const:
671 tok_const FieldType tok_identifier '=' ConstValue CommaOrSemicolonOptional
672 {
673 pdebug("Const -> tok_const FieldType tok_identifier = ConstValue");
Mark Sleeaa7671d2006-11-29 03:19:31 +0000674 if (g_parse_mode == PROGRAM) {
Jens Geyer12c09f42013-08-25 14:16:27 +0200675 validate_simple_identifier( $3);
Bryan Duxbury2d804702009-12-18 19:41:11 +0000676 g_scope->resolve_const_value($5, $2);
Mark Sleeaa7671d2006-11-29 03:19:31 +0000677 $$ = new t_const($2, $3, $5);
678 validate_const_type($$);
Mark Sleed0767c52007-07-27 22:14:41 +0000679
680 g_scope->add_constant($3, $$);
681 if (g_parent_scope != NULL) {
682 g_parent_scope->add_constant(g_parent_prefix + $3, $$);
683 }
Mark Sleeaa7671d2006-11-29 03:19:31 +0000684 } else {
685 $$ = NULL;
686 }
Mark Slee30152872006-11-28 01:24:07 +0000687 }
688
689ConstValue:
690 tok_int_constant
691 {
692 pdebug("ConstValue => tok_int_constant");
693 $$ = new t_const_value();
694 $$->set_integer($1);
Roger Meier887ff752011-08-19 11:25:39 +0000695 if (!g_allow_64bit_consts && ($1 < INT32_MIN || $1 > INT32_MAX)) {
Roger Meier5f2d34e2013-11-16 16:43:41 +0100696 pwarning(1, "64-bit constant \"%" PRIi64"\" may not work in all languages.\n", $1);
David Reissf1454162008-06-30 20:45:47 +0000697 }
Mark Slee30152872006-11-28 01:24:07 +0000698 }
699| tok_dub_constant
700 {
701 pdebug("ConstValue => tok_dub_constant");
702 $$ = new t_const_value();
703 $$->set_double($1);
704 }
705| tok_literal
706 {
707 pdebug("ConstValue => tok_literal");
Mark Sleed0767c52007-07-27 22:14:41 +0000708 $$ = new t_const_value($1);
Mark Slee30152872006-11-28 01:24:07 +0000709 }
Mark Slee67fc6342006-11-29 03:37:04 +0000710| tok_identifier
711 {
712 pdebug("ConstValue => tok_identifier");
Bryan Duxbury2d804702009-12-18 19:41:11 +0000713 $$ = new t_const_value();
714 $$->set_identifier($1);
Mark Slee67fc6342006-11-29 03:37:04 +0000715 }
Mark Slee30152872006-11-28 01:24:07 +0000716| ConstList
717 {
718 pdebug("ConstValue => ConstList");
719 $$ = $1;
720 }
721| ConstMap
722 {
723 pdebug("ConstValue => ConstMap");
Mark Slee27ed6ec2007-08-16 01:26:31 +0000724 $$ = $1;
Mark Slee30152872006-11-28 01:24:07 +0000725 }
726
727ConstList:
728 '[' ConstListContents ']'
729 {
730 pdebug("ConstList => [ ConstListContents ]");
731 $$ = $2;
732 }
733
734ConstListContents:
735 ConstListContents ConstValue CommaOrSemicolonOptional
736 {
737 pdebug("ConstListContents => ConstListContents ConstValue CommaOrSemicolonOptional");
738 $$ = $1;
739 $$->add_list($2);
740 }
741|
742 {
743 pdebug("ConstListContents =>");
744 $$ = new t_const_value();
745 $$->set_list();
746 }
747
748ConstMap:
749 '{' ConstMapContents '}'
750 {
751 pdebug("ConstMap => { ConstMapContents }");
752 $$ = $2;
753 }
754
755ConstMapContents:
756 ConstMapContents ConstValue ':' ConstValue CommaOrSemicolonOptional
757 {
758 pdebug("ConstMapContents => ConstMapContents ConstValue CommaOrSemicolonOptional");
759 $$ = $1;
760 $$->add_map($2, $4);
761 }
762|
763 {
764 pdebug("ConstMapContents =>");
765 $$ = new t_const_value();
766 $$->set_map();
Mark Slee31985722006-05-24 21:45:31 +0000767 }
768
Bryan Duxburyab3666e2009-09-01 23:03:47 +0000769StructHead:
770 tok_struct
771 {
772 $$ = struct_is_struct;
773 }
774| tok_union
775 {
776 $$ = struct_is_union;
777 }
778
Mark Slee31985722006-05-24 21:45:31 +0000779Struct:
Bryan Duxburyab3666e2009-09-01 23:03:47 +0000780 StructHead tok_identifier XsdAll '{' FieldList '}' TypeAnnotations
Mark Slee31985722006-05-24 21:45:31 +0000781 {
782 pdebug("Struct -> tok_struct tok_identifier { FieldList }");
Jens Geyer12c09f42013-08-25 14:16:27 +0200783 validate_simple_identifier( $2);
David Reisscdffe262007-08-14 17:12:31 +0000784 $5->set_xsd_all($3);
Bryan Duxburyab3666e2009-09-01 23:03:47 +0000785 $5->set_union($1 == struct_is_union);
David Reisscdffe262007-08-14 17:12:31 +0000786 $$ = $5;
David Reissa2309992008-12-10 01:52:48 +0000787 $$->set_name($2);
788 if ($7 != NULL) {
789 $$->annotations_ = $7->annotations_;
790 delete $7;
791 }
Mark Slee9cb7c612006-09-01 22:17:45 +0000792 }
Ben Craige9576752013-10-11 08:19:16 -0500793
Mark Slee36bfa2e2007-01-19 20:09:51 +0000794XsdAll:
Mark Slee782abbb2007-01-19 00:17:02 +0000795 tok_xsd_all
796 {
797 $$ = true;
798 }
799|
800 {
801 $$ = false;
802 }
803
Mark Slee36bfa2e2007-01-19 20:09:51 +0000804XsdOptional:
805 tok_xsd_optional
806 {
807 $$ = true;
808 }
809|
810 {
811 $$ = false;
812 }
813
Mark Slee7df0e2a2007-02-06 21:03:18 +0000814XsdNillable:
815 tok_xsd_nillable
816 {
817 $$ = true;
818 }
819|
820 {
821 $$ = false;
822 }
823
Mark Slee21135c32007-02-05 21:52:08 +0000824XsdAttributes:
Mark Slee748d83f2007-02-07 01:20:08 +0000825 tok_xsd_attrs '{' FieldList '}'
Mark Slee21135c32007-02-05 21:52:08 +0000826 {
Mark Slee748d83f2007-02-07 01:20:08 +0000827 $$ = $3;
Mark Slee21135c32007-02-05 21:52:08 +0000828 }
829|
830 {
831 $$ = NULL;
832 }
833
Mark Slee9cb7c612006-09-01 22:17:45 +0000834Xception:
Roger Meier30877382012-09-17 21:18:05 +0000835 tok_xception tok_identifier '{' FieldList '}' TypeAnnotations
Mark Slee9cb7c612006-09-01 22:17:45 +0000836 {
837 pdebug("Xception -> tok_xception tok_identifier { FieldList }");
Jens Geyer12c09f42013-08-25 14:16:27 +0200838 validate_simple_identifier( $2);
Mark Slee9cb7c612006-09-01 22:17:45 +0000839 $4->set_name($2);
840 $4->set_xception(true);
841 $$ = $4;
Roger Meier30877382012-09-17 21:18:05 +0000842 if ($6 != NULL) {
843 $$->annotations_ = $6->annotations_;
844 delete $6;
845 }
Mark Slee31985722006-05-24 21:45:31 +0000846 }
847
848Service:
Roger Meier30877382012-09-17 21:18:05 +0000849 tok_service tok_identifier Extends '{' FlagArgs FunctionList UnflagArgs '}' TypeAnnotations
Mark Slee31985722006-05-24 21:45:31 +0000850 {
851 pdebug("Service -> tok_service tok_identifier { FunctionList }");
Jens Geyer12c09f42013-08-25 14:16:27 +0200852 validate_simple_identifier( $2);
Mark Slee78165722007-09-10 22:08:49 +0000853 $$ = $6;
David Reisscdffe262007-08-14 17:12:31 +0000854 $$->set_name($2);
855 $$->set_extends($3);
Roger Meier30877382012-09-17 21:18:05 +0000856 if ($9 != NULL) {
857 $$->annotations_ = $9->annotations_;
858 delete $9;
859 }
Mark Sleef0712dc2006-10-25 19:03:57 +0000860 }
861
Mark Slee78165722007-09-10 22:08:49 +0000862FlagArgs:
863 {
864 g_arglist = 1;
865 }
866
867UnflagArgs:
868 {
869 g_arglist = 0;
870 }
871
Mark Slee36bfa2e2007-01-19 20:09:51 +0000872Extends:
Mark Sleef0712dc2006-10-25 19:03:57 +0000873 tok_extends tok_identifier
874 {
Mark Slee36bfa2e2007-01-19 20:09:51 +0000875 pdebug("Extends -> tok_extends tok_identifier");
Mark Sleef0712dc2006-10-25 19:03:57 +0000876 $$ = NULL;
877 if (g_parse_mode == PROGRAM) {
878 $$ = g_scope->get_service($2);
879 if ($$ == NULL) {
880 yyerror("Service \"%s\" has not been defined.", $2);
881 exit(1);
882 }
883 }
884 }
885|
886 {
887 $$ = NULL;
Mark Slee31985722006-05-24 21:45:31 +0000888 }
889
890FunctionList:
Mark Slee207cb462006-11-02 18:43:12 +0000891 FunctionList Function
Mark Slee31985722006-05-24 21:45:31 +0000892 {
893 pdebug("FunctionList -> FunctionList Function");
894 $$ = $1;
895 $1->add_function($2);
896 }
897|
898 {
899 pdebug("FunctionList -> ");
Mark Sleef0712dc2006-10-25 19:03:57 +0000900 $$ = new t_service(g_program);
Mark Slee31985722006-05-24 21:45:31 +0000901 }
902
903Function:
Roger Meier30877382012-09-17 21:18:05 +0000904 CaptureDocText Oneway FunctionType tok_identifier '(' FieldList ')' Throws TypeAnnotations CommaOrSemicolonOptional
Mark Slee31985722006-05-24 21:45:31 +0000905 {
Jens Geyer12c09f42013-08-25 14:16:27 +0200906 validate_simple_identifier( $4);
ccheeverf53b5cf2007-02-05 20:33:11 +0000907 $6->set_name(std::string($4) + "_args");
908 $$ = new t_function($3, $4, $6, $8, $2);
909 if ($1 != NULL) {
910 $$->set_doc($1);
911 }
Roger Meier30877382012-09-17 21:18:05 +0000912 if ($9 != NULL) {
913 $$->annotations_ = $9->annotations_;
914 delete $9;
915 }
Mark Slee31985722006-05-24 21:45:31 +0000916 }
917
David Reiss6985a422009-03-24 20:00:47 +0000918Oneway:
919 tok_oneway
Mark Slee31985722006-05-24 21:45:31 +0000920 {
Mark Slee52f643d2006-08-09 00:03:43 +0000921 $$ = true;
922 }
923|
924 {
925 $$ = false;
Mark Slee31985722006-05-24 21:45:31 +0000926 }
927
Mark Slee36bfa2e2007-01-19 20:09:51 +0000928Throws:
Mark Slee9cb7c612006-09-01 22:17:45 +0000929 tok_throws '(' FieldList ')'
930 {
Mark Slee36bfa2e2007-01-19 20:09:51 +0000931 pdebug("Throws -> tok_throws ( FieldList )");
Mark Slee9cb7c612006-09-01 22:17:45 +0000932 $$ = $3;
Mark Sleef07d48e2008-02-01 01:36:26 +0000933 if (g_parse_mode == PROGRAM && !validate_throws($$)) {
Mark Slee91f2b7b2008-01-31 01:49:16 +0000934 yyerror("Throws clause may not contain non-exception types");
935 exit(1);
936 }
Mark Slee9cb7c612006-09-01 22:17:45 +0000937 }
938|
939 {
Mark Sleef0712dc2006-10-25 19:03:57 +0000940 $$ = new t_struct(g_program);
Mark Slee9cb7c612006-09-01 22:17:45 +0000941 }
942
Mark Slee31985722006-05-24 21:45:31 +0000943FieldList:
Mark Slee207cb462006-11-02 18:43:12 +0000944 FieldList Field
Mark Slee31985722006-05-24 21:45:31 +0000945 {
946 pdebug("FieldList -> FieldList , Field");
947 $$ = $1;
Bryan Duxburyff219ac2009-04-10 21:51:00 +0000948 if (!($$->append($2))) {
Jens Geyer3a67c2f2013-02-03 22:30:41 +0100949 yyerror("\"%d: %s\" - field identifier/name has already been used", $2->get_key(), $2->get_name().c_str());
Mark Slee6f9ac3f2007-11-28 22:28:13 +0000950 exit(1);
951 }
Mark Slee31985722006-05-24 21:45:31 +0000952 }
953|
954 {
955 pdebug("FieldList -> ");
David Reiss00a8dd62009-03-19 08:14:12 +0000956 y_field_val = -1;
Mark Sleef0712dc2006-10-25 19:03:57 +0000957 $$ = new t_struct(g_program);
Mark Slee31985722006-05-24 21:45:31 +0000958 }
959
960Field:
Jens Geyer885c6792014-05-02 21:31:55 +0200961 CaptureDocText FieldIdentifier FieldRequiredness FieldType FieldReference tok_identifier FieldValue XsdOptional XsdNillable XsdAttributes TypeAnnotations CommaOrSemicolonOptional
Mark Slee31985722006-05-24 21:45:31 +0000962 {
Mark Sleef0712dc2006-10-25 19:03:57 +0000963 pdebug("tok_int_constant : Field -> FieldType tok_identifier");
Bryan Duxburyc7206a42011-08-17 23:17:04 +0000964 if ($2.auto_assigned) {
Jens Geyer885c6792014-05-02 21:31:55 +0200965 pwarning(1, "No field key specified for %s, resulting protocol may have conflicts or not be backwards compatible!\n", $6);
Bryan Duxburya145b4d2009-04-03 17:29:25 +0000966 if (g_strict >= 192) {
967 yyerror("Implicit field keys are deprecated and not allowed with -strict");
968 exit(1);
969 }
Mark Slee31985722006-05-24 21:45:31 +0000970 }
Jens Geyer885c6792014-05-02 21:31:55 +0200971 validate_simple_identifier($6);
972 $$ = new t_field($4, $6, $2.value);
973 $$->set_reference($5);
David Reiss8320a922007-08-14 19:59:26 +0000974 $$->set_req($3);
Jens Geyer885c6792014-05-02 21:31:55 +0200975 if ($7 != NULL) {
976 g_scope->resolve_const_value($7, $4);
977 validate_field_value($$, $7);
978 $$->set_value($7);
Mark Slee7ff32452007-02-01 05:26:18 +0000979 }
Jens Geyer885c6792014-05-02 21:31:55 +0200980 $$->set_xsd_optional($8);
981 $$->set_xsd_nillable($9);
ccheeverf53b5cf2007-02-05 20:33:11 +0000982 if ($1 != NULL) {
983 $$->set_doc($1);
984 }
David Reiss53c10e02010-03-05 07:51:51 +0000985 if ($10 != NULL) {
Jens Geyer885c6792014-05-02 21:31:55 +0200986 $$->set_xsd_attrs($10);
987 }
988 if ($11 != NULL) {
989 $$->annotations_ = $11->annotations_;
990 delete $11;
David Reiss53c10e02010-03-05 07:51:51 +0000991 }
Mark Slee31985722006-05-24 21:45:31 +0000992 }
Mark Slee7ff32452007-02-01 05:26:18 +0000993
994FieldIdentifier:
995 tok_int_constant ':'
Mark Slee31985722006-05-24 21:45:31 +0000996 {
Mark Slee7ff32452007-02-01 05:26:18 +0000997 if ($1 <= 0) {
Bryan Duxburyc7206a42011-08-17 23:17:04 +0000998 if (g_allow_neg_field_keys) {
999 /*
1000 * g_allow_neg_field_keys exists to allow users to add explicitly
1001 * specified key values to old .thrift files without breaking
1002 * protocol compatibility.
1003 */
1004 if ($1 != y_field_val) {
1005 /*
1006 * warn if the user-specified negative value isn't what
1007 * thrift would have auto-assigned.
1008 */
Roger Meier5f2d34e2013-11-16 16:43:41 +01001009 pwarning(1, "Nonpositive field key (%" PRIi64") differs from what would be "
Bryan Duxburyc7206a42011-08-17 23:17:04 +00001010 "auto-assigned by thrift (%d).\n", $1, y_field_val);
1011 }
1012 /*
1013 * Leave $1 as-is, and update y_field_val to be one less than $1.
1014 * The FieldList parsing will catch any duplicate key values.
1015 */
Ben Craige9576752013-10-11 08:19:16 -05001016 y_field_val = static_cast<int32_t>($1 - 1);
1017 $$.value = static_cast<int32_t>($1);
Bryan Duxburyc7206a42011-08-17 23:17:04 +00001018 $$.auto_assigned = false;
1019 } else {
Ben Craige9576752013-10-11 08:19:16 -05001020 pwarning(1, "Nonpositive value (%d) not allowed as a field key.\n",
Bryan Duxburyc7206a42011-08-17 23:17:04 +00001021 $1);
1022 $$.value = y_field_val--;
1023 $$.auto_assigned = true;
1024 }
1025 } else {
Ben Craige9576752013-10-11 08:19:16 -05001026 $$.value = static_cast<int32_t>($1);
Bryan Duxburyc7206a42011-08-17 23:17:04 +00001027 $$.auto_assigned = false;
Mark Sleef0712dc2006-10-25 19:03:57 +00001028 }
Mark Slee7ff32452007-02-01 05:26:18 +00001029 }
1030|
1031 {
Bryan Duxburyc7206a42011-08-17 23:17:04 +00001032 $$.value = y_field_val--;
1033 $$.auto_assigned = true;
Mark Slee7ff32452007-02-01 05:26:18 +00001034 }
1035
Jens Geyer885c6792014-05-02 21:31:55 +02001036FieldReference:
1037 tok_reference
1038 {
1039 $$ = true;
1040 }
1041|
1042 {
1043 $$ = false;
1044 }
1045
David Reiss8320a922007-08-14 19:59:26 +00001046FieldRequiredness:
1047 tok_required
1048 {
David Reiss45603e92009-09-02 22:15:55 +00001049 $$ = t_field::T_REQUIRED;
David Reiss8320a922007-08-14 19:59:26 +00001050 }
1051| tok_optional
1052 {
Mark Slee78165722007-09-10 22:08:49 +00001053 if (g_arglist) {
1054 if (g_parse_mode == PROGRAM) {
1055 pwarning(1, "optional keyword is ignored in argument lists.\n");
1056 }
David Reiss204420f2008-01-11 20:59:03 +00001057 $$ = t_field::T_OPT_IN_REQ_OUT;
Mark Slee78165722007-09-10 22:08:49 +00001058 } else {
David Reiss204420f2008-01-11 20:59:03 +00001059 $$ = t_field::T_OPTIONAL;
Mark Slee78165722007-09-10 22:08:49 +00001060 }
David Reiss8320a922007-08-14 19:59:26 +00001061 }
1062|
1063 {
David Reiss204420f2008-01-11 20:59:03 +00001064 $$ = t_field::T_OPT_IN_REQ_OUT;
David Reiss8320a922007-08-14 19:59:26 +00001065 }
1066
Mark Slee7ff32452007-02-01 05:26:18 +00001067FieldValue:
1068 '=' ConstValue
1069 {
Mark Slee27ed6ec2007-08-16 01:26:31 +00001070 if (g_parse_mode == PROGRAM) {
Mark Slee7ff32452007-02-01 05:26:18 +00001071 $$ = $2;
1072 } else {
1073 $$ = NULL;
1074 }
1075 }
1076|
1077 {
1078 $$ = NULL;
Mark Sleef0712dc2006-10-25 19:03:57 +00001079 }
Mark Slee31985722006-05-24 21:45:31 +00001080
Mark Slee31985722006-05-24 21:45:31 +00001081FunctionType:
1082 FieldType
1083 {
Mark Sleee8540632006-05-30 09:24:40 +00001084 pdebug("FunctionType -> FieldType");
Mark Slee31985722006-05-24 21:45:31 +00001085 $$ = $1;
1086 }
1087| tok_void
1088 {
Mark Sleee8540632006-05-30 09:24:40 +00001089 pdebug("FunctionType -> tok_void");
Mark Sleef0712dc2006-10-25 19:03:57 +00001090 $$ = g_type_void;
Mark Slee31985722006-05-24 21:45:31 +00001091 }
1092
1093FieldType:
1094 tok_identifier
1095 {
Mark Sleee8540632006-05-30 09:24:40 +00001096 pdebug("FieldType -> tok_identifier");
Mark Sleef0712dc2006-10-25 19:03:57 +00001097 if (g_parse_mode == INCLUDES) {
1098 // Ignore identifiers in include mode
1099 $$ = NULL;
1100 } else {
1101 // Lookup the identifier in the current scope
1102 $$ = g_scope->get_type($1);
1103 if ($$ == NULL) {
jfarrelle0e83162014-04-08 22:45:01 -04001104 /*
1105 * Either this type isn't yet declared, or it's never
1106 declared. Either way allow it and we'll figure it out
1107 during generation.
1108 */
Jens Geyerd000b242014-04-13 21:58:47 +02001109 $$ = new t_typedef(g_program, $1, true);
Mark Sleef0712dc2006-10-25 19:03:57 +00001110 }
Mark Sleee8540632006-05-30 09:24:40 +00001111 }
Mark Slee31985722006-05-24 21:45:31 +00001112 }
1113| BaseType
1114 {
Mark Sleee8540632006-05-30 09:24:40 +00001115 pdebug("FieldType -> BaseType");
1116 $$ = $1;
1117 }
1118| ContainerType
1119 {
1120 pdebug("FieldType -> ContainerType");
Mark Slee31985722006-05-24 21:45:31 +00001121 $$ = $1;
1122 }
1123
David Reissc8e30052009-07-27 17:02:42 +00001124BaseType: SimpleBaseType TypeAnnotations
1125 {
1126 pdebug("BaseType -> SimpleBaseType TypeAnnotations");
1127 if ($2 != NULL) {
1128 $$ = new t_base_type(*static_cast<t_base_type*>($1));
1129 $$->annotations_ = $2->annotations_;
1130 delete $2;
1131 } else {
1132 $$ = $1;
1133 }
1134 }
1135
1136SimpleBaseType:
Mark Slee31985722006-05-24 21:45:31 +00001137 tok_string
1138 {
1139 pdebug("BaseType -> tok_string");
Mark Sleef0712dc2006-10-25 19:03:57 +00001140 $$ = g_type_string;
Mark Slee31985722006-05-24 21:45:31 +00001141 }
Mark Slee8d725a22007-04-13 01:57:12 +00001142| tok_binary
1143 {
1144 pdebug("BaseType -> tok_binary");
1145 $$ = g_type_binary;
1146 }
Mark Sleeb6200d82007-01-19 19:14:36 +00001147| tok_slist
1148 {
1149 pdebug("BaseType -> tok_slist");
1150 $$ = g_type_slist;
1151 }
Mark Slee78f58e22006-09-02 04:17:07 +00001152| tok_bool
1153 {
1154 pdebug("BaseType -> tok_bool");
Mark Sleef0712dc2006-10-25 19:03:57 +00001155 $$ = g_type_bool;
Mark Slee78f58e22006-09-02 04:17:07 +00001156 }
Mark Slee31985722006-05-24 21:45:31 +00001157| tok_byte
1158 {
1159 pdebug("BaseType -> tok_byte");
Mark Sleef0712dc2006-10-25 19:03:57 +00001160 $$ = g_type_byte;
Mark Slee31985722006-05-24 21:45:31 +00001161 }
Mark Slee9cb7c612006-09-01 22:17:45 +00001162| tok_i16
1163 {
1164 pdebug("BaseType -> tok_i16");
Mark Sleef0712dc2006-10-25 19:03:57 +00001165 $$ = g_type_i16;
Mark Slee9cb7c612006-09-01 22:17:45 +00001166 }
Mark Slee31985722006-05-24 21:45:31 +00001167| tok_i32
1168 {
1169 pdebug("BaseType -> tok_i32");
Mark Sleef0712dc2006-10-25 19:03:57 +00001170 $$ = g_type_i32;
Mark Slee31985722006-05-24 21:45:31 +00001171 }
Mark Slee31985722006-05-24 21:45:31 +00001172| tok_i64
1173 {
1174 pdebug("BaseType -> tok_i64");
Mark Sleef0712dc2006-10-25 19:03:57 +00001175 $$ = g_type_i64;
Mark Slee31985722006-05-24 21:45:31 +00001176 }
Mark Sleec98d0502006-09-06 02:42:25 +00001177| tok_double
1178 {
1179 pdebug("BaseType -> tok_double");
Mark Sleef0712dc2006-10-25 19:03:57 +00001180 $$ = g_type_double;
Mark Sleec98d0502006-09-06 02:42:25 +00001181 }
Mark Slee31985722006-05-24 21:45:31 +00001182
David Reissa2309992008-12-10 01:52:48 +00001183ContainerType: SimpleContainerType TypeAnnotations
1184 {
1185 pdebug("ContainerType -> SimpleContainerType TypeAnnotations");
1186 $$ = $1;
1187 if ($2 != NULL) {
1188 $$->annotations_ = $2->annotations_;
1189 delete $2;
1190 }
1191 }
1192
1193SimpleContainerType:
Mark Sleee8540632006-05-30 09:24:40 +00001194 MapType
1195 {
David Reissa2309992008-12-10 01:52:48 +00001196 pdebug("SimpleContainerType -> MapType");
Mark Sleee8540632006-05-30 09:24:40 +00001197 $$ = $1;
1198 }
1199| SetType
1200 {
David Reissa2309992008-12-10 01:52:48 +00001201 pdebug("SimpleContainerType -> SetType");
Mark Sleee8540632006-05-30 09:24:40 +00001202 $$ = $1;
1203 }
1204| ListType
1205 {
David Reissa2309992008-12-10 01:52:48 +00001206 pdebug("SimpleContainerType -> ListType");
Mark Sleee8540632006-05-30 09:24:40 +00001207 $$ = $1;
1208 }
1209
1210MapType:
Mark Slee36bfa2e2007-01-19 20:09:51 +00001211 tok_map CppType '<' FieldType ',' FieldType '>'
Mark Sleee8540632006-05-30 09:24:40 +00001212 {
1213 pdebug("MapType -> tok_map <FieldType, FieldType>");
Mark Slee4f8da1d2006-10-12 02:47:27 +00001214 $$ = new t_map($4, $6);
1215 if ($2 != NULL) {
1216 ((t_container*)$$)->set_cpp_name(std::string($2));
1217 }
Mark Sleee8540632006-05-30 09:24:40 +00001218 }
1219
1220SetType:
Mark Slee36bfa2e2007-01-19 20:09:51 +00001221 tok_set CppType '<' FieldType '>'
Mark Sleee8540632006-05-30 09:24:40 +00001222 {
1223 pdebug("SetType -> tok_set<FieldType>");
Mark Slee4f8da1d2006-10-12 02:47:27 +00001224 $$ = new t_set($4);
1225 if ($2 != NULL) {
1226 ((t_container*)$$)->set_cpp_name(std::string($2));
1227 }
Mark Sleee8540632006-05-30 09:24:40 +00001228 }
1229
1230ListType:
Mark Slee36bfa2e2007-01-19 20:09:51 +00001231 tok_list '<' FieldType '>' CppType
Mark Sleee8540632006-05-30 09:24:40 +00001232 {
1233 pdebug("ListType -> tok_list<FieldType>");
Jens Geyer6fe77e82014-03-16 16:48:53 +02001234 check_for_list_of_bytes($3);
Mark Sleef0712dc2006-10-25 19:03:57 +00001235 $$ = new t_list($3);
1236 if ($5 != NULL) {
1237 ((t_container*)$$)->set_cpp_name(std::string($5));
Mark Slee4f8da1d2006-10-12 02:47:27 +00001238 }
1239 }
1240
Mark Slee36bfa2e2007-01-19 20:09:51 +00001241CppType:
Mark Sleeafc76542007-02-09 21:55:44 +00001242 tok_cpp_type tok_literal
Mark Slee4f8da1d2006-10-12 02:47:27 +00001243 {
Mark Sleeafc76542007-02-09 21:55:44 +00001244 $$ = $2;
Mark Slee4f8da1d2006-10-12 02:47:27 +00001245 }
1246|
1247 {
1248 $$ = NULL;
Mark Sleee8540632006-05-30 09:24:40 +00001249 }
1250
David Reissa2309992008-12-10 01:52:48 +00001251TypeAnnotations:
1252 '(' TypeAnnotationList ')'
1253 {
1254 pdebug("TypeAnnotations -> ( TypeAnnotationList )");
1255 $$ = $2;
1256 }
1257|
1258 {
1259 $$ = NULL;
1260 }
1261
1262TypeAnnotationList:
1263 TypeAnnotationList TypeAnnotation
1264 {
1265 pdebug("TypeAnnotationList -> TypeAnnotationList , TypeAnnotation");
1266 $$ = $1;
1267 $$->annotations_[$2->key] = $2->val;
1268 delete $2;
1269 }
1270|
1271 {
1272 /* Just use a dummy structure to hold the annotations. */
1273 $$ = new t_struct(g_program);
1274 }
1275
1276TypeAnnotation:
Jens Geyer89847df2014-05-02 23:50:04 +02001277 tok_identifier TypeAnnotationValue CommaOrSemicolonOptional
David Reissa2309992008-12-10 01:52:48 +00001278 {
Jens Geyer89847df2014-05-02 23:50:04 +02001279 pdebug("TypeAnnotation -> TypeAnnotationValue");
David Reissa2309992008-12-10 01:52:48 +00001280 $$ = new t_annotation;
1281 $$->key = $1;
Jens Geyer89847df2014-05-02 23:50:04 +02001282 $$->val = $2;
1283 }
1284
1285TypeAnnotationValue:
1286 '=' tok_literal
1287 {
1288 pdebug("TypeAnnotationValue -> = tok_literal");
1289 $$ = $2;
1290 }
1291|
1292 {
1293 pdebug("TypeAnnotationValue ->");
1294 $$ = strdup("1");
David Reissa2309992008-12-10 01:52:48 +00001295 }
1296
Todd Lipcon53ae9f32009-12-07 00:42:38 +00001297%%