blob: a582436a767e0f1dc1e470111adb6e8cbb090a9b [file] [log] [blame]
Mark Slee31985722006-05-24 21:45:31 +00001/**
2 * thrift - a lightweight cross-language rpc/serialization tool
3 *
4 * This file contains the main compiler engine for Thrift, which invokes the
5 * scanner/parser to build the thrift object tree. The interface generation
Mark Sleef5377b32006-10-10 01:42:59 +00006 * code for each language lives in a file by the language name under the
7 * generate/ folder, and all parse structures live in parse/
Mark Slee31985722006-05-24 21:45:31 +00008 *
9 * @author Mark Slee <mcslee@facebook.com>
10 */
11
12#include <stdlib.h>
13#include <stdio.h>
14#include <stdarg.h>
15#include <string>
Mark Sleef0712dc2006-10-25 19:03:57 +000016#include <sys/types.h>
17#include <sys/stat.h>
Mark Slee31985722006-05-24 21:45:31 +000018
Mark Sleef0712dc2006-10-25 19:03:57 +000019// Careful: must include globals first for extern definitions
Mark Slee31985722006-05-24 21:45:31 +000020#include "globals.h"
21
22#include "main.h"
23#include "parse/t_program.h"
Mark Sleef0712dc2006-10-25 19:03:57 +000024#include "parse/t_scope.h"
Mark Slee31985722006-05-24 21:45:31 +000025#include "generate/t_cpp_generator.h"
Mark Sleeb15a68b2006-06-07 06:46:24 +000026#include "generate/t_java_generator.h"
Mark Slee6e536442006-06-30 18:28:50 +000027#include "generate/t_php_generator.h"
Mark Sleefc89d392006-09-04 00:04:39 +000028#include "generate/t_py_generator.h"
Mark Slee6d7d5952007-01-27 01:44:22 +000029#include "generate/t_rb_generator.h"
Mark Slee0e0ff7e2007-01-18 22:59:59 +000030#include "generate/t_xsd_generator.h"
Mark Slee31985722006-05-24 21:45:31 +000031
32using namespace std;
33
Mark Sleef5377b32006-10-10 01:42:59 +000034/**
35 * Global program tree
36 */
Mark Slee31985722006-05-24 21:45:31 +000037t_program* g_program;
38
Mark Sleef5377b32006-10-10 01:42:59 +000039/**
Mark Sleef0712dc2006-10-25 19:03:57 +000040 * Global types
41 */
42
43t_type* g_type_void;
44t_type* g_type_string;
Mark Sleeb6200d82007-01-19 19:14:36 +000045t_type* g_type_slist;
Mark Sleef0712dc2006-10-25 19:03:57 +000046t_type* g_type_bool;
47t_type* g_type_byte;
48t_type* g_type_i16;
49t_type* g_type_i32;
50t_type* g_type_i64;
51t_type* g_type_double;
52
53/**
54 * Global scope
55 */
56t_scope* g_scope;
57
58/**
59 * Parent scope to also parse types
60 */
61t_scope* g_parent_scope;
62
63/**
64 * Prefix for putting types in parent scope
65 */
66string g_parent_prefix;
67
68/**
69 * Parsing pass
70 */
71PARSE_MODE g_parse_mode;
72
73/**
74 * Current directory of file being parsed
75 */
76string g_curdir;
77
78/**
79 * Current file being parsed
80 */
81string g_curpath;
82
83/**
Martin Kraemer32c66e12006-11-09 00:06:36 +000084 * Search path for inclusions
85 */
Mark Slee2329a832006-11-09 00:23:30 +000086vector<string> g_incl_searchpath;
Martin Kraemer32c66e12006-11-09 00:06:36 +000087
88/**
Mark Sleef5377b32006-10-10 01:42:59 +000089 * Global debug state
90 */
Mark Slee31985722006-05-24 21:45:31 +000091int g_debug = 0;
92
Mark Sleef5377b32006-10-10 01:42:59 +000093/**
Mark Sleef0712dc2006-10-25 19:03:57 +000094 * Warning level
95 */
96int g_warn = 1;
97
98/**
99 * Verbose output
100 */
101int g_verbose = 0;
102
103/**
Mark Sleef5377b32006-10-10 01:42:59 +0000104 * Global time string
105 */
Mark Slee31985722006-05-24 21:45:31 +0000106char* g_time_str;
107
Mark Slee31985722006-05-24 21:45:31 +0000108/**
Mark Sleef0712dc2006-10-25 19:03:57 +0000109 * Flags to control code generation
110 */
111bool gen_cpp = false;
112bool gen_java = false;
Mark Slee6d7d5952007-01-27 01:44:22 +0000113bool gen_rb = false;
Mark Sleef0712dc2006-10-25 19:03:57 +0000114bool gen_py = false;
Mark Slee0e0ff7e2007-01-18 22:59:59 +0000115bool gen_xsd = false;
Mark Sleef0712dc2006-10-25 19:03:57 +0000116bool gen_php = false;
117bool gen_phpi = false;
118bool gen_recurse = false;
119
120/**
Mark Slee31985722006-05-24 21:45:31 +0000121 * Report an error to the user. This is called yyerror for historical
122 * reasons (lex and yacc expect the error reporting routine to be called
123 * this). Call this function to report any errors to the user.
124 * yyerror takes printf style arguments.
125 *
126 * @param fmt C format string followed by additional arguments
127 */
128void yyerror(char* fmt, ...) {
129 va_list args;
130 fprintf(stderr,
Mark Sleef0712dc2006-10-25 19:03:57 +0000131 "[ERROR:%s:%d] (last token was '%s')\n",
132 g_curpath.c_str(),
Mark Slee31985722006-05-24 21:45:31 +0000133 yylineno,
134 yytext);
Mark Slee31985722006-05-24 21:45:31 +0000135
136 va_start(args, fmt);
137 vfprintf(stderr, fmt, args);
138 va_end(args);
139
140 fprintf(stderr, "\n");
141}
142
143/**
144 * Prints a debug message from the parser.
145 *
146 * @param fmt C format string followed by additional arguments
147 */
148void pdebug(char* fmt, ...) {
149 if (g_debug == 0) {
150 return;
151 }
152 va_list args;
Mark Slee30152872006-11-28 01:24:07 +0000153 printf("[PARSE:%d] ", yylineno);
Mark Sleef0712dc2006-10-25 19:03:57 +0000154 va_start(args, fmt);
155 vprintf(fmt, args);
156 va_end(args);
157 printf("\n");
158}
159
160/**
161 * Prints a verbose output mode message
162 *
163 * @param fmt C format string followed by additional arguments
164 */
165void pverbose(char* fmt, ...) {
166 if (g_verbose == 0) {
167 return;
168 }
169 va_list args;
170 va_start(args, fmt);
171 vprintf(fmt, args);
172 va_end(args);
173}
174
175/**
176 * Prints a warning message
177 *
178 * @param fmt C format string followed by additional arguments
179 */
180void pwarning(int level, char* fmt, ...) {
181 if (g_warn < level) {
182 return;
183 }
184 va_list args;
185 printf("[WARNING:%s:%d] ", g_curpath.c_str(), yylineno);
Mark Slee31985722006-05-24 21:45:31 +0000186 va_start(args, fmt);
187 vprintf(fmt, args);
188 va_end(args);
189 printf("\n");
190}
191
192/**
193 * Prints a failure message and exits
194 *
195 * @param fmt C format string followed by additional arguments
196 */
Mark Slee30152872006-11-28 01:24:07 +0000197void failure(const char* fmt, ...) {
Mark Slee31985722006-05-24 21:45:31 +0000198 va_list args;
Mark Sleef0712dc2006-10-25 19:03:57 +0000199 fprintf(stderr, "[FAILURE:%s:%d] ", g_curpath.c_str(), yylineno);
Mark Slee31985722006-05-24 21:45:31 +0000200 va_start(args, fmt);
201 vfprintf(stderr, fmt, args);
202 va_end(args);
203 printf("\n");
204 exit(1);
205}
206
207/**
Mark Sleef0712dc2006-10-25 19:03:57 +0000208 * Converts a string filename into a thrift program name
209 */
210string program_name(string filename) {
211 string::size_type slash = filename.rfind("/");
212 if (slash != string::npos) {
213 filename = filename.substr(slash+1);
214 }
215 string::size_type dot = filename.rfind(".");
216 if (dot != string::npos) {
217 filename = filename.substr(0, dot);
218 }
219 return filename;
220}
221
222/**
223 * Gets the directory path of a filename
224 */
225string directory_name(string filename) {
226 string::size_type slash = filename.rfind("/");
227 // No slash, just use the current directory
228 if (slash == string::npos) {
229 return ".";
230 }
231 return filename.substr(0, slash);
232}
233
234/**
235 * Finds the appropriate file path for the given filename
236 */
237string include_file(string filename) {
238 // Absolute path? Just try that
Martin Kraemer32c66e12006-11-09 00:06:36 +0000239 if (filename[0] == '/') {
240 // Realpath!
241 char rp[PATH_MAX];
242 if (realpath(filename.c_str(), rp) == NULL) {
243 pwarning(0, "Cannot open include file %s\n", filename.c_str());
244 return std::string();
245 }
246
247 // Stat this files
248 struct stat finfo;
249 if (stat(rp, &finfo) == 0) {
250 return rp;
251 }
252 } else { // relative path, start searching
253 // new search path with current dir global
254 vector<string> sp = g_incl_searchpath;
255 sp.insert(sp.begin(), g_curdir);
256
257 // iterate through paths
258 vector<string>::iterator it;
259 for (it = sp.begin(); it != sp.end(); it++) {
260 string sfilename = *(it) + "/" + filename;
261
262 // Realpath!
263 char rp[PATH_MAX];
264 if (realpath(sfilename.c_str(), rp) == NULL) {
265 continue;
266 }
267
268 // Stat this files
269 struct stat finfo;
270 if (stat(rp, &finfo) == 0) {
271 return rp;
272 }
273 }
Mark Sleef0712dc2006-10-25 19:03:57 +0000274 }
275
Mark Sleef0712dc2006-10-25 19:03:57 +0000276 // Uh oh
277 pwarning(0, "Could not find include file %s\n", filename.c_str());
278 return std::string();
279}
280
281/**
Mark Slee31985722006-05-24 21:45:31 +0000282 * Diplays the usage message and then exits with an error code.
283 */
284void usage() {
Mark Sleeb15a68b2006-06-07 06:46:24 +0000285 fprintf(stderr, "Usage: thrift [options] file\n");
286 fprintf(stderr, "Options:\n");
Mark Slee2329a832006-11-09 00:23:30 +0000287 fprintf(stderr, " -cpp Generate C++ output files\n");
288 fprintf(stderr, " -java Generate Java output files\n");
289 fprintf(stderr, " -php Generate PHP output files\n");
290 fprintf(stderr, " -phpi Generate PHP inlined files\n");
291 fprintf(stderr, " -py Generate Python output files\n");
Martin Kraemer32c66e12006-11-09 00:06:36 +0000292 fprintf(stderr, " -I dir Add a directory to the list of directories \n");
293 fprintf(stderr, " searched for include directives\n");
Mark Slee2329a832006-11-09 00:23:30 +0000294 fprintf(stderr, " -nowarn Suppress all compiler warnings (BAD!)\n");
295 fprintf(stderr, " -strict Strict compiler warnings on\n");
296 fprintf(stderr, " -v[erbose] Verbose mode\n");
297 fprintf(stderr, " -r[ecurse] Also generate included files\n");
298 fprintf(stderr, " -debug Parse debug trace to stdout\n");
Mark Slee31985722006-05-24 21:45:31 +0000299 exit(1);
300}
301
302/**
Mark Slee30152872006-11-28 01:24:07 +0000303 * You know, when I started working on Thrift I really thought it wasn't going
304 * to become a programming language because it was just a generator and it
305 * wouldn't need runtime type information and all that jazz. But then we
306 * decided to add constants, and all of a sudden that means runtime type
307 * validation and inference, except the "runtime" is the code generator
308 * runtime. Shit. I've been had.
309 */
310void validate_const_rec(std::string name, t_type* type, t_const_value* value) {
311 if (type->is_void()) {
312 throw "type error: cannot declare a void const: " + name;
313 }
314
315 if (type->is_base_type()) {
316 t_base_type::t_base tbase = ((t_base_type*)type)->get_base();
317 switch (tbase) {
318 case t_base_type::TYPE_STRING:
319 if (value->get_type() != t_const_value::CV_STRING) {
320 throw "type error: const \"" + name + "\" was declared as string";
321 }
322 break;
323 case t_base_type::TYPE_BOOL:
324 if (value->get_type() != t_const_value::CV_INTEGER) {
325 throw "type error: const \"" + name + "\" was declared as bool";
326 }
327 break;
328 case t_base_type::TYPE_BYTE:
329 if (value->get_type() != t_const_value::CV_INTEGER) {
330 throw "type error: const \"" + name + "\" was declared as byte";
331 }
332 break;
333 case t_base_type::TYPE_I16:
334 if (value->get_type() != t_const_value::CV_INTEGER) {
335 throw "type error: const \"" + name + "\" was declared as i16";
336 }
337 break;
338 case t_base_type::TYPE_I32:
339 if (value->get_type() != t_const_value::CV_INTEGER) {
340 throw "type error: const \"" + name + "\" was declared as i32";
341 }
342 break;
343 case t_base_type::TYPE_I64:
344 if (value->get_type() != t_const_value::CV_INTEGER) {
345 throw "type error: const \"" + name + "\" was declared as i64";
346 }
347 break;
348 case t_base_type::TYPE_DOUBLE:
349 if (value->get_type() != t_const_value::CV_INTEGER &&
350 value->get_type() != t_const_value::CV_DOUBLE) {
351 throw "type error: const \"" + name + "\" was declared as double";
352 }
353 break;
354 default:
355 throw "compiler error: no const of base type " + tbase + name;
356 }
357 } else if (type->is_enum()) {
358 if (value->get_type() != t_const_value::CV_INTEGER) {
359 throw "type error: const \"" + name + "\" was declared as enum";
360 }
361 } else if (type->is_struct() || type->is_xception()) {
362 if (value->get_type() != t_const_value::CV_MAP) {
363 throw "type error: const \"" + name + "\" was declared as struct/xception";
364 }
365 const vector<t_field*>& fields = ((t_struct*)type)->get_members();
366 vector<t_field*>::const_iterator f_iter;
367
368 const map<t_const_value*, t_const_value*>& val = value->get_map();
369 map<t_const_value*, t_const_value*>::const_iterator v_iter;
370 for (v_iter = val.begin(); v_iter != val.end(); ++v_iter) {
371 if (v_iter->first->get_type() != t_const_value::CV_STRING) {
372 throw "type error: " + name + " struct key must be string";
373 }
374 t_type* field_type = NULL;
375 for (f_iter = fields.begin(); f_iter != fields.end(); ++f_iter) {
376 if ((*f_iter)->get_name() == v_iter->first->get_string()) {
377 field_type = (*f_iter)->get_type();
378 }
379 }
380 if (field_type == NULL) {
381 throw "type error: " + type->get_name() + " has no field " + v_iter->first->get_string();
382 }
383
384 validate_const_rec(name + "." + v_iter->first->get_string(), field_type, v_iter->second);
385 }
386 } else if (type->is_map()) {
387 t_type* k_type = ((t_map*)type)->get_key_type();
388 t_type* v_type = ((t_map*)type)->get_val_type();
389 const map<t_const_value*, t_const_value*>& val = value->get_map();
390 map<t_const_value*, t_const_value*>::const_iterator v_iter;
391 for (v_iter = val.begin(); v_iter != val.end(); ++v_iter) {
392 validate_const_rec(name + "<key>", k_type, v_iter->first);
393 validate_const_rec(name + "<val>", v_type, v_iter->second);
394 }
395 } else if (type->is_list() || type->is_set()) {
396 t_type* e_type;
397 if (type->is_list()) {
398 e_type = ((t_list*)type)->get_elem_type();
399 } else {
400 e_type = ((t_set*)type)->get_elem_type();
401 }
402 const vector<t_const_value*>& val = value->get_list();
403 vector<t_const_value*>::const_iterator v_iter;
404 for (v_iter = val.begin(); v_iter != val.end(); ++v_iter) {
405 validate_const_rec(name + "<elem>", e_type, *v_iter);
406 }
407 }
408}
409
410/**
411 * Check the type of the parsed const information against its declared type
412 */
413void validate_const_type(t_const* c) {
414 validate_const_rec(c->get_name(), c->get_type(), c->get_value());
415}
416
417/**
Mark Slee7ff32452007-02-01 05:26:18 +0000418 * Check the type of a default value assigned to a field.
419 */
420void validate_field_value(t_field* field, t_const_value* cv) {
421 validate_const_rec(field->get_name(), field->get_type(), cv);
422}
423
424/**
Mark Sleef0712dc2006-10-25 19:03:57 +0000425 * Parses a program
426 */
427void parse(t_program* program, t_program* parent_program) {
428 // Get scope file path
429 string path = program->get_path();
430
431 // Set current dir global, which is used in the include_file function
432 g_curdir = directory_name(path);
433 g_curpath = path;
434
435 // Open the file
436 yyin = fopen(path.c_str(), "r");
437 if (yyin == 0) {
438 failure("Could not open input file: \"%s\"", path.c_str());
439 }
440
441 // Create new scope and scan for includes
442 pverbose("Scanning %s for includes\n", path.c_str());
443 g_parse_mode = INCLUDES;
444 g_program = program;
445 g_scope = program->scope();
Mark Slee30152872006-11-28 01:24:07 +0000446 try {
Mark Slee36bfa2e2007-01-19 20:09:51 +0000447 yylineno = 1;
Mark Slee30152872006-11-28 01:24:07 +0000448 if (yyparse() != 0) {
449 failure("Parser error during include pass.");
450 }
451 } catch (string x) {
452 failure(x.c_str());
Mark Sleef0712dc2006-10-25 19:03:57 +0000453 }
454 fclose(yyin);
455
456 // Recursively parse all the include programs
457 vector<t_program*>& includes = program->get_includes();
458 vector<t_program*>::iterator iter;
459 for (iter = includes.begin(); iter != includes.end(); ++iter) {
460 parse(*iter, program);
461 }
462
463 // Parse the program the file
464 g_parse_mode = PROGRAM;
465 g_program = program;
466 g_scope = program->scope();
467 g_parent_scope = (parent_program != NULL) ? parent_program->scope() : NULL;
468 g_parent_prefix = program->get_name() + ".";
469 g_curpath = path;
470 yyin = fopen(path.c_str(), "r");
471 if (yyin == 0) {
472 failure("Could not open input file: \"%s\"", path.c_str());
473 }
474 pverbose("Parsing %s for types\n", path.c_str());
Mark Slee36bfa2e2007-01-19 20:09:51 +0000475 yylineno = 1;
Mark Sleef0712dc2006-10-25 19:03:57 +0000476 if (yyparse() != 0) {
477 failure("Parser error during types pass.");
478 }
479 fclose(yyin);
480}
481
482/**
483 * Generate code
484 */
485void generate(t_program* program) {
486 // Oooohh, recursive code generation, hot!!
487 if (gen_recurse) {
488 const vector<t_program*>& includes = program->get_includes();
489 for (size_t i = 0; i < includes.size(); ++i) {
490 generate(includes[i]);
491 }
492 }
493
494 // Generate code!
495 try {
496 pverbose("Program: %s\n", program->get_path().c_str());
497
498 if (gen_cpp) {
499 pverbose("Generating C++\n");
500 t_cpp_generator* cpp = new t_cpp_generator(program);
501 cpp->generate_program();
502 delete cpp;
503 }
504
505 if (gen_java) {
506 pverbose("Generating Java\n");
507 t_java_generator* java = new t_java_generator(program);
508 java->generate_program();
509 delete java;
510 }
511
512 if (gen_php) {
513 pverbose("Generating PHP\n");
514 t_php_generator* php = new t_php_generator(program, false);
515 php->generate_program();
516 delete php;
517 }
518
519 if (gen_phpi) {
520 pverbose("Generating PHP-inline\n");
521 t_php_generator* phpi = new t_php_generator(program, true);
522 phpi->generate_program();
523 delete phpi;
524 }
525
526 if (gen_py) {
527 pverbose("Generating Python\n");
528 t_py_generator* py = new t_py_generator(program);
529 py->generate_program();
530 delete py;
531 }
Mark Slee0e0ff7e2007-01-18 22:59:59 +0000532
Mark Slee6d7d5952007-01-27 01:44:22 +0000533 if (gen_rb) {
534 pverbose("Generating Ruby\n");
535 t_rb_generator* rb = new t_rb_generator(program);
536 rb->generate_program();
537 delete rb;
538 }
539
Mark Slee0e0ff7e2007-01-18 22:59:59 +0000540 if (gen_xsd) {
541 pverbose("Generating XSD\n");
542 t_xsd_generator* xsd = new t_xsd_generator(program);
543 xsd->generate_program();
544 delete xsd;
545 }
546
Mark Sleef0712dc2006-10-25 19:03:57 +0000547 } catch (string s) {
548 printf("Error: %s\n", s.c_str());
549 } catch (const char* exc) {
550 printf("Error: %s\n", exc);
551 }
552
553}
554
555/**
Mark Sleef5377b32006-10-10 01:42:59 +0000556 * Parse it up.. then spit it back out, in pretty much every language. Alright
557 * not that many languages, but the cool ones that we care about.
Mark Slee31985722006-05-24 21:45:31 +0000558 */
559int main(int argc, char** argv) {
560 int i;
Mark Sleef5377b32006-10-10 01:42:59 +0000561
Mark Sleeb15a68b2006-06-07 06:46:24 +0000562 // Setup time string
563 time_t now = time(NULL);
564 g_time_str = ctime(&now);
Mark Slee31985722006-05-24 21:45:31 +0000565
Mark Sleef0712dc2006-10-25 19:03:57 +0000566 // Check for necessary arguments, you gotta have at least a filename and
567 // an output language flag
Mark Sleeb15a68b2006-06-07 06:46:24 +0000568 if (argc < 2) {
569 usage();
570 }
Mark Slee31985722006-05-24 21:45:31 +0000571
Mark Sleef5377b32006-10-10 01:42:59 +0000572 // Hacky parameter handling... I didn't feel like using a library sorry!
Mark Slee31985722006-05-24 21:45:31 +0000573 for (i = 1; i < argc-1; i++) {
Mark Sleefdbee812006-09-27 18:50:48 +0000574 char* arg;
Mark Slee2329a832006-11-09 00:23:30 +0000575
Mark Sleefdbee812006-09-27 18:50:48 +0000576 arg = strtok(argv[i], " ");
577 while (arg != NULL) {
Mark Slee2329a832006-11-09 00:23:30 +0000578 // Treat double dashes as single dashes
Mark Slee52cb2232006-11-10 22:32:07 +0000579 if (arg[0] == '-' && arg[1] == '-') {
Mark Slee2329a832006-11-09 00:23:30 +0000580 ++arg;
581 }
582
583 if (strcmp(arg, "-debug") == 0) {
Mark Sleefdbee812006-09-27 18:50:48 +0000584 g_debug = 1;
Mark Slee2329a832006-11-09 00:23:30 +0000585 } else if (strcmp(arg, "-nowarn") == 0) {
Mark Sleef0712dc2006-10-25 19:03:57 +0000586 g_warn = 0;
Mark Slee2329a832006-11-09 00:23:30 +0000587 } else if (strcmp(arg, "-strict") == 0) {
Mark Sleef0712dc2006-10-25 19:03:57 +0000588 g_warn = 2;
Mark Slee2329a832006-11-09 00:23:30 +0000589 } else if (strcmp(arg, "-v") == 0 || strcmp(arg, "-verbose") == 0 ) {
Mark Sleef0712dc2006-10-25 19:03:57 +0000590 g_verbose = 1;
Mark Slee2329a832006-11-09 00:23:30 +0000591 } else if (strcmp(arg, "-r") == 0 || strcmp(arg, "-recurse") == 0 ) {
Mark Sleef0712dc2006-10-25 19:03:57 +0000592 gen_recurse = true;
Mark Slee2329a832006-11-09 00:23:30 +0000593 } else if (strcmp(arg, "-cpp") == 0) {
Mark Sleefdbee812006-09-27 18:50:48 +0000594 gen_cpp = true;
Mark Slee2329a832006-11-09 00:23:30 +0000595 } else if (strcmp(arg, "-java") == 0) {
Mark Sleefdbee812006-09-27 18:50:48 +0000596 gen_java = true;
Mark Slee2329a832006-11-09 00:23:30 +0000597 } else if (strcmp(arg, "-php") == 0) {
Mark Sleefdbee812006-09-27 18:50:48 +0000598 gen_php = true;
Mark Slee2329a832006-11-09 00:23:30 +0000599 } else if (strcmp(arg, "-phpi") == 0) {
Mark Sleef5377b32006-10-10 01:42:59 +0000600 gen_phpi = true;
Mark Slee2329a832006-11-09 00:23:30 +0000601 } else if (strcmp(arg, "-py") == 0) {
Mark Sleefdbee812006-09-27 18:50:48 +0000602 gen_py = true;
Mark Slee6d7d5952007-01-27 01:44:22 +0000603 } else if (strcmp(arg, "-rb") == 0) {
604 gen_rb = true;
Mark Slee0e0ff7e2007-01-18 22:59:59 +0000605 } else if (strcmp(arg, "-xsd") == 0) {
606 gen_xsd = true;
Martin Kraemer32c66e12006-11-09 00:06:36 +0000607 } else if (strcmp(arg, "-I") == 0) {
608 // An argument of "-I\ asdf" is invalid and has unknown results
609 arg = argv[++i];
610
611 if (arg == NULL) {
612 fprintf(stderr, "!!! Missing Include directory");
613 usage();
614 }
615 g_incl_searchpath.push_back(arg);
Mark Sleefdbee812006-09-27 18:50:48 +0000616 } else {
617 fprintf(stderr, "!!! Unrecognized option: %s\n", arg);
618 usage();
619 }
620
621 // Tokenize more
622 arg = strtok(NULL, " ");
Mark Slee31985722006-05-24 21:45:31 +0000623 }
624 }
625
Mark Sleef0712dc2006-10-25 19:03:57 +0000626 // You gotta generate something!
Mark Slee6d7d5952007-01-27 01:44:22 +0000627 if (!gen_cpp && !gen_java && !gen_php && !gen_phpi && !gen_py && !gen_rb && !gen_xsd) {
Mark Sleeb15a68b2006-06-07 06:46:24 +0000628 fprintf(stderr, "!!! No output language(s) specified\n\n");
629 usage();
630 }
Mark Sleef0712dc2006-10-25 19:03:57 +0000631
632 // Real-pathify it
633 char rp[PATH_MAX];
634 if (realpath(argv[i], rp) == NULL) {
635 failure("Could not open input file: %s", argv[i]);
Mark Slee31985722006-05-24 21:45:31 +0000636 }
Mark Sleef0712dc2006-10-25 19:03:57 +0000637 string input_file(rp);
638
Mark Sleef5377b32006-10-10 01:42:59 +0000639 // Instance of the global parse tree
Mark Sleef0712dc2006-10-25 19:03:57 +0000640 t_program* program = new t_program(input_file);
641
642 // Initialize global types
643 g_type_void = new t_base_type("void", t_base_type::TYPE_VOID);
644 g_type_string = new t_base_type("string", t_base_type::TYPE_STRING);
Mark Sleeb6200d82007-01-19 19:14:36 +0000645 g_type_slist = new t_base_type("string", t_base_type::TYPE_STRING);
646 ((t_base_type*)g_type_slist)->set_string_list(true);
Mark Sleef0712dc2006-10-25 19:03:57 +0000647 g_type_bool = new t_base_type("bool", t_base_type::TYPE_BOOL);
648 g_type_byte = new t_base_type("byte", t_base_type::TYPE_BYTE);
649 g_type_i16 = new t_base_type("i16", t_base_type::TYPE_I16);
650 g_type_i32 = new t_base_type("i32", t_base_type::TYPE_I32);
651 g_type_i64 = new t_base_type("i64", t_base_type::TYPE_I64);
652 g_type_double = new t_base_type("double", t_base_type::TYPE_DOUBLE);
Mark Sleee8540632006-05-30 09:24:40 +0000653
Mark Sleef5377b32006-10-10 01:42:59 +0000654 // Parse it!
Mark Sleef0712dc2006-10-25 19:03:57 +0000655 parse(program, NULL);
Mark Slee31985722006-05-24 21:45:31 +0000656
Mark Sleef0712dc2006-10-25 19:03:57 +0000657 // Generate it!
658 generate(program);
Mark Sleeb15a68b2006-06-07 06:46:24 +0000659
Mark Sleef0712dc2006-10-25 19:03:57 +0000660 // Clean up. Who am I kidding... this program probably orphans heap memory
661 // all over the place, but who cares because it is about to exit and it is
662 // all referenced and used by this wacky parse tree up until now anyways.
Mark Sleeb15a68b2006-06-07 06:46:24 +0000663
Mark Sleef0712dc2006-10-25 19:03:57 +0000664 delete program;
665 delete g_type_void;
666 delete g_type_string;
667 delete g_type_bool;
668 delete g_type_byte;
669 delete g_type_i16;
670 delete g_type_i32;
671 delete g_type_i64;
672 delete g_type_double;
Mark Slee31985722006-05-24 21:45:31 +0000673
674 // Finished
Mark Slee31985722006-05-24 21:45:31 +0000675 return 0;
676}