blob: 7bd189f63ef1ba8f8d5b5e09f877f04ef7bfdde6 [file] [log] [blame]
Mark Slee31985722006-05-24 21:45:31 +00001%{
Mark Sleee9ce01c2007-05-16 02:29:53 +00002// Copyright (c) 2006- Facebook
3// Distributed under the Thrift Software License
4//
5// See accompanying file LICENSE or visit the Thrift site at:
6// http://developers.facebook.com/thrift/
Mark Slee31985722006-05-24 21:45:31 +00007
8/**
9 * Thrift parser.
10 *
11 * This parser is used on a thrift definition file.
12 *
13 * @author Mark Slee <mcslee@facebook.com>
14 */
15
16#include <stdio.h>
17#include "main.h"
18#include "globals.h"
19#include "parse/t_program.h"
Mark Sleef0712dc2006-10-25 19:03:57 +000020#include "parse/t_scope.h"
Mark Slee31985722006-05-24 21:45:31 +000021
Mark Sleef5377b32006-10-10 01:42:59 +000022/**
23 * This global variable is used for automatic numbering of field indices etc.
24 * when parsing the members of a struct. Field values are automatically
25 * assigned starting from -1 and working their way down.
26 */
Mark Slee9cb7c612006-09-01 22:17:45 +000027int y_field_val = -1;
Mark Slee78165722007-09-10 22:08:49 +000028int g_arglist = 0;
Mark Slee31985722006-05-24 21:45:31 +000029
30%}
31
Mark Sleef5377b32006-10-10 01:42:59 +000032/**
33 * This structure is used by the parser to hold the data types associated with
34 * various parse nodes.
35 */
Mark Slee31985722006-05-24 21:45:31 +000036%union {
Mark Slee30152872006-11-28 01:24:07 +000037 char* id;
38 int iconst;
39 double dconst;
40 bool tbool;
David Reisscdffe262007-08-14 17:12:31 +000041 t_doc* tdoc;
Mark Slee30152872006-11-28 01:24:07 +000042 t_type* ttype;
Mark Slee6a47fed2007-02-07 02:40:59 +000043 t_base_type* tbase;
Mark Slee30152872006-11-28 01:24:07 +000044 t_typedef* ttypedef;
45 t_enum* tenum;
46 t_enum_value* tenumv;
47 t_const* tconst;
48 t_const_value* tconstv;
49 t_struct* tstruct;
50 t_service* tservice;
51 t_function* tfunction;
52 t_field* tfield;
David Reisscdffe262007-08-14 17:12:31 +000053 char* dtext;
David Reiss8320a922007-08-14 19:59:26 +000054 t_field::e_req ereq;
Mark Slee31985722006-05-24 21:45:31 +000055}
56
Mark Sleef5377b32006-10-10 01:42:59 +000057/**
58 * Strings identifier
59 */
Mark Slee31985722006-05-24 21:45:31 +000060%token<id> tok_identifier
Mark Sleef0712dc2006-10-25 19:03:57 +000061%token<id> tok_literal
David Reisscdffe262007-08-14 17:12:31 +000062%token<dtext> tok_doctext
Mark Sleebd588222007-11-21 08:43:35 +000063%token<id> tok_st_identifier
Mark Sleef5377b32006-10-10 01:42:59 +000064
65/**
Mark Slee30152872006-11-28 01:24:07 +000066 * Constant values
Mark Sleef5377b32006-10-10 01:42:59 +000067 */
Mark Slee31985722006-05-24 21:45:31 +000068%token<iconst> tok_int_constant
Mark Slee30152872006-11-28 01:24:07 +000069%token<dconst> tok_dub_constant
Mark Slee31985722006-05-24 21:45:31 +000070
Mark Sleef5377b32006-10-10 01:42:59 +000071/**
David Reiss399442b2008-02-20 02:28:05 +000072 * Header keywords
Mark Sleef5377b32006-10-10 01:42:59 +000073 */
Mark Sleef0712dc2006-10-25 19:03:57 +000074%token tok_include
Mark Slee9cb7c612006-09-01 22:17:45 +000075%token tok_namespace
Mark Sleef0712dc2006-10-25 19:03:57 +000076%token tok_cpp_namespace
77%token tok_cpp_include
78%token tok_cpp_type
Mark Sleee888b372007-01-12 01:06:24 +000079%token tok_php_namespace
David Reissc6fc3292007-08-30 00:58:43 +000080%token tok_py_module
Mark Slee27ed6ec2007-08-16 01:26:31 +000081%token tok_perl_package
Mark Sleef0712dc2006-10-25 19:03:57 +000082%token tok_java_package
Mark Slee782abbb2007-01-19 00:17:02 +000083%token tok_xsd_all
Mark Slee36bfa2e2007-01-19 20:09:51 +000084%token tok_xsd_optional
Mark Slee7df0e2a2007-02-06 21:03:18 +000085%token tok_xsd_nillable
Mark Slee0d9199e2007-01-31 02:08:30 +000086%token tok_xsd_namespace
Mark Slee21135c32007-02-05 21:52:08 +000087%token tok_xsd_attrs
Mark Slee58dfb4f2007-07-06 02:45:25 +000088%token tok_ruby_namespace
Mark Sleebd588222007-11-21 08:43:35 +000089%token tok_smalltalk_category
David Reiss15457c92007-12-14 07:03:03 +000090%token tok_smalltalk_prefix
Mark Slee7e9eea42007-09-10 21:00:23 +000091%token tok_cocoa_prefix
David Reiss7f42bcf2008-01-11 20:59:12 +000092%token tok_csharp_namespace
Mark Slee9cb7c612006-09-01 22:17:45 +000093
Mark Sleef5377b32006-10-10 01:42:59 +000094/**
95 * Base datatype keywords
96 */
97%token tok_void
Mark Slee78f58e22006-09-02 04:17:07 +000098%token tok_bool
Mark Slee31985722006-05-24 21:45:31 +000099%token tok_byte
100%token tok_string
Mark Slee8d725a22007-04-13 01:57:12 +0000101%token tok_binary
Mark Sleeb6200d82007-01-19 19:14:36 +0000102%token tok_slist
Mark Slee6a47fed2007-02-07 02:40:59 +0000103%token tok_senum
Mark Slee9cb7c612006-09-01 22:17:45 +0000104%token tok_i16
Mark Slee31985722006-05-24 21:45:31 +0000105%token tok_i32
Mark Slee31985722006-05-24 21:45:31 +0000106%token tok_i64
Mark Slee9cb7c612006-09-01 22:17:45 +0000107%token tok_double
Mark Slee31985722006-05-24 21:45:31 +0000108
Mark Sleef5377b32006-10-10 01:42:59 +0000109/**
110 * Complex type keywords
111 */
Mark Slee31985722006-05-24 21:45:31 +0000112%token tok_map
113%token tok_list
114%token tok_set
115
Mark Sleef5377b32006-10-10 01:42:59 +0000116/**
117 * Function modifiers
Mark Slee27ed6ec2007-08-16 01:26:31 +0000118 */
Mark Slee31985722006-05-24 21:45:31 +0000119%token tok_async
120
Mark Sleef5377b32006-10-10 01:42:59 +0000121/**
122 * Thrift language keywords
123 */
Mark Slee31985722006-05-24 21:45:31 +0000124%token tok_typedef
125%token tok_struct
Mark Slee9cb7c612006-09-01 22:17:45 +0000126%token tok_xception
127%token tok_throws
Mark Sleef0712dc2006-10-25 19:03:57 +0000128%token tok_extends
Mark Slee31985722006-05-24 21:45:31 +0000129%token tok_service
130%token tok_enum
Mark Slee30152872006-11-28 01:24:07 +0000131%token tok_const
David Reiss8320a922007-08-14 19:59:26 +0000132%token tok_required
133%token tok_optional
Mark Slee31985722006-05-24 21:45:31 +0000134
Mark Sleef5377b32006-10-10 01:42:59 +0000135/**
136 * Grammar nodes
137 */
138
Mark Slee31985722006-05-24 21:45:31 +0000139%type<ttype> BaseType
Mark Sleee8540632006-05-30 09:24:40 +0000140%type<ttype> ContainerType
141%type<ttype> MapType
142%type<ttype> SetType
143%type<ttype> ListType
Mark Slee31985722006-05-24 21:45:31 +0000144
David Reisscdffe262007-08-14 17:12:31 +0000145%type<tdoc> Definition
Mark Sleef0712dc2006-10-25 19:03:57 +0000146%type<ttype> TypeDefinition
147
Mark Slee31985722006-05-24 21:45:31 +0000148%type<ttypedef> Typedef
149%type<ttype> DefinitionType
150
151%type<tfield> Field
Mark Slee7ff32452007-02-01 05:26:18 +0000152%type<iconst> FieldIdentifier
David Reiss8320a922007-08-14 19:59:26 +0000153%type<ereq> FieldRequiredness
Mark Slee31985722006-05-24 21:45:31 +0000154%type<ttype> FieldType
Mark Slee7ff32452007-02-01 05:26:18 +0000155%type<tconstv> FieldValue
Mark Sleee8540632006-05-30 09:24:40 +0000156%type<tstruct> FieldList
Mark Slee31985722006-05-24 21:45:31 +0000157
158%type<tenum> Enum
159%type<tenum> EnumDefList
Mark Slee30152872006-11-28 01:24:07 +0000160%type<tenumv> EnumDef
161
Mark Slee6a47fed2007-02-07 02:40:59 +0000162%type<ttypedef> Senum
163%type<tbase> SenumDefList
164%type<id> SenumDef
165
Mark Slee30152872006-11-28 01:24:07 +0000166%type<tconst> Const
167%type<tconstv> ConstValue
168%type<tconstv> ConstList
169%type<tconstv> ConstListContents
170%type<tconstv> ConstMap
171%type<tconstv> ConstMapContents
Mark Slee31985722006-05-24 21:45:31 +0000172
173%type<tstruct> Struct
Mark Slee9cb7c612006-09-01 22:17:45 +0000174%type<tstruct> Xception
Mark Slee31985722006-05-24 21:45:31 +0000175%type<tservice> Service
176
177%type<tfunction> Function
Mark Slee31985722006-05-24 21:45:31 +0000178%type<ttype> FunctionType
179%type<tservice> FunctionList
180
Mark Slee36bfa2e2007-01-19 20:09:51 +0000181%type<tstruct> Throws
182%type<tservice> Extends
183%type<tbool> Async
184%type<tbool> XsdAll
185%type<tbool> XsdOptional
Mark Slee7df0e2a2007-02-06 21:03:18 +0000186%type<tbool> XsdNillable
Mark Slee748d83f2007-02-07 01:20:08 +0000187%type<tstruct> XsdAttributes
Mark Slee36bfa2e2007-01-19 20:09:51 +0000188%type<id> CppType
Mark Slee52f643d2006-08-09 00:03:43 +0000189
David Reisscbd4bac2007-08-14 17:12:33 +0000190%type<dtext> CaptureDocText
ccheeverf53b5cf2007-02-05 20:33:11 +0000191
Mark Slee31985722006-05-24 21:45:31 +0000192%%
193
Mark Sleef5377b32006-10-10 01:42:59 +0000194/**
195 * Thrift Grammar Implementation.
196 *
197 * For the most part this source file works its way top down from what you
198 * might expect to find in a typical .thrift file, i.e. type definitions and
199 * namespaces up top followed by service definitions using those types.
200 */
Mark Slee31985722006-05-24 21:45:31 +0000201
202Program:
David Reisscbd4bac2007-08-14 17:12:33 +0000203 HeaderList DefinitionList
Mark Sleef0712dc2006-10-25 19:03:57 +0000204 {
205 pdebug("Program -> Headers DefinitionList");
David Reisscbd4bac2007-08-14 17:12:33 +0000206 /*
207 TODO(dreiss): Decide whether full-program doctext is worth the trouble.
David Reissc2532a92007-07-30 23:46:11 +0000208 if ($1 != NULL) {
209 g_program->set_doc($1);
210 }
David Reisscbd4bac2007-08-14 17:12:33 +0000211 */
212 clear_doctext();
Mark Sleef0712dc2006-10-25 19:03:57 +0000213 }
214
David Reisscbd4bac2007-08-14 17:12:33 +0000215CaptureDocText:
216 {
217 if (g_parse_mode == PROGRAM) {
Mark Sleebd588222007-11-21 08:43:35 +0000218 $$ = g_doctext;
David Reisscbd4bac2007-08-14 17:12:33 +0000219 g_doctext = NULL;
Mark Slee27ed6ec2007-08-16 01:26:31 +0000220 } else {
David Reisscbd4bac2007-08-14 17:12:33 +0000221 $$ = NULL;
222 }
223 }
224
225/* TODO(dreiss): Try to DestroyDocText in all sorts or random places. */
226DestroyDocText:
227 {
228 if (g_parse_mode == PROGRAM) {
229 clear_doctext();
230 }
231 }
232
233/* We have to DestroyDocText here, otherwise it catches the doctext
234 on the first real element. */
Mark Sleef0712dc2006-10-25 19:03:57 +0000235HeaderList:
David Reisscbd4bac2007-08-14 17:12:33 +0000236 HeaderList DestroyDocText Header
Mark Sleef0712dc2006-10-25 19:03:57 +0000237 {
238 pdebug("HeaderList -> HeaderList Header");
239 }
240|
241 {
242 pdebug("HeaderList -> ");
243 }
244
245Header:
246 Include
247 {
248 pdebug("Header -> Include");
249 }
David Reiss79eca142008-02-27 01:55:13 +0000250| tok_namespace tok_identifier tok_identifier
Mark Sleef0712dc2006-10-25 19:03:57 +0000251 {
David Reiss79eca142008-02-27 01:55:13 +0000252 pdebug("Header -> tok_namespace tok_identifier tok_identifier");
Mark Sleef0712dc2006-10-25 19:03:57 +0000253 if (g_parse_mode == PROGRAM) {
David Reiss79eca142008-02-27 01:55:13 +0000254 g_program->set_namespace($2, $3);
Mark Sleef0712dc2006-10-25 19:03:57 +0000255 }
256 }
257| tok_cpp_namespace tok_identifier
258 {
259 pdebug("Header -> tok_cpp_namespace tok_identifier");
260 if (g_parse_mode == PROGRAM) {
261 g_program->set_cpp_namespace($2);
262 }
263 }
264| tok_cpp_include tok_literal
265 {
266 pdebug("Header -> tok_cpp_include tok_literal");
267 if (g_parse_mode == PROGRAM) {
268 g_program->add_cpp_include($2);
269 }
270 }
Mark Sleee888b372007-01-12 01:06:24 +0000271| tok_php_namespace tok_identifier
272 {
273 pdebug("Header -> tok_php_namespace tok_identifier");
274 if (g_parse_mode == PROGRAM) {
275 g_program->set_php_namespace($2);
276 }
277 }
David Reissc6fc3292007-08-30 00:58:43 +0000278| tok_py_module tok_identifier
279 {
280 pdebug("Header -> tok_py_module tok_identifier");
281 if (g_parse_mode == PROGRAM) {
282 g_program->set_py_module($2);
283 }
284 }
Mark Slee27ed6ec2007-08-16 01:26:31 +0000285| tok_perl_package tok_identifier
286 {
287 pdebug("Header -> tok_perl_namespace tok_identifier");
288 if (g_parse_mode == PROGRAM) {
289 g_program->set_perl_package($2);
290 }
291 }
Mark Slee58dfb4f2007-07-06 02:45:25 +0000292| tok_ruby_namespace tok_identifier
293 {
294 pdebug("Header -> tok_ruby_namespace tok_identifier");
295 if (g_parse_mode == PROGRAM) {
296 g_program->set_ruby_namespace($2);
297 }
298 }
Mark Sleebd588222007-11-21 08:43:35 +0000299| tok_smalltalk_category tok_st_identifier
300 {
301 pdebug("Header -> tok_smalltalk_category tok_st_identifier");
302 if (g_parse_mode == PROGRAM) {
303 g_program->set_smalltalk_category($2);
304 }
305 }
David Reiss15457c92007-12-14 07:03:03 +0000306| tok_smalltalk_prefix tok_identifier
307 {
308 pdebug("Header -> tok_smalltalk_prefix tok_identifier");
309 if (g_parse_mode == PROGRAM) {
310 g_program->set_smalltalk_prefix($2);
311 }
312 }
Mark Sleef0712dc2006-10-25 19:03:57 +0000313| tok_java_package tok_identifier
314 {
315 pdebug("Header -> tok_java_package tok_identifier");
316 if (g_parse_mode == PROGRAM) {
317 g_program->set_java_package($2);
318 }
319 }
Mark Slee7e9eea42007-09-10 21:00:23 +0000320| tok_cocoa_prefix tok_identifier
321 {
322 pdebug("Header -> tok_cocoa_prefix tok_identifier");
323 if (g_parse_mode == PROGRAM) {
324 g_program->set_cocoa_prefix($2);
325 }
326 }
Mark Slee0d9199e2007-01-31 02:08:30 +0000327| tok_xsd_namespace tok_literal
328 {
329 pdebug("Header -> tok_xsd_namespace tok_literal");
330 if (g_parse_mode == PROGRAM) {
331 g_program->set_xsd_namespace($2);
332 }
333 }
David Reiss7f42bcf2008-01-11 20:59:12 +0000334| tok_csharp_namespace tok_identifier
335 {
336 pdebug("Header -> tok_csharp_package tok_identifier");
337 if (g_parse_mode == PROGRAM) {
338 g_program->set_csharp_namespace($2);
339 }
340 }
Mark Sleef0712dc2006-10-25 19:03:57 +0000341
342Include:
343 tok_include tok_literal
344 {
Mark Slee27ed6ec2007-08-16 01:26:31 +0000345 pdebug("Include -> tok_include tok_literal");
Mark Sleef0712dc2006-10-25 19:03:57 +0000346 if (g_parse_mode == INCLUDES) {
347 std::string path = include_file(std::string($2));
348 if (!path.empty()) {
kholst76f2c882008-01-16 02:47:41 +0000349 g_program->add_include(path, std::string($2));
Mark Sleef0712dc2006-10-25 19:03:57 +0000350 }
351 }
352 }
Mark Slee31985722006-05-24 21:45:31 +0000353
354DefinitionList:
David Reisscbd4bac2007-08-14 17:12:33 +0000355 DefinitionList CaptureDocText Definition
Mark Slee31985722006-05-24 21:45:31 +0000356 {
357 pdebug("DefinitionList -> DefinitionList Definition");
David Reisscdffe262007-08-14 17:12:31 +0000358 if ($2 != NULL && $3 != NULL) {
359 $3->set_doc($2);
360 }
Mark Slee31985722006-05-24 21:45:31 +0000361 }
362|
363 {
364 pdebug("DefinitionList -> ");
365 }
366
367Definition:
Mark Slee30152872006-11-28 01:24:07 +0000368 Const
369 {
370 pdebug("Definition -> Const");
371 if (g_parse_mode == PROGRAM) {
372 g_program->add_const($1);
Mark Sleebd588222007-11-21 08:43:35 +0000373 }
David Reisscdffe262007-08-14 17:12:31 +0000374 $$ = $1;
Mark Slee30152872006-11-28 01:24:07 +0000375 }
376| TypeDefinition
Mark Slee9cb7c612006-09-01 22:17:45 +0000377 {
Mark Sleef0712dc2006-10-25 19:03:57 +0000378 pdebug("Definition -> TypeDefinition");
379 if (g_parse_mode == PROGRAM) {
380 g_scope->add_type($1->get_name(), $1);
381 if (g_parent_scope != NULL) {
382 g_parent_scope->add_type(g_parent_prefix + $1->get_name(), $1);
383 }
384 }
David Reisscdffe262007-08-14 17:12:31 +0000385 $$ = $1;
Mark Slee9cb7c612006-09-01 22:17:45 +0000386 }
Mark Slee31985722006-05-24 21:45:31 +0000387| Service
388 {
389 pdebug("Definition -> Service");
Mark Sleef0712dc2006-10-25 19:03:57 +0000390 if (g_parse_mode == PROGRAM) {
391 g_scope->add_service($1->get_name(), $1);
392 if (g_parent_scope != NULL) {
393 g_parent_scope->add_service(g_parent_prefix + $1->get_name(), $1);
394 }
395 g_program->add_service($1);
396 }
David Reisscdffe262007-08-14 17:12:31 +0000397 $$ = $1;
Mark Slee9cb7c612006-09-01 22:17:45 +0000398 }
399
Mark Sleef0712dc2006-10-25 19:03:57 +0000400TypeDefinition:
401 Typedef
Mark Slee9cb7c612006-09-01 22:17:45 +0000402 {
Mark Sleef0712dc2006-10-25 19:03:57 +0000403 pdebug("TypeDefinition -> Typedef");
404 if (g_parse_mode == PROGRAM) {
405 g_program->add_typedef($1);
406 }
407 }
408| Enum
409 {
410 pdebug("TypeDefinition -> Enum");
411 if (g_parse_mode == PROGRAM) {
412 g_program->add_enum($1);
413 }
414 }
Mark Slee6a47fed2007-02-07 02:40:59 +0000415| Senum
416 {
417 pdebug("TypeDefinition -> Senum");
418 if (g_parse_mode == PROGRAM) {
419 g_program->add_typedef($1);
420 }
421 }
Mark Sleef0712dc2006-10-25 19:03:57 +0000422| Struct
423 {
424 pdebug("TypeDefinition -> Struct");
425 if (g_parse_mode == PROGRAM) {
426 g_program->add_struct($1);
427 }
428 }
429| Xception
Mark Slee27ed6ec2007-08-16 01:26:31 +0000430 {
Mark Sleef0712dc2006-10-25 19:03:57 +0000431 pdebug("TypeDefinition -> Xception");
432 if (g_parse_mode == PROGRAM) {
433 g_program->add_xception($1);
434 }
Mark Slee9cb7c612006-09-01 22:17:45 +0000435 }
Mark Slee31985722006-05-24 21:45:31 +0000436
437Typedef:
Mark Sleebd588222007-11-21 08:43:35 +0000438 tok_typedef DefinitionType tok_identifier
Mark Slee31985722006-05-24 21:45:31 +0000439 {
440 pdebug("TypeDef -> tok_typedef DefinitionType tok_identifier");
David Reisscdffe262007-08-14 17:12:31 +0000441 t_typedef *td = new t_typedef(g_program, $2, $3);
Mark Slee31985722006-05-24 21:45:31 +0000442 $$ = td;
443 }
444
Mark Slee6a47fed2007-02-07 02:40:59 +0000445CommaOrSemicolonOptional:
446 ','
447 {}
448| ';'
449 {}
450|
451 {}
ccheeverf53b5cf2007-02-05 20:33:11 +0000452
Mark Slee31985722006-05-24 21:45:31 +0000453Enum:
David Reisscdffe262007-08-14 17:12:31 +0000454 tok_enum tok_identifier '{' EnumDefList '}'
Mark Slee31985722006-05-24 21:45:31 +0000455 {
456 pdebug("Enum -> tok_enum tok_identifier { EnumDefList }");
David Reisscdffe262007-08-14 17:12:31 +0000457 $$ = $4;
458 $$->set_name($2);
Mark Slee31985722006-05-24 21:45:31 +0000459 }
460
461EnumDefList:
Mark Slee207cb462006-11-02 18:43:12 +0000462 EnumDefList EnumDef
Mark Slee31985722006-05-24 21:45:31 +0000463 {
464 pdebug("EnumDefList -> EnumDefList EnumDef");
465 $$ = $1;
Mark Slee207cb462006-11-02 18:43:12 +0000466 $$->append($2);
Mark Slee31985722006-05-24 21:45:31 +0000467 }
468|
469 {
470 pdebug("EnumDefList -> ");
Mark Sleef0712dc2006-10-25 19:03:57 +0000471 $$ = new t_enum(g_program);
Mark Slee31985722006-05-24 21:45:31 +0000472 }
473
474EnumDef:
David Reisscbd4bac2007-08-14 17:12:33 +0000475 CaptureDocText tok_identifier '=' tok_int_constant CommaOrSemicolonOptional
Mark Slee31985722006-05-24 21:45:31 +0000476 {
Mark Slee30152872006-11-28 01:24:07 +0000477 pdebug("EnumDef -> tok_identifier = tok_int_constant");
ccheeverf53b5cf2007-02-05 20:33:11 +0000478 if ($4 < 0) {
479 pwarning(1, "Negative value supplied for enum %s.\n", $2);
Mark Slee31985722006-05-24 21:45:31 +0000480 }
ccheeverf53b5cf2007-02-05 20:33:11 +0000481 $$ = new t_enum_value($2, $4);
482 if ($1 != NULL) {
483 $$->set_doc($1);
484 }
Mark Sleed0767c52007-07-27 22:14:41 +0000485 if (g_parse_mode == PROGRAM) {
486 g_scope->add_constant($2, new t_const(g_type_i32, $2, new t_const_value($4)));
487 if (g_parent_scope != NULL) {
488 g_parent_scope->add_constant(g_parent_prefix + $2, new t_const(g_type_i32, $2, new t_const_value($4)));
489 }
490 }
Mark Slee31985722006-05-24 21:45:31 +0000491 }
492|
David Reisscbd4bac2007-08-14 17:12:33 +0000493 CaptureDocText tok_identifier CommaOrSemicolonOptional
Mark Slee31985722006-05-24 21:45:31 +0000494 {
Mark Slee30152872006-11-28 01:24:07 +0000495 pdebug("EnumDef -> tok_identifier");
ccheeverf53b5cf2007-02-05 20:33:11 +0000496 $$ = new t_enum_value($2);
497 if ($1 != NULL) {
498 $$->set_doc($1);
499 }
Mark Slee30152872006-11-28 01:24:07 +0000500 }
501
Mark Slee6a47fed2007-02-07 02:40:59 +0000502Senum:
David Reisscdffe262007-08-14 17:12:31 +0000503 tok_senum tok_identifier '{' SenumDefList '}'
Mark Slee6a47fed2007-02-07 02:40:59 +0000504 {
505 pdebug("Senum -> tok_senum tok_identifier { SenumDefList }");
David Reisscdffe262007-08-14 17:12:31 +0000506 $$ = new t_typedef(g_program, $4, $2);
Mark Slee6a47fed2007-02-07 02:40:59 +0000507 }
508
509SenumDefList:
510 SenumDefList SenumDef
511 {
512 pdebug("SenumDefList -> SenumDefList SenumDef");
513 $$ = $1;
514 $$->add_string_enum_val($2);
515 }
516|
517 {
518 pdebug("SenumDefList -> ");
519 $$ = new t_base_type("string", t_base_type::TYPE_STRING);
520 $$->set_string_enum(true);
521 }
522
523SenumDef:
524 tok_literal CommaOrSemicolonOptional
525 {
526 pdebug("SenumDef -> tok_literal");
527 $$ = $1;
528 }
529
Mark Slee30152872006-11-28 01:24:07 +0000530Const:
531 tok_const FieldType tok_identifier '=' ConstValue CommaOrSemicolonOptional
532 {
533 pdebug("Const -> tok_const FieldType tok_identifier = ConstValue");
Mark Sleeaa7671d2006-11-29 03:19:31 +0000534 if (g_parse_mode == PROGRAM) {
535 $$ = new t_const($2, $3, $5);
536 validate_const_type($$);
Mark Sleed0767c52007-07-27 22:14:41 +0000537
538 g_scope->add_constant($3, $$);
539 if (g_parent_scope != NULL) {
540 g_parent_scope->add_constant(g_parent_prefix + $3, $$);
541 }
542
Mark Sleeaa7671d2006-11-29 03:19:31 +0000543 } else {
544 $$ = NULL;
545 }
Mark Slee30152872006-11-28 01:24:07 +0000546 }
547
548ConstValue:
549 tok_int_constant
550 {
551 pdebug("ConstValue => tok_int_constant");
552 $$ = new t_const_value();
553 $$->set_integer($1);
554 }
555| tok_dub_constant
556 {
557 pdebug("ConstValue => tok_dub_constant");
558 $$ = new t_const_value();
559 $$->set_double($1);
560 }
561| tok_literal
562 {
563 pdebug("ConstValue => tok_literal");
Mark Sleed0767c52007-07-27 22:14:41 +0000564 $$ = new t_const_value($1);
Mark Slee30152872006-11-28 01:24:07 +0000565 }
Mark Slee67fc6342006-11-29 03:37:04 +0000566| tok_identifier
567 {
568 pdebug("ConstValue => tok_identifier");
Mark Sleed0767c52007-07-27 22:14:41 +0000569 t_const* constant = g_scope->get_constant($1);
570 if (constant != NULL) {
571 $$ = constant->get_value();
572 } else {
573 if (g_parse_mode == PROGRAM) {
574 pwarning(1, "Constant strings should be quoted: %s\n", $1);
575 }
576 $$ = new t_const_value($1);
577 }
Mark Slee67fc6342006-11-29 03:37:04 +0000578 }
Mark Slee30152872006-11-28 01:24:07 +0000579| ConstList
580 {
581 pdebug("ConstValue => ConstList");
582 $$ = $1;
583 }
584| ConstMap
585 {
586 pdebug("ConstValue => ConstMap");
Mark Slee27ed6ec2007-08-16 01:26:31 +0000587 $$ = $1;
Mark Slee30152872006-11-28 01:24:07 +0000588 }
589
590ConstList:
591 '[' ConstListContents ']'
592 {
593 pdebug("ConstList => [ ConstListContents ]");
594 $$ = $2;
595 }
596
597ConstListContents:
598 ConstListContents ConstValue CommaOrSemicolonOptional
599 {
600 pdebug("ConstListContents => ConstListContents ConstValue CommaOrSemicolonOptional");
601 $$ = $1;
602 $$->add_list($2);
603 }
604|
605 {
606 pdebug("ConstListContents =>");
607 $$ = new t_const_value();
608 $$->set_list();
609 }
610
611ConstMap:
612 '{' ConstMapContents '}'
613 {
614 pdebug("ConstMap => { ConstMapContents }");
615 $$ = $2;
616 }
617
618ConstMapContents:
619 ConstMapContents ConstValue ':' ConstValue CommaOrSemicolonOptional
620 {
621 pdebug("ConstMapContents => ConstMapContents ConstValue CommaOrSemicolonOptional");
622 $$ = $1;
623 $$->add_map($2, $4);
624 }
625|
626 {
627 pdebug("ConstMapContents =>");
628 $$ = new t_const_value();
629 $$->set_map();
Mark Slee31985722006-05-24 21:45:31 +0000630 }
631
632Struct:
David Reisscdffe262007-08-14 17:12:31 +0000633 tok_struct tok_identifier XsdAll '{' FieldList '}'
Mark Slee31985722006-05-24 21:45:31 +0000634 {
635 pdebug("Struct -> tok_struct tok_identifier { FieldList }");
David Reisscdffe262007-08-14 17:12:31 +0000636 $5->set_name($2);
637 $5->set_xsd_all($3);
638 $$ = $5;
Mark Slee9cb7c612006-09-01 22:17:45 +0000639 y_field_val = -1;
640 }
641
Mark Slee36bfa2e2007-01-19 20:09:51 +0000642XsdAll:
Mark Slee782abbb2007-01-19 00:17:02 +0000643 tok_xsd_all
644 {
645 $$ = true;
646 }
647|
648 {
649 $$ = false;
650 }
651
Mark Slee36bfa2e2007-01-19 20:09:51 +0000652XsdOptional:
653 tok_xsd_optional
654 {
655 $$ = true;
656 }
657|
658 {
659 $$ = false;
660 }
661
Mark Slee7df0e2a2007-02-06 21:03:18 +0000662XsdNillable:
663 tok_xsd_nillable
664 {
665 $$ = true;
666 }
667|
668 {
669 $$ = false;
670 }
671
Mark Slee21135c32007-02-05 21:52:08 +0000672XsdAttributes:
Mark Slee748d83f2007-02-07 01:20:08 +0000673 tok_xsd_attrs '{' FieldList '}'
Mark Slee21135c32007-02-05 21:52:08 +0000674 {
Mark Slee748d83f2007-02-07 01:20:08 +0000675 $$ = $3;
Mark Slee21135c32007-02-05 21:52:08 +0000676 }
677|
678 {
679 $$ = NULL;
680 }
681
Mark Slee9cb7c612006-09-01 22:17:45 +0000682Xception:
683 tok_xception tok_identifier '{' FieldList '}'
684 {
685 pdebug("Xception -> tok_xception tok_identifier { FieldList }");
686 $4->set_name($2);
687 $4->set_xception(true);
688 $$ = $4;
689 y_field_val = -1;
Mark Slee31985722006-05-24 21:45:31 +0000690 }
691
692Service:
Mark Slee78165722007-09-10 22:08:49 +0000693 tok_service tok_identifier Extends '{' FlagArgs FunctionList UnflagArgs '}'
Mark Slee31985722006-05-24 21:45:31 +0000694 {
695 pdebug("Service -> tok_service tok_identifier { FunctionList }");
Mark Slee78165722007-09-10 22:08:49 +0000696 $$ = $6;
David Reisscdffe262007-08-14 17:12:31 +0000697 $$->set_name($2);
698 $$->set_extends($3);
Mark Sleef0712dc2006-10-25 19:03:57 +0000699 }
700
Mark Slee78165722007-09-10 22:08:49 +0000701FlagArgs:
702 {
703 g_arglist = 1;
704 }
705
706UnflagArgs:
707 {
708 g_arglist = 0;
709 }
710
Mark Slee36bfa2e2007-01-19 20:09:51 +0000711Extends:
Mark Sleef0712dc2006-10-25 19:03:57 +0000712 tok_extends tok_identifier
713 {
Mark Slee36bfa2e2007-01-19 20:09:51 +0000714 pdebug("Extends -> tok_extends tok_identifier");
Mark Sleef0712dc2006-10-25 19:03:57 +0000715 $$ = NULL;
716 if (g_parse_mode == PROGRAM) {
717 $$ = g_scope->get_service($2);
718 if ($$ == NULL) {
719 yyerror("Service \"%s\" has not been defined.", $2);
720 exit(1);
721 }
722 }
723 }
724|
725 {
726 $$ = NULL;
Mark Slee31985722006-05-24 21:45:31 +0000727 }
728
729FunctionList:
Mark Slee207cb462006-11-02 18:43:12 +0000730 FunctionList Function
Mark Slee31985722006-05-24 21:45:31 +0000731 {
732 pdebug("FunctionList -> FunctionList Function");
733 $$ = $1;
734 $1->add_function($2);
735 }
736|
737 {
738 pdebug("FunctionList -> ");
Mark Sleef0712dc2006-10-25 19:03:57 +0000739 $$ = new t_service(g_program);
Mark Slee31985722006-05-24 21:45:31 +0000740 }
741
742Function:
David Reisscbd4bac2007-08-14 17:12:33 +0000743 CaptureDocText Async FunctionType tok_identifier '(' FieldList ')' Throws CommaOrSemicolonOptional
Mark Slee31985722006-05-24 21:45:31 +0000744 {
ccheeverf53b5cf2007-02-05 20:33:11 +0000745 $6->set_name(std::string($4) + "_args");
746 $$ = new t_function($3, $4, $6, $8, $2);
747 if ($1 != NULL) {
748 $$->set_doc($1);
749 }
Mark Slee9cb7c612006-09-01 22:17:45 +0000750 y_field_val = -1;
Mark Slee31985722006-05-24 21:45:31 +0000751 }
752
Mark Slee36bfa2e2007-01-19 20:09:51 +0000753Async:
Mark Slee52f643d2006-08-09 00:03:43 +0000754 tok_async
Mark Slee31985722006-05-24 21:45:31 +0000755 {
Mark Slee52f643d2006-08-09 00:03:43 +0000756 $$ = true;
757 }
758|
759 {
760 $$ = false;
Mark Slee31985722006-05-24 21:45:31 +0000761 }
762
Mark Slee36bfa2e2007-01-19 20:09:51 +0000763Throws:
Mark Slee9cb7c612006-09-01 22:17:45 +0000764 tok_throws '(' FieldList ')'
765 {
Mark Slee36bfa2e2007-01-19 20:09:51 +0000766 pdebug("Throws -> tok_throws ( FieldList )");
Mark Slee9cb7c612006-09-01 22:17:45 +0000767 $$ = $3;
Mark Sleef07d48e2008-02-01 01:36:26 +0000768 if (g_parse_mode == PROGRAM && !validate_throws($$)) {
Mark Slee91f2b7b2008-01-31 01:49:16 +0000769 yyerror("Throws clause may not contain non-exception types");
770 exit(1);
771 }
Mark Slee9cb7c612006-09-01 22:17:45 +0000772 }
773|
774 {
Mark Sleef0712dc2006-10-25 19:03:57 +0000775 $$ = new t_struct(g_program);
Mark Slee9cb7c612006-09-01 22:17:45 +0000776 }
777
Mark Slee31985722006-05-24 21:45:31 +0000778FieldList:
Mark Slee207cb462006-11-02 18:43:12 +0000779 FieldList Field
Mark Slee31985722006-05-24 21:45:31 +0000780 {
781 pdebug("FieldList -> FieldList , Field");
782 $$ = $1;
Mark Slee6f9ac3f2007-11-28 22:28:13 +0000783 if (!($$->validate_field($2))) {
784 yyerror("Field identifier %d for \"%s\" has already been used", $2->get_key(), $2->get_name().c_str());
785 exit(1);
786 }
Mark Slee207cb462006-11-02 18:43:12 +0000787 $$->append($2);
Mark Slee31985722006-05-24 21:45:31 +0000788 }
789|
790 {
791 pdebug("FieldList -> ");
Mark Sleef0712dc2006-10-25 19:03:57 +0000792 $$ = new t_struct(g_program);
Mark Slee31985722006-05-24 21:45:31 +0000793 }
794
795Field:
David Reiss8320a922007-08-14 19:59:26 +0000796 CaptureDocText FieldIdentifier FieldRequiredness FieldType tok_identifier FieldValue XsdOptional XsdNillable XsdAttributes CommaOrSemicolonOptional
Mark Slee31985722006-05-24 21:45:31 +0000797 {
Mark Sleef0712dc2006-10-25 19:03:57 +0000798 pdebug("tok_int_constant : Field -> FieldType tok_identifier");
ccheeverf53b5cf2007-02-05 20:33:11 +0000799 if ($2 < 0) {
David Reiss8320a922007-08-14 19:59:26 +0000800 pwarning(2, "No field key specified for %s, resulting protocol may have conflicts or not be backwards compatible!\n", $5);
Mark Slee31985722006-05-24 21:45:31 +0000801 }
David Reiss8320a922007-08-14 19:59:26 +0000802 $$ = new t_field($4, $5, $2);
803 $$->set_req($3);
804 if ($6 != NULL) {
805 validate_field_value($$, $6);
806 $$->set_value($6);
Mark Slee7ff32452007-02-01 05:26:18 +0000807 }
David Reiss8320a922007-08-14 19:59:26 +0000808 $$->set_xsd_optional($7);
809 $$->set_xsd_nillable($8);
ccheeverf53b5cf2007-02-05 20:33:11 +0000810 if ($1 != NULL) {
811 $$->set_doc($1);
812 }
David Reiss8320a922007-08-14 19:59:26 +0000813 if ($9 != NULL) {
814 $$->set_xsd_attrs($9);
Mark Slee21135c32007-02-05 21:52:08 +0000815 }
Mark Slee31985722006-05-24 21:45:31 +0000816 }
Mark Slee7ff32452007-02-01 05:26:18 +0000817
818FieldIdentifier:
819 tok_int_constant ':'
Mark Slee31985722006-05-24 21:45:31 +0000820 {
Mark Slee7ff32452007-02-01 05:26:18 +0000821 if ($1 <= 0) {
822 pwarning(1, "Nonpositive value (%d) not allowed as a field key.\n", $1);
823 $1 = y_field_val--;
Mark Sleef0712dc2006-10-25 19:03:57 +0000824 }
Mark Slee7ff32452007-02-01 05:26:18 +0000825 $$ = $1;
826 }
827|
828 {
829 $$ = y_field_val--;
830 }
831
David Reiss8320a922007-08-14 19:59:26 +0000832FieldRequiredness:
833 tok_required
834 {
Mark Slee78165722007-09-10 22:08:49 +0000835 if (g_arglist) {
836 if (g_parse_mode == PROGRAM) {
837 pwarning(1, "required keyword is ignored in argument lists.\n");
838 }
David Reiss204420f2008-01-11 20:59:03 +0000839 $$ = t_field::T_OPT_IN_REQ_OUT;
Mark Slee78165722007-09-10 22:08:49 +0000840 } else {
David Reiss204420f2008-01-11 20:59:03 +0000841 $$ = t_field::T_REQUIRED;
Mark Slee78165722007-09-10 22:08:49 +0000842 }
David Reiss8320a922007-08-14 19:59:26 +0000843 }
844| tok_optional
845 {
Mark Slee78165722007-09-10 22:08:49 +0000846 if (g_arglist) {
847 if (g_parse_mode == PROGRAM) {
848 pwarning(1, "optional keyword is ignored in argument lists.\n");
849 }
David Reiss204420f2008-01-11 20:59:03 +0000850 $$ = t_field::T_OPT_IN_REQ_OUT;
Mark Slee78165722007-09-10 22:08:49 +0000851 } else {
David Reiss204420f2008-01-11 20:59:03 +0000852 $$ = t_field::T_OPTIONAL;
Mark Slee78165722007-09-10 22:08:49 +0000853 }
David Reiss8320a922007-08-14 19:59:26 +0000854 }
855|
856 {
David Reiss204420f2008-01-11 20:59:03 +0000857 $$ = t_field::T_OPT_IN_REQ_OUT;
David Reiss8320a922007-08-14 19:59:26 +0000858 }
859
Mark Slee7ff32452007-02-01 05:26:18 +0000860FieldValue:
861 '=' ConstValue
862 {
Mark Slee27ed6ec2007-08-16 01:26:31 +0000863 if (g_parse_mode == PROGRAM) {
Mark Slee7ff32452007-02-01 05:26:18 +0000864 $$ = $2;
865 } else {
866 $$ = NULL;
867 }
868 }
869|
870 {
871 $$ = NULL;
Mark Sleef0712dc2006-10-25 19:03:57 +0000872 }
Mark Slee31985722006-05-24 21:45:31 +0000873
874DefinitionType:
875 BaseType
876 {
877 pdebug("DefinitionType -> BaseType");
878 $$ = $1;
879 }
Mark Sleee8540632006-05-30 09:24:40 +0000880| ContainerType
881 {
882 pdebug("DefinitionType -> ContainerType");
883 $$ = $1;
884 }
Mark Slee31985722006-05-24 21:45:31 +0000885
886FunctionType:
887 FieldType
888 {
Mark Sleee8540632006-05-30 09:24:40 +0000889 pdebug("FunctionType -> FieldType");
Mark Slee31985722006-05-24 21:45:31 +0000890 $$ = $1;
891 }
892| tok_void
893 {
Mark Sleee8540632006-05-30 09:24:40 +0000894 pdebug("FunctionType -> tok_void");
Mark Sleef0712dc2006-10-25 19:03:57 +0000895 $$ = g_type_void;
Mark Slee31985722006-05-24 21:45:31 +0000896 }
897
898FieldType:
899 tok_identifier
900 {
Mark Sleee8540632006-05-30 09:24:40 +0000901 pdebug("FieldType -> tok_identifier");
Mark Sleef0712dc2006-10-25 19:03:57 +0000902 if (g_parse_mode == INCLUDES) {
903 // Ignore identifiers in include mode
904 $$ = NULL;
905 } else {
906 // Lookup the identifier in the current scope
907 $$ = g_scope->get_type($1);
908 if ($$ == NULL) {
909 yyerror("Type \"%s\" has not been defined.", $1);
910 exit(1);
911 }
Mark Sleee8540632006-05-30 09:24:40 +0000912 }
Mark Slee31985722006-05-24 21:45:31 +0000913 }
914| BaseType
915 {
Mark Sleee8540632006-05-30 09:24:40 +0000916 pdebug("FieldType -> BaseType");
917 $$ = $1;
918 }
919| ContainerType
920 {
921 pdebug("FieldType -> ContainerType");
Mark Slee31985722006-05-24 21:45:31 +0000922 $$ = $1;
923 }
924
925BaseType:
926 tok_string
927 {
928 pdebug("BaseType -> tok_string");
Mark Sleef0712dc2006-10-25 19:03:57 +0000929 $$ = g_type_string;
Mark Slee31985722006-05-24 21:45:31 +0000930 }
Mark Slee8d725a22007-04-13 01:57:12 +0000931| tok_binary
932 {
933 pdebug("BaseType -> tok_binary");
934 $$ = g_type_binary;
935 }
Mark Sleeb6200d82007-01-19 19:14:36 +0000936| tok_slist
937 {
938 pdebug("BaseType -> tok_slist");
939 $$ = g_type_slist;
940 }
Mark Slee78f58e22006-09-02 04:17:07 +0000941| tok_bool
942 {
943 pdebug("BaseType -> tok_bool");
Mark Sleef0712dc2006-10-25 19:03:57 +0000944 $$ = g_type_bool;
Mark Slee78f58e22006-09-02 04:17:07 +0000945 }
Mark Slee31985722006-05-24 21:45:31 +0000946| tok_byte
947 {
948 pdebug("BaseType -> tok_byte");
Mark Sleef0712dc2006-10-25 19:03:57 +0000949 $$ = g_type_byte;
Mark Slee31985722006-05-24 21:45:31 +0000950 }
Mark Slee9cb7c612006-09-01 22:17:45 +0000951| tok_i16
952 {
953 pdebug("BaseType -> tok_i16");
Mark Sleef0712dc2006-10-25 19:03:57 +0000954 $$ = g_type_i16;
Mark Slee9cb7c612006-09-01 22:17:45 +0000955 }
Mark Slee31985722006-05-24 21:45:31 +0000956| tok_i32
957 {
958 pdebug("BaseType -> tok_i32");
Mark Sleef0712dc2006-10-25 19:03:57 +0000959 $$ = g_type_i32;
Mark Slee31985722006-05-24 21:45:31 +0000960 }
Mark Slee31985722006-05-24 21:45:31 +0000961| tok_i64
962 {
963 pdebug("BaseType -> tok_i64");
Mark Sleef0712dc2006-10-25 19:03:57 +0000964 $$ = g_type_i64;
Mark Slee31985722006-05-24 21:45:31 +0000965 }
Mark Sleec98d0502006-09-06 02:42:25 +0000966| tok_double
967 {
968 pdebug("BaseType -> tok_double");
Mark Sleef0712dc2006-10-25 19:03:57 +0000969 $$ = g_type_double;
Mark Sleec98d0502006-09-06 02:42:25 +0000970 }
Mark Slee31985722006-05-24 21:45:31 +0000971
Mark Sleee8540632006-05-30 09:24:40 +0000972ContainerType:
973 MapType
974 {
975 pdebug("ContainerType -> MapType");
976 $$ = $1;
977 }
978| SetType
979 {
980 pdebug("ContainerType -> SetType");
981 $$ = $1;
982 }
983| ListType
984 {
985 pdebug("ContainerType -> ListType");
986 $$ = $1;
987 }
988
989MapType:
Mark Slee36bfa2e2007-01-19 20:09:51 +0000990 tok_map CppType '<' FieldType ',' FieldType '>'
Mark Sleee8540632006-05-30 09:24:40 +0000991 {
992 pdebug("MapType -> tok_map <FieldType, FieldType>");
Mark Slee4f8da1d2006-10-12 02:47:27 +0000993 $$ = new t_map($4, $6);
994 if ($2 != NULL) {
995 ((t_container*)$$)->set_cpp_name(std::string($2));
996 }
Mark Sleee8540632006-05-30 09:24:40 +0000997 }
998
999SetType:
Mark Slee36bfa2e2007-01-19 20:09:51 +00001000 tok_set CppType '<' FieldType '>'
Mark Sleee8540632006-05-30 09:24:40 +00001001 {
1002 pdebug("SetType -> tok_set<FieldType>");
Mark Slee4f8da1d2006-10-12 02:47:27 +00001003 $$ = new t_set($4);
1004 if ($2 != NULL) {
1005 ((t_container*)$$)->set_cpp_name(std::string($2));
1006 }
Mark Sleee8540632006-05-30 09:24:40 +00001007 }
1008
1009ListType:
Mark Slee36bfa2e2007-01-19 20:09:51 +00001010 tok_list '<' FieldType '>' CppType
Mark Sleee8540632006-05-30 09:24:40 +00001011 {
1012 pdebug("ListType -> tok_list<FieldType>");
Mark Sleef0712dc2006-10-25 19:03:57 +00001013 $$ = new t_list($3);
1014 if ($5 != NULL) {
1015 ((t_container*)$$)->set_cpp_name(std::string($5));
Mark Slee4f8da1d2006-10-12 02:47:27 +00001016 }
1017 }
1018
Mark Slee36bfa2e2007-01-19 20:09:51 +00001019CppType:
Mark Sleeafc76542007-02-09 21:55:44 +00001020 tok_cpp_type tok_literal
Mark Slee4f8da1d2006-10-12 02:47:27 +00001021 {
Mark Sleeafc76542007-02-09 21:55:44 +00001022 $$ = $2;
Mark Slee4f8da1d2006-10-12 02:47:27 +00001023 }
1024|
1025 {
1026 $$ = NULL;
Mark Sleee8540632006-05-30 09:24:40 +00001027 }
1028
Mark Slee31985722006-05-24 21:45:31 +00001029%%