/* * Copyright 2001-2007 Adrian Thurston */ /* This file is part of Ragel. * * Ragel is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * Ragel is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with Ragel; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include #include #include #include #include #include #include "rlgen-ruby.h" #include "xmlparse.h" #include "pcheck.h" #include "vector.h" #include "version.h" #include "common.h" #include "rubycodegen.h" using std::istream; using std::ifstream; using std::ostream; using std::ios; using std::cin; using std::cout; using std::cerr; using std::endl; /* Io globals. */ istream *inStream = 0; ostream *outStream = 0; output_filter *outFilter = 0; char *outputFileName = 0; /* Graphviz dot file generation. */ bool graphvizDone = false; int numSplitPartitions = 0; bool printPrintables = false; /* Print a summary of the options. */ void usage() { cout << "usage: " PROGNAME " [options] file\n" "general:\n" " -h, -H, -?, --help Print this usage and exit\n" " -v, --version Print version information and exit\n" " -o Write output to \n" ; } /* Print version information. */ void version() { cout << "Ragel Code Generator for Ruby" << endl << "Version " VERSION << ", " PUBDATE << endl << "Copyright (c) 2001-2007 by Adrian Thurston" << endl; } ostream &error() { gblErrorCount += 1; cerr << PROGNAME ": "; return cerr; } /* * Callbacks invoked by the XML data parser. */ /* Invoked by the parser when the root element is opened. */ ostream *openOutput( char *inputFile ) { if ( hostLang->lang != HostLang::Ruby ) { error() << "this code generator is for Ruby only" << endl; exit(1); } /* If the output format is code and no output file name is given, then * make a default. */ if ( outputFileName == 0 ) { char *ext = findFileExtension( inputFile ); if ( ext != 0 && strcmp( ext, ".rh" ) == 0 ) outputFileName = fileNameFromStem( inputFile, ".h" ); else outputFileName = fileNameFromStem( inputFile, ".rb" ); } /* Make sure we are not writing to the same file as the input file. */ if ( outputFileName != 0 && strcmp( inputFile, outputFileName ) == 0 ) { error() << "output file \"" << outputFileName << "\" is the same as the input file" << endl; } if ( outputFileName != 0 ) { /* Create the filter on the output and open it. */ outFilter = new output_filter( outputFileName ); outFilter->open( outputFileName, ios::out|ios::trunc ); if ( !outFilter->is_open() ) { error() << "error opening " << outputFileName << " for writing" << endl; exit(1); } /* Open the output stream, attaching it to the filter. */ outStream = new ostream( outFilter ); } else { /* Writing out ot std out. */ outStream = &cout; } return outStream; } /* Invoked by the parser when a ragel definition is opened. */ CodeGenData *makeCodeGen( char *sourceFileName, char *fsmName, ostream &out, bool wantComplete ) { CodeGenData *codeGen = new RubyCodeGen(out); codeGen->sourceFileName = sourceFileName; codeGen->fsmName = fsmName; codeGen->wantComplete = wantComplete; return codeGen; } /* Main, process args and call yyparse to start scanning input. */ int main(int argc, char **argv) { ParamCheck pc("o:vHh?-:", argc, argv); char *xmlInputFileName = 0; while ( pc.check() ) { switch ( pc.state ) { case ParamCheck::match: switch ( pc.parameter ) { /* Output. */ case 'o': if ( *pc.parameterArg == 0 ) error() << "a zero length output file name was given" << endl; else if ( outputFileName != 0 ) error() << "more than one output file name was given" << endl; else { /* Ok, remember the output file name. */ outputFileName = pc.parameterArg; } break; /* Version and help. */ case 'v': version(); exit(0); case 'H': case 'h': case '?': usage(); exit(0); case '-': if ( strcasecmp(pc.parameterArg, "help") == 0 ) { usage(); exit(0); } else if ( strcasecmp(pc.parameterArg, "version") == 0 ) { version(); exit(0); } else { error() << "--" << pc.parameterArg << " is an invalid argument" << endl; break; } } break; case ParamCheck::invalid: error() << "-" << pc.parameter << " is an invalid argument" << endl; break; case ParamCheck::noparam: if ( *pc.curArg == 0 ) error() << "a zero length input file name was given" << endl; else if ( xmlInputFileName != 0 ) error() << "more than one input file name was given" << endl; else { /* OK, Remember the filename. */ xmlInputFileName = pc.curArg; } break; } } /* Bail on above errors. */ if ( gblErrorCount > 0 ) exit(1); /* Open the input file for reading. */ if ( xmlInputFileName != 0 ) { /* Open the input file for reading. */ ifstream *inFile = new ifstream( xmlInputFileName ); inStream = inFile; if ( ! inFile->is_open() ) error() << "could not open " << xmlInputFileName << " for reading" << endl; } else { xmlInputFileName = ""; inStream = &cin; } /* Bail on above errors. */ if ( gblErrorCount > 0 ) exit(1); bool wantComplete = true; bool outputActive = true; /* Parse the input! */ xml_parse( *inStream, xmlInputFileName, outputActive, wantComplete ); /* If writing to a file, delete the ostream, causing it to flush. * Standard out is flushed automatically. */ if ( outputFileName != 0 ) { delete outStream; delete outFilter; } /* Finished, final check for errors.. */ if ( gblErrorCount > 0 ) { /* If we opened an output file, remove it. */ if ( outputFileName != 0 ) unlink( outputFileName ); exit(1); } return 0; }