// ********************************************************************** // // Copyright (c) 2003-2009 ZeroC, Inc. All rights reserved. // // This copy of Ice is licensed to you under the terms described in the // ICE_LICENSE file included in this distribution. // // ********************************************************************** #include #include #include #include #include #include #include #ifdef __BCPLUSPLUS__ # include #endif using namespace std; using namespace Slice; static IceUtil::StaticMutex _mutex = ICE_STATIC_MUTEX_INITIALIZER; static bool _interrupted = false; void interruptedCallback(int signal) { IceUtil::StaticMutex::Lock lock(_mutex); _interrupted = true; } void usage(const char* n) { cerr << "Usage: " << n << " [options] slice-files...\n"; cerr << "Options:\n" "-h, --help Show this message.\n" "-v, --version Display the Ice version.\n" "-DNAME Define NAME as 1.\n" "-DNAME=DEF Define NAME as DEF.\n" "-UNAME Remove any definition for NAME.\n" "-IDIR Put DIR in the include file search path.\n" "-E Print preprocessor output on stdout.\n" "--output-dir DIR Create files in the directory DIR.\n" "--tie Generate TIE classes.\n" "--impl Generate sample implementations.\n" "--impl-tie Generate sample TIE implementations.\n" "--depend Generate Makefile dependencies.\n" "--depend-xml Generate dependencies in XML format.\n" "--list-generated Emit list of generated files in XML format.\n" "-d, --debug Print debug messages.\n" "--ice Permit `Ice' prefix (for building Ice source code only)\n" "--checksum CLASS Generate checksums for Slice definitions into CLASS.\n" "--stream Generate marshaling support for public stream API.\n" "--meta META Define global metadata directive META.\n" ; } int compile(int argc, char* argv[]) { IceUtilInternal::Options opts; opts.addOpt("h", "help"); opts.addOpt("v", "version"); opts.addOpt("D", "", IceUtilInternal::Options::NeedArg, "", IceUtilInternal::Options::Repeat); opts.addOpt("U", "", IceUtilInternal::Options::NeedArg, "", IceUtilInternal::Options::Repeat); opts.addOpt("I", "", IceUtilInternal::Options::NeedArg, "", IceUtilInternal::Options::Repeat); opts.addOpt("E"); opts.addOpt("", "output-dir", IceUtilInternal::Options::NeedArg); opts.addOpt("", "tie"); opts.addOpt("", "impl"); opts.addOpt("", "impl-tie"); opts.addOpt("", "depend"); opts.addOpt("", "depend-xml"); opts.addOpt("", "list-generated"); opts.addOpt("d", "debug"); opts.addOpt("", "ice"); opts.addOpt("", "checksum", IceUtilInternal::Options::NeedArg); opts.addOpt("", "stream"); opts.addOpt("", "meta", IceUtilInternal::Options::NeedArg, "", IceUtilInternal::Options::Repeat); vectorargs; try { #if defined(__BCPLUSPLUS__) && (__BCPLUSPLUS__ >= 0x0600) IceUtil::DummyBCC dummy; #endif args = opts.parse(argc, (const char**)argv); } catch(const IceUtilInternal::BadOptException& e) { cerr << argv[0] << ": error: " << e.reason << endl; usage(argv[0]); return EXIT_FAILURE; } if(opts.isSet("help")) { usage(argv[0]); return EXIT_SUCCESS; } if(opts.isSet("version")) { cerr << ICE_STRING_VERSION << endl; return EXIT_SUCCESS; } vector cppArgs; vector optargs = opts.argVec("D"); vector::const_iterator i; for(i = optargs.begin(); i != optargs.end(); ++i) { cppArgs.push_back("-D" + *i); } optargs = opts.argVec("U"); for(i = optargs.begin(); i != optargs.end(); ++i) { cppArgs.push_back("-U" + *i); } vector includePaths = opts.argVec("I"); for(i = includePaths.begin(); i != includePaths.end(); ++i) { cppArgs.push_back("-I" + Preprocessor::normalizeIncludePath(*i)); } bool preprocess = opts.isSet("E"); string output = opts.optArg("output-dir"); bool tie = opts.isSet("tie"); bool impl = opts.isSet("impl"); bool implTie = opts.isSet("impl-tie"); bool depend = opts.isSet("depend"); bool dependxml = opts.isSet("depend-xml"); bool debug = opts.isSet("debug"); bool ice = opts.isSet("ice"); string checksumClass = opts.optArg("checksum"); bool stream = opts.isSet("stream"); bool listGenerated = opts.isSet("list-generated"); StringList globalMetadata; vector v = opts.argVec("meta"); copy(v.begin(), v.end(), back_inserter(globalMetadata)); if(args.empty()) { getErrorStream() << argv[0] << ": error: no input file" << endl; usage(argv[0]); return EXIT_FAILURE; } if(impl && implTie) { getErrorStream() << argv[0] << ": error: cannot specify both --impl and --impl-tie" << endl; usage(argv[0]); return EXIT_FAILURE; } int status = EXIT_SUCCESS; ChecksumMap checksums; IceUtil::CtrlCHandler ctrlCHandler; ctrlCHandler.setCallback(interruptedCallback); if(dependxml) { cout << "\n" << endl; } for(i = args.begin(); i != args.end(); ++i) { if(depend || dependxml) { Preprocessor icecpp(argv[0], *i, cppArgs); FILE* cppHandle = icecpp.preprocess(false); if(cppHandle == 0) { return EXIT_FAILURE; } UnitPtr u = Unit::createUnit(false, false, ice); int parseStatus = u->parse(*i, cppHandle, debug); u->destroy(); if(parseStatus == EXIT_FAILURE) { return EXIT_FAILURE; } if(!icecpp.printMakefileDependencies(depend ? Preprocessor::Java : Preprocessor::JavaXML, includePaths)) { return EXIT_FAILURE; } if(!icecpp.close()) { return EXIT_FAILURE; } } else { ostringstream os; if(listGenerated) { Slice::setErrorStream(os); } FileTracker::instance()->setSource(*i); Preprocessor icecpp(argv[0], *i, cppArgs); FILE* cppHandle = icecpp.preprocess(true); if(cppHandle == 0) { FileTracker::instance()->setOutput(os.str(), true); status = EXIT_FAILURE; break; } if(preprocess) { char buf[4096]; while(fgets(buf, static_cast(sizeof(buf)), cppHandle) != NULL) { if(fputs(buf, stdout) == EOF) { return EXIT_FAILURE; } } if(!icecpp.close()) { return EXIT_FAILURE; } } else { UnitPtr p = Unit::createUnit(false, false, ice, globalMetadata); int parseStatus = p->parse(*i, cppHandle, debug, Ice); if(!icecpp.close()) { p->destroy(); return EXIT_FAILURE; } if(parseStatus == EXIT_FAILURE) { p->destroy(); FileTracker::instance()->setOutput(os.str(), true); status = EXIT_FAILURE; } else { try { Gen gen(argv[0], icecpp.getBaseName(), includePaths, output); gen.generate(p, stream); if(tie) { gen.generateTie(p); } if(impl) { gen.generateImpl(p); } if(implTie) { gen.generateImplTie(p); } if(!checksumClass.empty()) { // // Calculate checksums for the Slice definitions in the unit. // ChecksumMap m = createChecksums(p); copy(m.begin(), m.end(), inserter(checksums, checksums.begin())); } FileTracker::instance()->setOutput(os.str(), false); } catch(const Slice::FileException& ex) { // // If a file could not be created then cleanup any files we've already created. // FileTracker::instance()->cleanup(); p->destroy(); os << argv[0] << ": error: " << ex.reason() << endl; FileTracker::instance()->setOutput(os.str(), true); status = EXIT_FAILURE; break; } } p->destroy(); } } { IceUtil::StaticMutex::Lock lock(_mutex); if(_interrupted) { // // If the translator was interrupted then cleanup any files we've already created. // FileTracker::instance()->cleanup(); return EXIT_FAILURE; } } } if(dependxml) { cout << "\n"; } if(status == EXIT_SUCCESS && !checksumClass.empty()) { try { Gen::writeChecksumClass(checksumClass, output, checksums); } catch(const Slice::FileException& ex) { // If a file could not be created, then // cleanup any created files. FileTracker::instance()->cleanup(); getErrorStream() << argv[0] << ": error: " << ex.reason() << endl; return EXIT_FAILURE; } } if(listGenerated) { FileTracker::instance()->dumpxml(); } return status; } int main(int argc, char* argv[]) { try { return compile(argc, argv); } catch(const IceUtil::Exception& ex) { cerr << ex.what() << endl; #ifdef __GNUC__ cerr << ex.ice_stackTrace() << endl; #endif return EXIT_FAILURE; } catch(const std::exception& ex) { cerr << ex.what() << endl; return EXIT_FAILURE; } catch(const std::string& msg) { cerr << msg << endl; return EXIT_FAILURE; } catch(const char* msg) { cerr << msg << endl; return EXIT_FAILURE; } catch(...) { cerr << "unknown exception" << endl; return EXIT_FAILURE; } }