blob: 40e7a063fd3f2df9d49a5c937715d7ff45b7665f [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;
Jens Geyerae0b22c2014-09-04 23:04:21 +020056/**
57 * This global variable is used for automatic numbering of enum values.
58 * y_enum_val is the last value assigned; the next auto-assigned value will be
59 * y_enum_val+1, and then it continues working upwards. Explicitly specified
60 * enum values reset y_enum_val to that value.
61 */
62int32_t y_enum_val = -1;
Mark Slee78165722007-09-10 22:08:49 +000063int g_arglist = 0;
Bryan Duxburyab3666e2009-09-01 23:03:47 +000064const int struct_is_struct = 0;
65const int struct_is_union = 1;
Mark Slee31985722006-05-24 21:45:31 +000066
67%}
68
Mark Sleef5377b32006-10-10 01:42:59 +000069/**
70 * This structure is used by the parser to hold the data types associated with
71 * various parse nodes.
72 */
Mark Slee31985722006-05-24 21:45:31 +000073%union {
Mark Slee30152872006-11-28 01:24:07 +000074 char* id;
David Reissf1454162008-06-30 20:45:47 +000075 int64_t iconst;
Mark Slee30152872006-11-28 01:24:07 +000076 double dconst;
77 bool tbool;
David Reisscdffe262007-08-14 17:12:31 +000078 t_doc* tdoc;
Mark Slee30152872006-11-28 01:24:07 +000079 t_type* ttype;
Mark Slee6a47fed2007-02-07 02:40:59 +000080 t_base_type* tbase;
Mark Slee30152872006-11-28 01:24:07 +000081 t_typedef* ttypedef;
82 t_enum* tenum;
83 t_enum_value* tenumv;
84 t_const* tconst;
85 t_const_value* tconstv;
86 t_struct* tstruct;
87 t_service* tservice;
88 t_function* tfunction;
89 t_field* tfield;
David Reisscdffe262007-08-14 17:12:31 +000090 char* dtext;
David Reiss8320a922007-08-14 19:59:26 +000091 t_field::e_req ereq;
David Reissa2309992008-12-10 01:52:48 +000092 t_annotation* tannot;
Bryan Duxburyc7206a42011-08-17 23:17:04 +000093 t_field_id tfieldid;
Mark Slee31985722006-05-24 21:45:31 +000094}
95
Mark Sleef5377b32006-10-10 01:42:59 +000096/**
97 * Strings identifier
98 */
Mark Slee31985722006-05-24 21:45:31 +000099%token<id> tok_identifier
Mark Sleef0712dc2006-10-25 19:03:57 +0000100%token<id> tok_literal
David Reisscdffe262007-08-14 17:12:31 +0000101%token<dtext> tok_doctext
Mark Sleebd588222007-11-21 08:43:35 +0000102%token<id> tok_st_identifier
Mark Sleef5377b32006-10-10 01:42:59 +0000103
104/**
Mark Slee30152872006-11-28 01:24:07 +0000105 * Constant values
Mark Sleef5377b32006-10-10 01:42:59 +0000106 */
Mark Slee31985722006-05-24 21:45:31 +0000107%token<iconst> tok_int_constant
Mark Slee30152872006-11-28 01:24:07 +0000108%token<dconst> tok_dub_constant
Mark Slee31985722006-05-24 21:45:31 +0000109
Mark Sleef5377b32006-10-10 01:42:59 +0000110/**
David Reiss399442b2008-02-20 02:28:05 +0000111 * Header keywords
Mark Sleef5377b32006-10-10 01:42:59 +0000112 */
Mark Sleef0712dc2006-10-25 19:03:57 +0000113%token tok_include
Mark Slee9cb7c612006-09-01 22:17:45 +0000114%token tok_namespace
Mark Sleef0712dc2006-10-25 19:03:57 +0000115%token tok_cpp_namespace
116%token tok_cpp_include
117%token tok_cpp_type
Mark Sleee888b372007-01-12 01:06:24 +0000118%token tok_php_namespace
David Reissc6fc3292007-08-30 00:58:43 +0000119%token tok_py_module
Mark Slee27ed6ec2007-08-16 01:26:31 +0000120%token tok_perl_package
Mark Sleef0712dc2006-10-25 19:03:57 +0000121%token tok_java_package
Mark Slee782abbb2007-01-19 00:17:02 +0000122%token tok_xsd_all
Mark Slee36bfa2e2007-01-19 20:09:51 +0000123%token tok_xsd_optional
Mark Slee7df0e2a2007-02-06 21:03:18 +0000124%token tok_xsd_nillable
Mark Slee0d9199e2007-01-31 02:08:30 +0000125%token tok_xsd_namespace
Mark Slee21135c32007-02-05 21:52:08 +0000126%token tok_xsd_attrs
Mark Slee58dfb4f2007-07-06 02:45:25 +0000127%token tok_ruby_namespace
Mark Sleebd588222007-11-21 08:43:35 +0000128%token tok_smalltalk_category
David Reiss15457c92007-12-14 07:03:03 +0000129%token tok_smalltalk_prefix
Mark Slee7e9eea42007-09-10 21:00:23 +0000130%token tok_cocoa_prefix
David Reiss7f42bcf2008-01-11 20:59:12 +0000131%token tok_csharp_namespace
Jake Farrell7ae13e12011-10-18 14:35:26 +0000132%token tok_delphi_namespace
Mark Slee9cb7c612006-09-01 22:17:45 +0000133
Mark Sleef5377b32006-10-10 01:42:59 +0000134/**
135 * Base datatype keywords
136 */
137%token tok_void
Mark Slee78f58e22006-09-02 04:17:07 +0000138%token tok_bool
Mark Slee31985722006-05-24 21:45:31 +0000139%token tok_byte
140%token tok_string
Mark Slee8d725a22007-04-13 01:57:12 +0000141%token tok_binary
Mark Sleeb6200d82007-01-19 19:14:36 +0000142%token tok_slist
Mark Slee6a47fed2007-02-07 02:40:59 +0000143%token tok_senum
Mark Slee9cb7c612006-09-01 22:17:45 +0000144%token tok_i16
Mark Slee31985722006-05-24 21:45:31 +0000145%token tok_i32
Mark Slee31985722006-05-24 21:45:31 +0000146%token tok_i64
Mark Slee9cb7c612006-09-01 22:17:45 +0000147%token tok_double
Mark Slee31985722006-05-24 21:45:31 +0000148
Mark Sleef5377b32006-10-10 01:42:59 +0000149/**
150 * Complex type keywords
151 */
Mark Slee31985722006-05-24 21:45:31 +0000152%token tok_map
153%token tok_list
154%token tok_set
155
Mark Sleef5377b32006-10-10 01:42:59 +0000156/**
157 * Function modifiers
Mark Slee27ed6ec2007-08-16 01:26:31 +0000158 */
David Reiss6985a422009-03-24 20:00:47 +0000159%token tok_oneway
Mark Slee31985722006-05-24 21:45:31 +0000160
Mark Sleef5377b32006-10-10 01:42:59 +0000161/**
162 * Thrift language keywords
163 */
Mark Slee31985722006-05-24 21:45:31 +0000164%token tok_typedef
165%token tok_struct
Mark Slee9cb7c612006-09-01 22:17:45 +0000166%token tok_xception
167%token tok_throws
Mark Sleef0712dc2006-10-25 19:03:57 +0000168%token tok_extends
Mark Slee31985722006-05-24 21:45:31 +0000169%token tok_service
170%token tok_enum
Mark Slee30152872006-11-28 01:24:07 +0000171%token tok_const
David Reiss8320a922007-08-14 19:59:26 +0000172%token tok_required
173%token tok_optional
Bryan Duxburyab3666e2009-09-01 23:03:47 +0000174%token tok_union
Jens Geyer885c6792014-05-02 21:31:55 +0200175%token tok_reference
Mark Slee31985722006-05-24 21:45:31 +0000176
Mark Sleef5377b32006-10-10 01:42:59 +0000177/**
178 * Grammar nodes
179 */
180
Mark Slee31985722006-05-24 21:45:31 +0000181%type<ttype> BaseType
David Reissc8e30052009-07-27 17:02:42 +0000182%type<ttype> SimpleBaseType
Mark Sleee8540632006-05-30 09:24:40 +0000183%type<ttype> ContainerType
David Reissa2309992008-12-10 01:52:48 +0000184%type<ttype> SimpleContainerType
Mark Sleee8540632006-05-30 09:24:40 +0000185%type<ttype> MapType
186%type<ttype> SetType
187%type<ttype> ListType
Mark Slee31985722006-05-24 21:45:31 +0000188
David Reisscdffe262007-08-14 17:12:31 +0000189%type<tdoc> Definition
Mark Sleef0712dc2006-10-25 19:03:57 +0000190%type<ttype> TypeDefinition
191
Mark Slee31985722006-05-24 21:45:31 +0000192%type<ttypedef> Typedef
Mark Slee31985722006-05-24 21:45:31 +0000193
David Reissa2309992008-12-10 01:52:48 +0000194%type<ttype> TypeAnnotations
195%type<ttype> TypeAnnotationList
196%type<tannot> TypeAnnotation
Jens Geyer89847df2014-05-02 23:50:04 +0200197%type<id> TypeAnnotationValue
David Reissa2309992008-12-10 01:52:48 +0000198
Mark Slee31985722006-05-24 21:45:31 +0000199%type<tfield> Field
Bryan Duxburyc7206a42011-08-17 23:17:04 +0000200%type<tfieldid> FieldIdentifier
David Reiss8320a922007-08-14 19:59:26 +0000201%type<ereq> FieldRequiredness
Mark Slee31985722006-05-24 21:45:31 +0000202%type<ttype> FieldType
Mark Slee7ff32452007-02-01 05:26:18 +0000203%type<tconstv> FieldValue
Mark Sleee8540632006-05-30 09:24:40 +0000204%type<tstruct> FieldList
Jens Geyer885c6792014-05-02 21:31:55 +0200205%type<tbool> FieldReference
Mark Slee31985722006-05-24 21:45:31 +0000206
207%type<tenum> Enum
208%type<tenum> EnumDefList
Mark Slee30152872006-11-28 01:24:07 +0000209%type<tenumv> EnumDef
Jens Geyerae0b22c2014-09-04 23:04:21 +0200210%type<tenumv> EnumValue
Mark Slee30152872006-11-28 01:24:07 +0000211
Mark Slee6a47fed2007-02-07 02:40:59 +0000212%type<ttypedef> Senum
213%type<tbase> SenumDefList
214%type<id> SenumDef
215
Mark Slee30152872006-11-28 01:24:07 +0000216%type<tconst> Const
217%type<tconstv> ConstValue
218%type<tconstv> ConstList
219%type<tconstv> ConstListContents
220%type<tconstv> ConstMap
221%type<tconstv> ConstMapContents
Mark Slee31985722006-05-24 21:45:31 +0000222
Bryan Duxburyab3666e2009-09-01 23:03:47 +0000223%type<iconst> StructHead
Mark Slee31985722006-05-24 21:45:31 +0000224%type<tstruct> Struct
Mark Slee9cb7c612006-09-01 22:17:45 +0000225%type<tstruct> Xception
Mark Slee31985722006-05-24 21:45:31 +0000226%type<tservice> Service
227
228%type<tfunction> Function
Mark Slee31985722006-05-24 21:45:31 +0000229%type<ttype> FunctionType
230%type<tservice> FunctionList
231
Mark Slee36bfa2e2007-01-19 20:09:51 +0000232%type<tstruct> Throws
233%type<tservice> Extends
David Reiss6985a422009-03-24 20:00:47 +0000234%type<tbool> Oneway
Mark Slee36bfa2e2007-01-19 20:09:51 +0000235%type<tbool> XsdAll
236%type<tbool> XsdOptional
Mark Slee7df0e2a2007-02-06 21:03:18 +0000237%type<tbool> XsdNillable
Mark Slee748d83f2007-02-07 01:20:08 +0000238%type<tstruct> XsdAttributes
Mark Slee36bfa2e2007-01-19 20:09:51 +0000239%type<id> CppType
Mark Slee52f643d2006-08-09 00:03:43 +0000240
David Reisscbd4bac2007-08-14 17:12:33 +0000241%type<dtext> CaptureDocText
ccheeverf53b5cf2007-02-05 20:33:11 +0000242
Mark Slee31985722006-05-24 21:45:31 +0000243%%
244
Mark Sleef5377b32006-10-10 01:42:59 +0000245/**
246 * Thrift Grammar Implementation.
247 *
248 * For the most part this source file works its way top down from what you
249 * might expect to find in a typical .thrift file, i.e. type definitions and
250 * namespaces up top followed by service definitions using those types.
251 */
Mark Slee31985722006-05-24 21:45:31 +0000252
253Program:
David Reisscbd4bac2007-08-14 17:12:33 +0000254 HeaderList DefinitionList
Mark Sleef0712dc2006-10-25 19:03:57 +0000255 {
256 pdebug("Program -> Headers DefinitionList");
Jens Geyere8379b52014-01-25 00:59:45 +0100257 if((g_program_doctext_candidate != NULL) && (g_program_doctext_status != ALREADY_PROCESSED))
258 {
259 g_program->set_doc(g_program_doctext_candidate);
260 g_program_doctext_status = ALREADY_PROCESSED;
David Reissc2532a92007-07-30 23:46:11 +0000261 }
David Reisscbd4bac2007-08-14 17:12:33 +0000262 clear_doctext();
Mark Sleef0712dc2006-10-25 19:03:57 +0000263 }
264
David Reisscbd4bac2007-08-14 17:12:33 +0000265CaptureDocText:
266 {
267 if (g_parse_mode == PROGRAM) {
Mark Sleebd588222007-11-21 08:43:35 +0000268 $$ = g_doctext;
David Reisscbd4bac2007-08-14 17:12:33 +0000269 g_doctext = NULL;
Mark Slee27ed6ec2007-08-16 01:26:31 +0000270 } else {
David Reisscbd4bac2007-08-14 17:12:33 +0000271 $$ = NULL;
272 }
273 }
274
275/* TODO(dreiss): Try to DestroyDocText in all sorts or random places. */
276DestroyDocText:
277 {
278 if (g_parse_mode == PROGRAM) {
279 clear_doctext();
280 }
281 }
282
283/* We have to DestroyDocText here, otherwise it catches the doctext
284 on the first real element. */
Mark Sleef0712dc2006-10-25 19:03:57 +0000285HeaderList:
David Reisscbd4bac2007-08-14 17:12:33 +0000286 HeaderList DestroyDocText Header
Mark Sleef0712dc2006-10-25 19:03:57 +0000287 {
288 pdebug("HeaderList -> HeaderList Header");
289 }
290|
291 {
292 pdebug("HeaderList -> ");
293 }
294
295Header:
296 Include
297 {
298 pdebug("Header -> Include");
299 }
David Reiss79eca142008-02-27 01:55:13 +0000300| tok_namespace tok_identifier tok_identifier
Mark Sleef0712dc2006-10-25 19:03:57 +0000301 {
David Reiss79eca142008-02-27 01:55:13 +0000302 pdebug("Header -> tok_namespace tok_identifier tok_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100303 declare_valid_program_doctext();
Mark Sleef0712dc2006-10-25 19:03:57 +0000304 if (g_parse_mode == PROGRAM) {
David Reiss79eca142008-02-27 01:55:13 +0000305 g_program->set_namespace($2, $3);
Mark Sleef0712dc2006-10-25 19:03:57 +0000306 }
307 }
David Reissfb790d72010-09-02 16:41:45 +0000308| tok_namespace '*' tok_identifier
309 {
310 pdebug("Header -> tok_namespace * tok_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100311 declare_valid_program_doctext();
David Reissfb790d72010-09-02 16:41:45 +0000312 if (g_parse_mode == PROGRAM) {
313 g_program->set_namespace("*", $3);
314 }
315 }
David Reiss9a08dc62008-02-27 01:55:17 +0000316/* TODO(dreiss): Get rid of this once everyone is using the new hotness. */
Mark Sleef0712dc2006-10-25 19:03:57 +0000317| tok_cpp_namespace tok_identifier
318 {
David Reiss9a08dc62008-02-27 01:55:17 +0000319 pwarning(1, "'cpp_namespace' is deprecated. Use 'namespace cpp' instead");
Mark Sleef0712dc2006-10-25 19:03:57 +0000320 pdebug("Header -> tok_cpp_namespace tok_identifier");
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) {
David Reiss9a08dc62008-02-27 01:55:17 +0000323 g_program->set_namespace("cpp", $2);
Mark Sleef0712dc2006-10-25 19:03:57 +0000324 }
325 }
326| tok_cpp_include tok_literal
327 {
328 pdebug("Header -> tok_cpp_include tok_literal");
Jens Geyere8379b52014-01-25 00:59:45 +0100329 declare_valid_program_doctext();
Mark Sleef0712dc2006-10-25 19:03:57 +0000330 if (g_parse_mode == PROGRAM) {
331 g_program->add_cpp_include($2);
332 }
333 }
Mark Sleee888b372007-01-12 01:06:24 +0000334| tok_php_namespace tok_identifier
335 {
David Reiss554ea6f2009-02-17 20:28:37 +0000336 pwarning(1, "'php_namespace' is deprecated. Use 'namespace php' instead");
Mark Sleee888b372007-01-12 01:06:24 +0000337 pdebug("Header -> tok_php_namespace tok_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100338 declare_valid_program_doctext();
Mark Sleee888b372007-01-12 01:06:24 +0000339 if (g_parse_mode == PROGRAM) {
David Reiss554ea6f2009-02-17 20:28:37 +0000340 g_program->set_namespace("php", $2);
Mark Sleee888b372007-01-12 01:06:24 +0000341 }
342 }
David Reiss320e45c2008-03-27 21:41:54 +0000343/* TODO(dreiss): Get rid of this once everyone is using the new hotness. */
David Reissc6fc3292007-08-30 00:58:43 +0000344| tok_py_module tok_identifier
345 {
David Reiss320e45c2008-03-27 21:41:54 +0000346 pwarning(1, "'py_module' is deprecated. Use 'namespace py' instead");
David Reissc6fc3292007-08-30 00:58:43 +0000347 pdebug("Header -> tok_py_module tok_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100348 declare_valid_program_doctext();
David Reissc6fc3292007-08-30 00:58:43 +0000349 if (g_parse_mode == PROGRAM) {
David Reiss320e45c2008-03-27 21:41:54 +0000350 g_program->set_namespace("py", $2);
David Reissc6fc3292007-08-30 00:58:43 +0000351 }
352 }
David Reiss07ef3a92008-03-27 21:42:39 +0000353/* TODO(dreiss): Get rid of this once everyone is using the new hotness. */
Mark Slee27ed6ec2007-08-16 01:26:31 +0000354| tok_perl_package tok_identifier
355 {
David Reiss07ef3a92008-03-27 21:42:39 +0000356 pwarning(1, "'perl_package' is deprecated. Use 'namespace perl' instead");
Mark Slee27ed6ec2007-08-16 01:26:31 +0000357 pdebug("Header -> tok_perl_namespace tok_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100358 declare_valid_program_doctext();
Mark Slee27ed6ec2007-08-16 01:26:31 +0000359 if (g_parse_mode == PROGRAM) {
David Reiss07ef3a92008-03-27 21:42:39 +0000360 g_program->set_namespace("perl", $2);
Mark Slee27ed6ec2007-08-16 01:26:31 +0000361 }
362 }
David Reiss6a4b82c2008-03-27 21:42:16 +0000363/* TODO(dreiss): Get rid of this once everyone is using the new hotness. */
Mark Slee58dfb4f2007-07-06 02:45:25 +0000364| tok_ruby_namespace tok_identifier
365 {
David Reiss6a4b82c2008-03-27 21:42:16 +0000366 pwarning(1, "'ruby_namespace' is deprecated. Use 'namespace rb' instead");
Mark Slee58dfb4f2007-07-06 02:45:25 +0000367 pdebug("Header -> tok_ruby_namespace tok_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100368 declare_valid_program_doctext();
Mark Slee58dfb4f2007-07-06 02:45:25 +0000369 if (g_parse_mode == PROGRAM) {
David Reiss6a4b82c2008-03-27 21:42:16 +0000370 g_program->set_namespace("rb", $2);
Mark Slee58dfb4f2007-07-06 02:45:25 +0000371 }
372 }
David Reiss3b455012008-03-27 21:40:46 +0000373/* TODO(dreiss): Get rid of this once everyone is using the new hotness. */
Mark Sleebd588222007-11-21 08:43:35 +0000374| tok_smalltalk_category tok_st_identifier
375 {
David Reiss3b455012008-03-27 21:40:46 +0000376 pwarning(1, "'smalltalk_category' is deprecated. Use 'namespace smalltalk.category' instead");
Mark Sleebd588222007-11-21 08:43:35 +0000377 pdebug("Header -> tok_smalltalk_category tok_st_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100378 declare_valid_program_doctext();
Mark Sleebd588222007-11-21 08:43:35 +0000379 if (g_parse_mode == PROGRAM) {
David Reiss3b455012008-03-27 21:40:46 +0000380 g_program->set_namespace("smalltalk.category", $2);
Mark Sleebd588222007-11-21 08:43:35 +0000381 }
382 }
David Reiss3b455012008-03-27 21:40:46 +0000383/* TODO(dreiss): Get rid of this once everyone is using the new hotness. */
David Reiss15457c92007-12-14 07:03:03 +0000384| tok_smalltalk_prefix tok_identifier
385 {
David Reiss3b455012008-03-27 21:40:46 +0000386 pwarning(1, "'smalltalk_prefix' is deprecated. Use 'namespace smalltalk.prefix' instead");
David Reiss15457c92007-12-14 07:03:03 +0000387 pdebug("Header -> tok_smalltalk_prefix tok_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100388 declare_valid_program_doctext();
David Reiss15457c92007-12-14 07:03:03 +0000389 if (g_parse_mode == PROGRAM) {
David Reiss3b455012008-03-27 21:40:46 +0000390 g_program->set_namespace("smalltalk.prefix", $2);
David Reiss15457c92007-12-14 07:03:03 +0000391 }
392 }
David Reiss771f8c72008-02-27 01:55:25 +0000393/* TODO(dreiss): Get rid of this once everyone is using the new hotness. */
Mark Sleef0712dc2006-10-25 19:03:57 +0000394| tok_java_package tok_identifier
395 {
David Reiss9f646152008-03-02 21:59:48 +0000396 pwarning(1, "'java_package' is deprecated. Use 'namespace java' instead");
Mark Sleef0712dc2006-10-25 19:03:57 +0000397 pdebug("Header -> tok_java_package tok_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100398 declare_valid_program_doctext();
Mark Sleef0712dc2006-10-25 19:03:57 +0000399 if (g_parse_mode == PROGRAM) {
David Reiss771f8c72008-02-27 01:55:25 +0000400 g_program->set_namespace("java", $2);
Mark Sleef0712dc2006-10-25 19:03:57 +0000401 }
402 }
David Reiss54b602b2008-03-27 21:41:06 +0000403/* TODO(dreiss): Get rid of this once everyone is using the new hotness. */
Mark Slee7e9eea42007-09-10 21:00:23 +0000404| tok_cocoa_prefix tok_identifier
405 {
David Reiss54b602b2008-03-27 21:41:06 +0000406 pwarning(1, "'cocoa_prefix' is deprecated. Use 'namespace cocoa' instead");
Mark Slee7e9eea42007-09-10 21:00:23 +0000407 pdebug("Header -> tok_cocoa_prefix tok_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100408 declare_valid_program_doctext();
Mark Slee7e9eea42007-09-10 21:00:23 +0000409 if (g_parse_mode == PROGRAM) {
David Reiss54b602b2008-03-27 21:41:06 +0000410 g_program->set_namespace("cocoa", $2);
Mark Slee7e9eea42007-09-10 21:00:23 +0000411 }
412 }
David Reiss92e10d82009-02-17 20:28:19 +0000413/* TODO(dreiss): Get rid of this once everyone is using the new hotness. */
Mark Slee0d9199e2007-01-31 02:08:30 +0000414| tok_xsd_namespace tok_literal
415 {
David Reiss92e10d82009-02-17 20:28:19 +0000416 pwarning(1, "'xsd_namespace' is deprecated. Use 'namespace xsd' instead");
Mark Slee0d9199e2007-01-31 02:08:30 +0000417 pdebug("Header -> tok_xsd_namespace tok_literal");
Jens Geyere8379b52014-01-25 00:59:45 +0100418 declare_valid_program_doctext();
Mark Slee0d9199e2007-01-31 02:08:30 +0000419 if (g_parse_mode == PROGRAM) {
David Reiss92e10d82009-02-17 20:28:19 +0000420 g_program->set_namespace("cocoa", $2);
Mark Slee0d9199e2007-01-31 02:08:30 +0000421 }
422 }
David Reiss9d65bf02008-03-27 21:41:37 +0000423/* TODO(dreiss): Get rid of this once everyone is using the new hotness. */
David Reiss7f42bcf2008-01-11 20:59:12 +0000424| tok_csharp_namespace tok_identifier
425 {
David Reiss9d65bf02008-03-27 21:41:37 +0000426 pwarning(1, "'csharp_namespace' is deprecated. Use 'namespace csharp' instead");
David Reiss919ae802008-03-27 21:41:11 +0000427 pdebug("Header -> tok_csharp_namespace tok_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100428 declare_valid_program_doctext();
David Reiss7f42bcf2008-01-11 20:59:12 +0000429 if (g_parse_mode == PROGRAM) {
David Reiss9d65bf02008-03-27 21:41:37 +0000430 g_program->set_namespace("csharp", $2);
David Reiss7f42bcf2008-01-11 20:59:12 +0000431 }
432 }
Jake Farrell7ae13e12011-10-18 14:35:26 +0000433/* TODO(dreiss): Get rid of this once everyone is using the new hotness. */
434| tok_delphi_namespace tok_identifier
435 {
436 pwarning(1, "'delphi_namespace' is deprecated. Use 'namespace delphi' instead");
437 pdebug("Header -> tok_delphi_namespace tok_identifier");
Jens Geyere8379b52014-01-25 00:59:45 +0100438 declare_valid_program_doctext();
Jake Farrell7ae13e12011-10-18 14:35:26 +0000439 if (g_parse_mode == PROGRAM) {
440 g_program->set_namespace("delphi", $2);
441 }
442 }
Mark Sleef0712dc2006-10-25 19:03:57 +0000443
444Include:
445 tok_include tok_literal
446 {
Mark Slee27ed6ec2007-08-16 01:26:31 +0000447 pdebug("Include -> tok_include tok_literal");
Jens Geyere8379b52014-01-25 00:59:45 +0100448 declare_valid_program_doctext();
Mark Sleef0712dc2006-10-25 19:03:57 +0000449 if (g_parse_mode == INCLUDES) {
450 std::string path = include_file(std::string($2));
451 if (!path.empty()) {
kholst76f2c882008-01-16 02:47:41 +0000452 g_program->add_include(path, std::string($2));
Mark Sleef0712dc2006-10-25 19:03:57 +0000453 }
454 }
455 }
Mark Slee31985722006-05-24 21:45:31 +0000456
457DefinitionList:
David Reisscbd4bac2007-08-14 17:12:33 +0000458 DefinitionList CaptureDocText Definition
Mark Slee31985722006-05-24 21:45:31 +0000459 {
460 pdebug("DefinitionList -> DefinitionList Definition");
David Reisscdffe262007-08-14 17:12:31 +0000461 if ($2 != NULL && $3 != NULL) {
462 $3->set_doc($2);
463 }
Mark Slee31985722006-05-24 21:45:31 +0000464 }
465|
466 {
467 pdebug("DefinitionList -> ");
468 }
469
470Definition:
Mark Slee30152872006-11-28 01:24:07 +0000471 Const
472 {
473 pdebug("Definition -> Const");
474 if (g_parse_mode == PROGRAM) {
475 g_program->add_const($1);
Mark Sleebd588222007-11-21 08:43:35 +0000476 }
David Reisscdffe262007-08-14 17:12:31 +0000477 $$ = $1;
Mark Slee30152872006-11-28 01:24:07 +0000478 }
479| TypeDefinition
Mark Slee9cb7c612006-09-01 22:17:45 +0000480 {
Mark Sleef0712dc2006-10-25 19:03:57 +0000481 pdebug("Definition -> TypeDefinition");
482 if (g_parse_mode == PROGRAM) {
483 g_scope->add_type($1->get_name(), $1);
484 if (g_parent_scope != NULL) {
485 g_parent_scope->add_type(g_parent_prefix + $1->get_name(), $1);
486 }
Roger Meierba406d32013-07-15 22:41:34 +0200487 if (! g_program->is_unique_typename($1)) {
488 yyerror("Type \"%s\" is already defined.", $1->get_name().c_str());
489 exit(1);
490 }
Mark Sleef0712dc2006-10-25 19:03:57 +0000491 }
David Reisscdffe262007-08-14 17:12:31 +0000492 $$ = $1;
Mark Slee9cb7c612006-09-01 22:17:45 +0000493 }
Mark Slee31985722006-05-24 21:45:31 +0000494| Service
495 {
496 pdebug("Definition -> Service");
Mark Sleef0712dc2006-10-25 19:03:57 +0000497 if (g_parse_mode == PROGRAM) {
498 g_scope->add_service($1->get_name(), $1);
499 if (g_parent_scope != NULL) {
500 g_parent_scope->add_service(g_parent_prefix + $1->get_name(), $1);
501 }
502 g_program->add_service($1);
Roger Meierba406d32013-07-15 22:41:34 +0200503 if (! g_program->is_unique_typename($1)) {
504 yyerror("Type \"%s\" is already defined.", $1->get_name().c_str());
505 exit(1);
506 }
Mark Sleef0712dc2006-10-25 19:03:57 +0000507 }
David Reisscdffe262007-08-14 17:12:31 +0000508 $$ = $1;
Mark Slee9cb7c612006-09-01 22:17:45 +0000509 }
510
Mark Sleef0712dc2006-10-25 19:03:57 +0000511TypeDefinition:
512 Typedef
Mark Slee9cb7c612006-09-01 22:17:45 +0000513 {
Mark Sleef0712dc2006-10-25 19:03:57 +0000514 pdebug("TypeDefinition -> Typedef");
515 if (g_parse_mode == PROGRAM) {
516 g_program->add_typedef($1);
517 }
518 }
519| Enum
520 {
521 pdebug("TypeDefinition -> Enum");
522 if (g_parse_mode == PROGRAM) {
523 g_program->add_enum($1);
524 }
525 }
Mark Slee6a47fed2007-02-07 02:40:59 +0000526| Senum
527 {
528 pdebug("TypeDefinition -> Senum");
529 if (g_parse_mode == PROGRAM) {
530 g_program->add_typedef($1);
531 }
532 }
Mark Sleef0712dc2006-10-25 19:03:57 +0000533| Struct
534 {
535 pdebug("TypeDefinition -> Struct");
536 if (g_parse_mode == PROGRAM) {
537 g_program->add_struct($1);
538 }
539 }
540| Xception
Mark Slee27ed6ec2007-08-16 01:26:31 +0000541 {
Mark Sleef0712dc2006-10-25 19:03:57 +0000542 pdebug("TypeDefinition -> Xception");
543 if (g_parse_mode == PROGRAM) {
544 g_program->add_xception($1);
545 }
Mark Slee9cb7c612006-09-01 22:17:45 +0000546 }
Mark Slee31985722006-05-24 21:45:31 +0000547
Jens Geyer089bcd32014-09-11 22:36:41 +0200548CommaOrSemicolonOptional:
549 ','
550 {}
551| ';'
552 {}
553|
554 {}
555
Mark Slee31985722006-05-24 21:45:31 +0000556Typedef:
Jens Geyer089bcd32014-09-11 22:36:41 +0200557 tok_typedef FieldType tok_identifier TypeAnnotations CommaOrSemicolonOptional
Mark Slee31985722006-05-24 21:45:31 +0000558 {
David Reiss4dd78012010-03-09 05:19:08 +0000559 pdebug("TypeDef -> tok_typedef FieldType tok_identifier");
Jens Geyer12c09f42013-08-25 14:16:27 +0200560 validate_simple_identifier( $3);
David Reisscdffe262007-08-14 17:12:31 +0000561 t_typedef *td = new t_typedef(g_program, $2, $3);
Mark Slee31985722006-05-24 21:45:31 +0000562 $$ = td;
Roger Meier30877382012-09-17 21:18:05 +0000563 if ($4 != NULL) {
564 $$->annotations_ = $4->annotations_;
565 delete $4;
566 }
Mark Slee31985722006-05-24 21:45:31 +0000567 }
568
569Enum:
Roger Meier30877382012-09-17 21:18:05 +0000570 tok_enum tok_identifier '{' EnumDefList '}' TypeAnnotations
Mark Slee31985722006-05-24 21:45:31 +0000571 {
572 pdebug("Enum -> tok_enum tok_identifier { EnumDefList }");
David Reisscdffe262007-08-14 17:12:31 +0000573 $$ = $4;
Jens Geyer12c09f42013-08-25 14:16:27 +0200574 validate_simple_identifier( $2);
David Reisscdffe262007-08-14 17:12:31 +0000575 $$->set_name($2);
Roger Meier30877382012-09-17 21:18:05 +0000576 if ($6 != NULL) {
577 $$->annotations_ = $6->annotations_;
578 delete $6;
579 }
Jens Geyerae0b22c2014-09-04 23:04:21 +0200580
Bryan Duxbury9f0a7862010-09-12 14:38:36 +0000581 // make constants for all the enum values
582 if (g_parse_mode == PROGRAM) {
583 const std::vector<t_enum_value*>& enum_values = $$->get_constants();
584 std::vector<t_enum_value*>::const_iterator c_iter;
585 for (c_iter = enum_values.begin(); c_iter != enum_values.end(); ++c_iter) {
586 std::string const_name = $$->get_name() + "." + (*c_iter)->get_name();
587 t_const_value* const_val = new t_const_value((*c_iter)->get_value());
588 const_val->set_enum($$);
589 g_scope->add_constant(const_name, new t_const(g_type_i32, (*c_iter)->get_name(), const_val));
590 if (g_parent_scope != NULL) {
591 g_parent_scope->add_constant(g_parent_prefix + const_name, new t_const(g_type_i32, (*c_iter)->get_name(), const_val));
592 }
593 }
594 }
Mark Slee31985722006-05-24 21:45:31 +0000595 }
596
597EnumDefList:
Mark Slee207cb462006-11-02 18:43:12 +0000598 EnumDefList EnumDef
Mark Slee31985722006-05-24 21:45:31 +0000599 {
600 pdebug("EnumDefList -> EnumDefList EnumDef");
601 $$ = $1;
Mark Slee207cb462006-11-02 18:43:12 +0000602 $$->append($2);
Mark Slee31985722006-05-24 21:45:31 +0000603 }
604|
605 {
606 pdebug("EnumDefList -> ");
Mark Sleef0712dc2006-10-25 19:03:57 +0000607 $$ = new t_enum(g_program);
Jens Geyerae0b22c2014-09-04 23:04:21 +0200608 y_enum_val = -1;
Mark Slee31985722006-05-24 21:45:31 +0000609 }
610
611EnumDef:
Jens Geyerae0b22c2014-09-04 23:04:21 +0200612 CaptureDocText EnumValue TypeAnnotations CommaOrSemicolonOptional
Mark Slee31985722006-05-24 21:45:31 +0000613 {
Jens Geyerae0b22c2014-09-04 23:04:21 +0200614 pdebug("EnumDef -> EnumValue");
615 $$ = $2;
ccheeverf53b5cf2007-02-05 20:33:11 +0000616 if ($1 != NULL) {
617 $$->set_doc($1);
618 }
Jens Geyerae0b22c2014-09-04 23:04:21 +0200619 if (g_parse_mode == PROGRAM) {
620 // The scope constants never get deleted, so it's okay for us
621 // to share a single t_const object between our scope and the parent
622 // scope
623 t_const* constant = new t_const(g_type_i32, $2->get_name(),
624 new t_const_value($2->get_value()));
625 g_scope->add_constant($2->get_name(), constant);
626 if (g_parent_scope != NULL) {
627 g_parent_scope->add_constant(g_parent_prefix + $2->get_name(),
628 constant);
629 }
ccheeverf53b5cf2007-02-05 20:33:11 +0000630 }
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
Jens Geyerae0b22c2014-09-04 23:04:21 +0200637EnumValue:
638 tok_identifier '=' tok_int_constant
639 {
640 pdebug("EnumValue -> tok_identifier = tok_int_constant");
641 if ($3 < INT32_MIN || $3 > INT32_MAX) {
642 // Note: this used to be just a warning. However, since thrift always
643 // treats enums as i32 values, I'm changing it to a fatal error.
644 // I doubt this will affect many people, but users who run into this
645 // will have to update their thrift files to manually specify the
646 // truncated i32 value that thrift has always been using anyway.
647 failure("64-bit value supplied for enum %s will be truncated.", $1);
648 }
649 y_enum_val = static_cast<int32_t>($3);
650 $$ = new t_enum_value($1, y_enum_val);
651 }
652 |
653 tok_identifier
654 {
655 pdebug("EnumValue -> tok_identifier");
656 validate_simple_identifier( $1);
657 if (y_enum_val == INT32_MAX) {
658 failure("enum value overflow at enum %s", $1);
659 }
660 ++y_enum_val;
661 $$ = new t_enum_value($1, y_enum_val);
662 }
663
Mark Slee6a47fed2007-02-07 02:40:59 +0000664Senum:
Roger Meier30877382012-09-17 21:18:05 +0000665 tok_senum tok_identifier '{' SenumDefList '}' TypeAnnotations
Mark Slee6a47fed2007-02-07 02:40:59 +0000666 {
667 pdebug("Senum -> tok_senum tok_identifier { SenumDefList }");
Jens Geyer12c09f42013-08-25 14:16:27 +0200668 validate_simple_identifier( $2);
David Reisscdffe262007-08-14 17:12:31 +0000669 $$ = new t_typedef(g_program, $4, $2);
Roger Meier30877382012-09-17 21:18:05 +0000670 if ($6 != NULL) {
671 $$->annotations_ = $6->annotations_;
672 delete $6;
673 }
Mark Slee6a47fed2007-02-07 02:40:59 +0000674 }
675
676SenumDefList:
677 SenumDefList SenumDef
678 {
679 pdebug("SenumDefList -> SenumDefList SenumDef");
680 $$ = $1;
681 $$->add_string_enum_val($2);
682 }
683|
684 {
685 pdebug("SenumDefList -> ");
686 $$ = new t_base_type("string", t_base_type::TYPE_STRING);
687 $$->set_string_enum(true);
688 }
689
690SenumDef:
691 tok_literal CommaOrSemicolonOptional
692 {
693 pdebug("SenumDef -> tok_literal");
694 $$ = $1;
695 }
696
Mark Slee30152872006-11-28 01:24:07 +0000697Const:
698 tok_const FieldType tok_identifier '=' ConstValue CommaOrSemicolonOptional
699 {
700 pdebug("Const -> tok_const FieldType tok_identifier = ConstValue");
Mark Sleeaa7671d2006-11-29 03:19:31 +0000701 if (g_parse_mode == PROGRAM) {
Jens Geyer12c09f42013-08-25 14:16:27 +0200702 validate_simple_identifier( $3);
Bryan Duxbury2d804702009-12-18 19:41:11 +0000703 g_scope->resolve_const_value($5, $2);
Mark Sleeaa7671d2006-11-29 03:19:31 +0000704 $$ = new t_const($2, $3, $5);
705 validate_const_type($$);
Mark Sleed0767c52007-07-27 22:14:41 +0000706
707 g_scope->add_constant($3, $$);
708 if (g_parent_scope != NULL) {
709 g_parent_scope->add_constant(g_parent_prefix + $3, $$);
710 }
Mark Sleeaa7671d2006-11-29 03:19:31 +0000711 } else {
712 $$ = NULL;
713 }
Mark Slee30152872006-11-28 01:24:07 +0000714 }
715
716ConstValue:
717 tok_int_constant
718 {
719 pdebug("ConstValue => tok_int_constant");
720 $$ = new t_const_value();
721 $$->set_integer($1);
Roger Meier887ff752011-08-19 11:25:39 +0000722 if (!g_allow_64bit_consts && ($1 < INT32_MIN || $1 > INT32_MAX)) {
Roger Meier5f2d34e2013-11-16 16:43:41 +0100723 pwarning(1, "64-bit constant \"%" PRIi64"\" may not work in all languages.\n", $1);
David Reissf1454162008-06-30 20:45:47 +0000724 }
Mark Slee30152872006-11-28 01:24:07 +0000725 }
726| tok_dub_constant
727 {
728 pdebug("ConstValue => tok_dub_constant");
729 $$ = new t_const_value();
730 $$->set_double($1);
731 }
732| tok_literal
733 {
734 pdebug("ConstValue => tok_literal");
Mark Sleed0767c52007-07-27 22:14:41 +0000735 $$ = new t_const_value($1);
Mark Slee30152872006-11-28 01:24:07 +0000736 }
Mark Slee67fc6342006-11-29 03:37:04 +0000737| tok_identifier
738 {
739 pdebug("ConstValue => tok_identifier");
Bryan Duxbury2d804702009-12-18 19:41:11 +0000740 $$ = new t_const_value();
741 $$->set_identifier($1);
Mark Slee67fc6342006-11-29 03:37:04 +0000742 }
Mark Slee30152872006-11-28 01:24:07 +0000743| ConstList
744 {
745 pdebug("ConstValue => ConstList");
746 $$ = $1;
747 }
748| ConstMap
749 {
750 pdebug("ConstValue => ConstMap");
Mark Slee27ed6ec2007-08-16 01:26:31 +0000751 $$ = $1;
Mark Slee30152872006-11-28 01:24:07 +0000752 }
753
754ConstList:
755 '[' ConstListContents ']'
756 {
757 pdebug("ConstList => [ ConstListContents ]");
758 $$ = $2;
759 }
760
761ConstListContents:
762 ConstListContents ConstValue CommaOrSemicolonOptional
763 {
764 pdebug("ConstListContents => ConstListContents ConstValue CommaOrSemicolonOptional");
765 $$ = $1;
766 $$->add_list($2);
767 }
768|
769 {
770 pdebug("ConstListContents =>");
771 $$ = new t_const_value();
772 $$->set_list();
773 }
774
775ConstMap:
776 '{' ConstMapContents '}'
777 {
778 pdebug("ConstMap => { ConstMapContents }");
779 $$ = $2;
780 }
781
782ConstMapContents:
783 ConstMapContents ConstValue ':' ConstValue CommaOrSemicolonOptional
784 {
785 pdebug("ConstMapContents => ConstMapContents ConstValue CommaOrSemicolonOptional");
786 $$ = $1;
787 $$->add_map($2, $4);
788 }
789|
790 {
791 pdebug("ConstMapContents =>");
792 $$ = new t_const_value();
793 $$->set_map();
Mark Slee31985722006-05-24 21:45:31 +0000794 }
795
Bryan Duxburyab3666e2009-09-01 23:03:47 +0000796StructHead:
797 tok_struct
798 {
799 $$ = struct_is_struct;
800 }
801| tok_union
802 {
803 $$ = struct_is_union;
804 }
805
Mark Slee31985722006-05-24 21:45:31 +0000806Struct:
Bryan Duxburyab3666e2009-09-01 23:03:47 +0000807 StructHead tok_identifier XsdAll '{' FieldList '}' TypeAnnotations
Mark Slee31985722006-05-24 21:45:31 +0000808 {
809 pdebug("Struct -> tok_struct tok_identifier { FieldList }");
Jens Geyer12c09f42013-08-25 14:16:27 +0200810 validate_simple_identifier( $2);
David Reisscdffe262007-08-14 17:12:31 +0000811 $5->set_xsd_all($3);
Bryan Duxburyab3666e2009-09-01 23:03:47 +0000812 $5->set_union($1 == struct_is_union);
David Reisscdffe262007-08-14 17:12:31 +0000813 $$ = $5;
David Reissa2309992008-12-10 01:52:48 +0000814 $$->set_name($2);
815 if ($7 != NULL) {
816 $$->annotations_ = $7->annotations_;
817 delete $7;
818 }
Mark Slee9cb7c612006-09-01 22:17:45 +0000819 }
Ben Craige9576752013-10-11 08:19:16 -0500820
Mark Slee36bfa2e2007-01-19 20:09:51 +0000821XsdAll:
Mark Slee782abbb2007-01-19 00:17:02 +0000822 tok_xsd_all
823 {
824 $$ = true;
825 }
826|
827 {
828 $$ = false;
829 }
830
Mark Slee36bfa2e2007-01-19 20:09:51 +0000831XsdOptional:
832 tok_xsd_optional
833 {
834 $$ = true;
835 }
836|
837 {
838 $$ = false;
839 }
840
Mark Slee7df0e2a2007-02-06 21:03:18 +0000841XsdNillable:
842 tok_xsd_nillable
843 {
844 $$ = true;
845 }
846|
847 {
848 $$ = false;
849 }
850
Mark Slee21135c32007-02-05 21:52:08 +0000851XsdAttributes:
Mark Slee748d83f2007-02-07 01:20:08 +0000852 tok_xsd_attrs '{' FieldList '}'
Mark Slee21135c32007-02-05 21:52:08 +0000853 {
Mark Slee748d83f2007-02-07 01:20:08 +0000854 $$ = $3;
Mark Slee21135c32007-02-05 21:52:08 +0000855 }
856|
857 {
858 $$ = NULL;
859 }
860
Mark Slee9cb7c612006-09-01 22:17:45 +0000861Xception:
Roger Meier30877382012-09-17 21:18:05 +0000862 tok_xception tok_identifier '{' FieldList '}' TypeAnnotations
Mark Slee9cb7c612006-09-01 22:17:45 +0000863 {
864 pdebug("Xception -> tok_xception tok_identifier { FieldList }");
Jens Geyer12c09f42013-08-25 14:16:27 +0200865 validate_simple_identifier( $2);
Mark Slee9cb7c612006-09-01 22:17:45 +0000866 $4->set_name($2);
867 $4->set_xception(true);
868 $$ = $4;
Roger Meier30877382012-09-17 21:18:05 +0000869 if ($6 != NULL) {
870 $$->annotations_ = $6->annotations_;
871 delete $6;
872 }
Mark Slee31985722006-05-24 21:45:31 +0000873 }
874
875Service:
Roger Meier30877382012-09-17 21:18:05 +0000876 tok_service tok_identifier Extends '{' FlagArgs FunctionList UnflagArgs '}' TypeAnnotations
Mark Slee31985722006-05-24 21:45:31 +0000877 {
878 pdebug("Service -> tok_service tok_identifier { FunctionList }");
Jens Geyer12c09f42013-08-25 14:16:27 +0200879 validate_simple_identifier( $2);
Mark Slee78165722007-09-10 22:08:49 +0000880 $$ = $6;
David Reisscdffe262007-08-14 17:12:31 +0000881 $$->set_name($2);
882 $$->set_extends($3);
Roger Meier30877382012-09-17 21:18:05 +0000883 if ($9 != NULL) {
884 $$->annotations_ = $9->annotations_;
885 delete $9;
886 }
Mark Sleef0712dc2006-10-25 19:03:57 +0000887 }
888
Mark Slee78165722007-09-10 22:08:49 +0000889FlagArgs:
890 {
891 g_arglist = 1;
892 }
893
894UnflagArgs:
895 {
896 g_arglist = 0;
897 }
898
Mark Slee36bfa2e2007-01-19 20:09:51 +0000899Extends:
Mark Sleef0712dc2006-10-25 19:03:57 +0000900 tok_extends tok_identifier
901 {
Mark Slee36bfa2e2007-01-19 20:09:51 +0000902 pdebug("Extends -> tok_extends tok_identifier");
Mark Sleef0712dc2006-10-25 19:03:57 +0000903 $$ = NULL;
904 if (g_parse_mode == PROGRAM) {
905 $$ = g_scope->get_service($2);
906 if ($$ == NULL) {
907 yyerror("Service \"%s\" has not been defined.", $2);
908 exit(1);
909 }
910 }
911 }
912|
913 {
914 $$ = NULL;
Mark Slee31985722006-05-24 21:45:31 +0000915 }
916
917FunctionList:
Mark Slee207cb462006-11-02 18:43:12 +0000918 FunctionList Function
Mark Slee31985722006-05-24 21:45:31 +0000919 {
920 pdebug("FunctionList -> FunctionList Function");
921 $$ = $1;
922 $1->add_function($2);
923 }
924|
925 {
926 pdebug("FunctionList -> ");
Mark Sleef0712dc2006-10-25 19:03:57 +0000927 $$ = new t_service(g_program);
Mark Slee31985722006-05-24 21:45:31 +0000928 }
929
930Function:
Roger Meier30877382012-09-17 21:18:05 +0000931 CaptureDocText Oneway FunctionType tok_identifier '(' FieldList ')' Throws TypeAnnotations CommaOrSemicolonOptional
Mark Slee31985722006-05-24 21:45:31 +0000932 {
Jens Geyer12c09f42013-08-25 14:16:27 +0200933 validate_simple_identifier( $4);
ccheeverf53b5cf2007-02-05 20:33:11 +0000934 $6->set_name(std::string($4) + "_args");
935 $$ = new t_function($3, $4, $6, $8, $2);
936 if ($1 != NULL) {
937 $$->set_doc($1);
938 }
Roger Meier30877382012-09-17 21:18:05 +0000939 if ($9 != NULL) {
940 $$->annotations_ = $9->annotations_;
941 delete $9;
942 }
Mark Slee31985722006-05-24 21:45:31 +0000943 }
944
David Reiss6985a422009-03-24 20:00:47 +0000945Oneway:
946 tok_oneway
Mark Slee31985722006-05-24 21:45:31 +0000947 {
Mark Slee52f643d2006-08-09 00:03:43 +0000948 $$ = true;
949 }
950|
951 {
952 $$ = false;
Mark Slee31985722006-05-24 21:45:31 +0000953 }
954
Mark Slee36bfa2e2007-01-19 20:09:51 +0000955Throws:
Mark Slee9cb7c612006-09-01 22:17:45 +0000956 tok_throws '(' FieldList ')'
957 {
Mark Slee36bfa2e2007-01-19 20:09:51 +0000958 pdebug("Throws -> tok_throws ( FieldList )");
Mark Slee9cb7c612006-09-01 22:17:45 +0000959 $$ = $3;
Mark Sleef07d48e2008-02-01 01:36:26 +0000960 if (g_parse_mode == PROGRAM && !validate_throws($$)) {
Mark Slee91f2b7b2008-01-31 01:49:16 +0000961 yyerror("Throws clause may not contain non-exception types");
962 exit(1);
963 }
Mark Slee9cb7c612006-09-01 22:17:45 +0000964 }
965|
966 {
Mark Sleef0712dc2006-10-25 19:03:57 +0000967 $$ = new t_struct(g_program);
Mark Slee9cb7c612006-09-01 22:17:45 +0000968 }
969
Mark Slee31985722006-05-24 21:45:31 +0000970FieldList:
Mark Slee207cb462006-11-02 18:43:12 +0000971 FieldList Field
Mark Slee31985722006-05-24 21:45:31 +0000972 {
973 pdebug("FieldList -> FieldList , Field");
974 $$ = $1;
Bryan Duxburyff219ac2009-04-10 21:51:00 +0000975 if (!($$->append($2))) {
Jens Geyer3a67c2f2013-02-03 22:30:41 +0100976 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 +0000977 exit(1);
978 }
Mark Slee31985722006-05-24 21:45:31 +0000979 }
980|
981 {
982 pdebug("FieldList -> ");
David Reiss00a8dd62009-03-19 08:14:12 +0000983 y_field_val = -1;
Mark Sleef0712dc2006-10-25 19:03:57 +0000984 $$ = new t_struct(g_program);
Mark Slee31985722006-05-24 21:45:31 +0000985 }
986
987Field:
Jens Geyer885c6792014-05-02 21:31:55 +0200988 CaptureDocText FieldIdentifier FieldRequiredness FieldType FieldReference tok_identifier FieldValue XsdOptional XsdNillable XsdAttributes TypeAnnotations CommaOrSemicolonOptional
Mark Slee31985722006-05-24 21:45:31 +0000989 {
Mark Sleef0712dc2006-10-25 19:03:57 +0000990 pdebug("tok_int_constant : Field -> FieldType tok_identifier");
Bryan Duxburyc7206a42011-08-17 23:17:04 +0000991 if ($2.auto_assigned) {
Jens Geyer885c6792014-05-02 21:31:55 +0200992 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 +0000993 if (g_strict >= 192) {
994 yyerror("Implicit field keys are deprecated and not allowed with -strict");
995 exit(1);
996 }
Mark Slee31985722006-05-24 21:45:31 +0000997 }
Jens Geyer885c6792014-05-02 21:31:55 +0200998 validate_simple_identifier($6);
999 $$ = new t_field($4, $6, $2.value);
1000 $$->set_reference($5);
David Reiss8320a922007-08-14 19:59:26 +00001001 $$->set_req($3);
Jens Geyer885c6792014-05-02 21:31:55 +02001002 if ($7 != NULL) {
1003 g_scope->resolve_const_value($7, $4);
1004 validate_field_value($$, $7);
1005 $$->set_value($7);
Mark Slee7ff32452007-02-01 05:26:18 +00001006 }
Jens Geyer885c6792014-05-02 21:31:55 +02001007 $$->set_xsd_optional($8);
1008 $$->set_xsd_nillable($9);
ccheeverf53b5cf2007-02-05 20:33:11 +00001009 if ($1 != NULL) {
1010 $$->set_doc($1);
1011 }
David Reiss53c10e02010-03-05 07:51:51 +00001012 if ($10 != NULL) {
Jens Geyer885c6792014-05-02 21:31:55 +02001013 $$->set_xsd_attrs($10);
1014 }
1015 if ($11 != NULL) {
1016 $$->annotations_ = $11->annotations_;
1017 delete $11;
David Reiss53c10e02010-03-05 07:51:51 +00001018 }
Mark Slee31985722006-05-24 21:45:31 +00001019 }
Mark Slee7ff32452007-02-01 05:26:18 +00001020
1021FieldIdentifier:
1022 tok_int_constant ':'
Mark Slee31985722006-05-24 21:45:31 +00001023 {
Mark Slee7ff32452007-02-01 05:26:18 +00001024 if ($1 <= 0) {
Bryan Duxburyc7206a42011-08-17 23:17:04 +00001025 if (g_allow_neg_field_keys) {
1026 /*
1027 * g_allow_neg_field_keys exists to allow users to add explicitly
1028 * specified key values to old .thrift files without breaking
1029 * protocol compatibility.
1030 */
1031 if ($1 != y_field_val) {
1032 /*
1033 * warn if the user-specified negative value isn't what
1034 * thrift would have auto-assigned.
1035 */
Roger Meier5f2d34e2013-11-16 16:43:41 +01001036 pwarning(1, "Nonpositive field key (%" PRIi64") differs from what would be "
Bryan Duxburyc7206a42011-08-17 23:17:04 +00001037 "auto-assigned by thrift (%d).\n", $1, y_field_val);
1038 }
1039 /*
1040 * Leave $1 as-is, and update y_field_val to be one less than $1.
1041 * The FieldList parsing will catch any duplicate key values.
1042 */
Ben Craige9576752013-10-11 08:19:16 -05001043 y_field_val = static_cast<int32_t>($1 - 1);
1044 $$.value = static_cast<int32_t>($1);
Bryan Duxburyc7206a42011-08-17 23:17:04 +00001045 $$.auto_assigned = false;
1046 } else {
Ben Craige9576752013-10-11 08:19:16 -05001047 pwarning(1, "Nonpositive value (%d) not allowed as a field key.\n",
Bryan Duxburyc7206a42011-08-17 23:17:04 +00001048 $1);
1049 $$.value = y_field_val--;
1050 $$.auto_assigned = true;
1051 }
1052 } else {
Ben Craige9576752013-10-11 08:19:16 -05001053 $$.value = static_cast<int32_t>($1);
Bryan Duxburyc7206a42011-08-17 23:17:04 +00001054 $$.auto_assigned = false;
Mark Sleef0712dc2006-10-25 19:03:57 +00001055 }
Mark Slee7ff32452007-02-01 05:26:18 +00001056 }
1057|
1058 {
Bryan Duxburyc7206a42011-08-17 23:17:04 +00001059 $$.value = y_field_val--;
1060 $$.auto_assigned = true;
Mark Slee7ff32452007-02-01 05:26:18 +00001061 }
1062
Jens Geyer885c6792014-05-02 21:31:55 +02001063FieldReference:
1064 tok_reference
1065 {
1066 $$ = true;
1067 }
1068|
1069 {
1070 $$ = false;
1071 }
1072
David Reiss8320a922007-08-14 19:59:26 +00001073FieldRequiredness:
1074 tok_required
1075 {
David Reiss45603e92009-09-02 22:15:55 +00001076 $$ = t_field::T_REQUIRED;
David Reiss8320a922007-08-14 19:59:26 +00001077 }
1078| tok_optional
1079 {
Mark Slee78165722007-09-10 22:08:49 +00001080 if (g_arglist) {
1081 if (g_parse_mode == PROGRAM) {
1082 pwarning(1, "optional keyword is ignored in argument lists.\n");
1083 }
David Reiss204420f2008-01-11 20:59:03 +00001084 $$ = t_field::T_OPT_IN_REQ_OUT;
Mark Slee78165722007-09-10 22:08:49 +00001085 } else {
David Reiss204420f2008-01-11 20:59:03 +00001086 $$ = t_field::T_OPTIONAL;
Mark Slee78165722007-09-10 22:08:49 +00001087 }
David Reiss8320a922007-08-14 19:59:26 +00001088 }
1089|
1090 {
David Reiss204420f2008-01-11 20:59:03 +00001091 $$ = t_field::T_OPT_IN_REQ_OUT;
David Reiss8320a922007-08-14 19:59:26 +00001092 }
1093
Mark Slee7ff32452007-02-01 05:26:18 +00001094FieldValue:
1095 '=' ConstValue
1096 {
Mark Slee27ed6ec2007-08-16 01:26:31 +00001097 if (g_parse_mode == PROGRAM) {
Mark Slee7ff32452007-02-01 05:26:18 +00001098 $$ = $2;
1099 } else {
1100 $$ = NULL;
1101 }
1102 }
1103|
1104 {
1105 $$ = NULL;
Mark Sleef0712dc2006-10-25 19:03:57 +00001106 }
Mark Slee31985722006-05-24 21:45:31 +00001107
Mark Slee31985722006-05-24 21:45:31 +00001108FunctionType:
1109 FieldType
1110 {
Mark Sleee8540632006-05-30 09:24:40 +00001111 pdebug("FunctionType -> FieldType");
Mark Slee31985722006-05-24 21:45:31 +00001112 $$ = $1;
1113 }
1114| tok_void
1115 {
Mark Sleee8540632006-05-30 09:24:40 +00001116 pdebug("FunctionType -> tok_void");
Mark Sleef0712dc2006-10-25 19:03:57 +00001117 $$ = g_type_void;
Mark Slee31985722006-05-24 21:45:31 +00001118 }
1119
1120FieldType:
1121 tok_identifier
1122 {
Mark Sleee8540632006-05-30 09:24:40 +00001123 pdebug("FieldType -> tok_identifier");
Mark Sleef0712dc2006-10-25 19:03:57 +00001124 if (g_parse_mode == INCLUDES) {
1125 // Ignore identifiers in include mode
1126 $$ = NULL;
1127 } else {
1128 // Lookup the identifier in the current scope
1129 $$ = g_scope->get_type($1);
1130 if ($$ == NULL) {
jfarrelle0e83162014-04-08 22:45:01 -04001131 /*
1132 * Either this type isn't yet declared, or it's never
1133 declared. Either way allow it and we'll figure it out
1134 during generation.
1135 */
Jens Geyerd000b242014-04-13 21:58:47 +02001136 $$ = new t_typedef(g_program, $1, true);
Mark Sleef0712dc2006-10-25 19:03:57 +00001137 }
Mark Sleee8540632006-05-30 09:24:40 +00001138 }
Mark Slee31985722006-05-24 21:45:31 +00001139 }
1140| BaseType
1141 {
Mark Sleee8540632006-05-30 09:24:40 +00001142 pdebug("FieldType -> BaseType");
1143 $$ = $1;
1144 }
1145| ContainerType
1146 {
1147 pdebug("FieldType -> ContainerType");
Mark Slee31985722006-05-24 21:45:31 +00001148 $$ = $1;
1149 }
1150
David Reissc8e30052009-07-27 17:02:42 +00001151BaseType: SimpleBaseType TypeAnnotations
1152 {
1153 pdebug("BaseType -> SimpleBaseType TypeAnnotations");
1154 if ($2 != NULL) {
1155 $$ = new t_base_type(*static_cast<t_base_type*>($1));
1156 $$->annotations_ = $2->annotations_;
1157 delete $2;
1158 } else {
1159 $$ = $1;
1160 }
1161 }
1162
1163SimpleBaseType:
Mark Slee31985722006-05-24 21:45:31 +00001164 tok_string
1165 {
1166 pdebug("BaseType -> tok_string");
Mark Sleef0712dc2006-10-25 19:03:57 +00001167 $$ = g_type_string;
Mark Slee31985722006-05-24 21:45:31 +00001168 }
Mark Slee8d725a22007-04-13 01:57:12 +00001169| tok_binary
1170 {
1171 pdebug("BaseType -> tok_binary");
1172 $$ = g_type_binary;
1173 }
Mark Sleeb6200d82007-01-19 19:14:36 +00001174| tok_slist
1175 {
1176 pdebug("BaseType -> tok_slist");
1177 $$ = g_type_slist;
1178 }
Mark Slee78f58e22006-09-02 04:17:07 +00001179| tok_bool
1180 {
1181 pdebug("BaseType -> tok_bool");
Mark Sleef0712dc2006-10-25 19:03:57 +00001182 $$ = g_type_bool;
Mark Slee78f58e22006-09-02 04:17:07 +00001183 }
Mark Slee31985722006-05-24 21:45:31 +00001184| tok_byte
1185 {
1186 pdebug("BaseType -> tok_byte");
Mark Sleef0712dc2006-10-25 19:03:57 +00001187 $$ = g_type_byte;
Mark Slee31985722006-05-24 21:45:31 +00001188 }
Mark Slee9cb7c612006-09-01 22:17:45 +00001189| tok_i16
1190 {
1191 pdebug("BaseType -> tok_i16");
Mark Sleef0712dc2006-10-25 19:03:57 +00001192 $$ = g_type_i16;
Mark Slee9cb7c612006-09-01 22:17:45 +00001193 }
Mark Slee31985722006-05-24 21:45:31 +00001194| tok_i32
1195 {
1196 pdebug("BaseType -> tok_i32");
Mark Sleef0712dc2006-10-25 19:03:57 +00001197 $$ = g_type_i32;
Mark Slee31985722006-05-24 21:45:31 +00001198 }
Mark Slee31985722006-05-24 21:45:31 +00001199| tok_i64
1200 {
1201 pdebug("BaseType -> tok_i64");
Mark Sleef0712dc2006-10-25 19:03:57 +00001202 $$ = g_type_i64;
Mark Slee31985722006-05-24 21:45:31 +00001203 }
Mark Sleec98d0502006-09-06 02:42:25 +00001204| tok_double
1205 {
1206 pdebug("BaseType -> tok_double");
Mark Sleef0712dc2006-10-25 19:03:57 +00001207 $$ = g_type_double;
Mark Sleec98d0502006-09-06 02:42:25 +00001208 }
Mark Slee31985722006-05-24 21:45:31 +00001209
David Reissa2309992008-12-10 01:52:48 +00001210ContainerType: SimpleContainerType TypeAnnotations
1211 {
1212 pdebug("ContainerType -> SimpleContainerType TypeAnnotations");
1213 $$ = $1;
1214 if ($2 != NULL) {
1215 $$->annotations_ = $2->annotations_;
1216 delete $2;
1217 }
1218 }
1219
1220SimpleContainerType:
Mark Sleee8540632006-05-30 09:24:40 +00001221 MapType
1222 {
David Reissa2309992008-12-10 01:52:48 +00001223 pdebug("SimpleContainerType -> MapType");
Mark Sleee8540632006-05-30 09:24:40 +00001224 $$ = $1;
1225 }
1226| SetType
1227 {
David Reissa2309992008-12-10 01:52:48 +00001228 pdebug("SimpleContainerType -> SetType");
Mark Sleee8540632006-05-30 09:24:40 +00001229 $$ = $1;
1230 }
1231| ListType
1232 {
David Reissa2309992008-12-10 01:52:48 +00001233 pdebug("SimpleContainerType -> ListType");
Mark Sleee8540632006-05-30 09:24:40 +00001234 $$ = $1;
1235 }
1236
1237MapType:
Mark Slee36bfa2e2007-01-19 20:09:51 +00001238 tok_map CppType '<' FieldType ',' FieldType '>'
Mark Sleee8540632006-05-30 09:24:40 +00001239 {
1240 pdebug("MapType -> tok_map <FieldType, FieldType>");
Mark Slee4f8da1d2006-10-12 02:47:27 +00001241 $$ = new t_map($4, $6);
1242 if ($2 != NULL) {
1243 ((t_container*)$$)->set_cpp_name(std::string($2));
1244 }
Mark Sleee8540632006-05-30 09:24:40 +00001245 }
1246
1247SetType:
Mark Slee36bfa2e2007-01-19 20:09:51 +00001248 tok_set CppType '<' FieldType '>'
Mark Sleee8540632006-05-30 09:24:40 +00001249 {
1250 pdebug("SetType -> tok_set<FieldType>");
Mark Slee4f8da1d2006-10-12 02:47:27 +00001251 $$ = new t_set($4);
1252 if ($2 != NULL) {
1253 ((t_container*)$$)->set_cpp_name(std::string($2));
1254 }
Mark Sleee8540632006-05-30 09:24:40 +00001255 }
1256
1257ListType:
Mark Slee36bfa2e2007-01-19 20:09:51 +00001258 tok_list '<' FieldType '>' CppType
Mark Sleee8540632006-05-30 09:24:40 +00001259 {
1260 pdebug("ListType -> tok_list<FieldType>");
Jens Geyer6fe77e82014-03-16 16:48:53 +02001261 check_for_list_of_bytes($3);
Mark Sleef0712dc2006-10-25 19:03:57 +00001262 $$ = new t_list($3);
1263 if ($5 != NULL) {
1264 ((t_container*)$$)->set_cpp_name(std::string($5));
Mark Slee4f8da1d2006-10-12 02:47:27 +00001265 }
1266 }
1267
Mark Slee36bfa2e2007-01-19 20:09:51 +00001268CppType:
Mark Sleeafc76542007-02-09 21:55:44 +00001269 tok_cpp_type tok_literal
Mark Slee4f8da1d2006-10-12 02:47:27 +00001270 {
Mark Sleeafc76542007-02-09 21:55:44 +00001271 $$ = $2;
Mark Slee4f8da1d2006-10-12 02:47:27 +00001272 }
1273|
1274 {
1275 $$ = NULL;
Mark Sleee8540632006-05-30 09:24:40 +00001276 }
1277
David Reissa2309992008-12-10 01:52:48 +00001278TypeAnnotations:
1279 '(' TypeAnnotationList ')'
1280 {
1281 pdebug("TypeAnnotations -> ( TypeAnnotationList )");
1282 $$ = $2;
1283 }
1284|
1285 {
1286 $$ = NULL;
1287 }
1288
1289TypeAnnotationList:
1290 TypeAnnotationList TypeAnnotation
1291 {
1292 pdebug("TypeAnnotationList -> TypeAnnotationList , TypeAnnotation");
1293 $$ = $1;
1294 $$->annotations_[$2->key] = $2->val;
1295 delete $2;
1296 }
1297|
1298 {
1299 /* Just use a dummy structure to hold the annotations. */
1300 $$ = new t_struct(g_program);
1301 }
1302
1303TypeAnnotation:
Jens Geyer89847df2014-05-02 23:50:04 +02001304 tok_identifier TypeAnnotationValue CommaOrSemicolonOptional
David Reissa2309992008-12-10 01:52:48 +00001305 {
Jens Geyer89847df2014-05-02 23:50:04 +02001306 pdebug("TypeAnnotation -> TypeAnnotationValue");
David Reissa2309992008-12-10 01:52:48 +00001307 $$ = new t_annotation;
1308 $$->key = $1;
Jens Geyer89847df2014-05-02 23:50:04 +02001309 $$->val = $2;
1310 }
1311
1312TypeAnnotationValue:
1313 '=' tok_literal
1314 {
1315 pdebug("TypeAnnotationValue -> = tok_literal");
1316 $$ = $2;
1317 }
1318|
1319 {
1320 pdebug("TypeAnnotationValue ->");
1321 $$ = strdup("1");
David Reissa2309992008-12-10 01:52:48 +00001322 }
1323
Todd Lipcon53ae9f32009-12-07 00:42:38 +00001324%%