// ********************************************************************** // // Copyright (c) 2003-2006 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 using namespace std; using namespace Ice; using namespace IceInternal; namespace IceInternal { extern IceUtil::Handle theCollector; } void Ice::collectGarbage() { if(theCollector) { theCollector->collectGarbage(); } } StringSeq Ice::argsToStringSeq(int argc, char* argv[]) { StringSeq result; for(int i = 0; i < argc; i++) { result.push_back(argv[i]); } return result; } void Ice::stringSeqToArgs(const StringSeq& args, int& argc, char* argv[]) { // // Shift all elements in argv which are present in args to the // beginning of argv. We record the original value of argc so // that we can know later if we've shifted the array. // const int argcOrig = argc; int i = 0; while(i < argc) { if(find(args.begin(), args.end(), argv[i]) == args.end()) { for(int j = i; j < argc - 1; j++) { argv[j] = argv[j + 1]; } --argc; } else { ++i; } } // // Make sure that argv[argc] == 0, the ISO C++ standard requires this. // We can only do this if we've shifted the array, otherwise argv[argc] // may point to an invalid address. // if(argv && argcOrig != argc) { argv[argc] = 0; } } PropertiesPtr Ice::createProperties(const StringConverterPtr& converter) { return new PropertiesI(converter); } PropertiesPtr Ice::createProperties(StringSeq& args, const PropertiesPtr& defaults, const StringConverterPtr& converter) { return new PropertiesI(args, defaults, converter); } PropertiesPtr Ice::createProperties(int& argc, char* argv[], const PropertiesPtr& defaults, const StringConverterPtr& converter) { StringSeq args = argsToStringSeq(argc, argv); PropertiesPtr properties = createProperties(args, defaults, converter); stringSeqToArgs(args, argc, argv); return properties; } inline void checkIceVersion(Int version) { #ifndef ICE_IGNORE_VERSION // // Major and minor version numbers must match. // if(ICE_INT_VERSION / 100 != version / 100) { throw VersionMismatchException(__FILE__, __LINE__); } // // The caller's patch level cannot be greater than library's patch level. (Patch level changes are // backward-compatible, but not forward-compatible.) // if(version % 100 > ICE_INT_VERSION % 100) { throw VersionMismatchException(__FILE__, __LINE__); } #endif } CommunicatorPtr Ice::initialize(int& argc, char* argv[], const InitializationData& initializationData, Int version) { checkIceVersion(version); InitializationData initData = initializationData; initData.properties = createProperties(argc, argv, initData.properties, initData.stringConverter); CommunicatorI* communicatorI = new CommunicatorI(initData); CommunicatorPtr result = communicatorI; // For exception safety. communicatorI->finishSetup(argc, argv); return result; } CommunicatorPtr Ice::initialize(StringSeq& args, const InitializationData& initializationData, Int version) { int origArgc; char** argv; CommunicatorPtr communicator; try { // // Make a dummy argc/argv. // (We can't use argsToStringSeq() because that requires an already initialized argv.) // int argc = args.size(); origArgc = argc; argv = new char*[args.size() + 1]; int i; for(i = 0; i != argc; ++i) { argv[i] = new char[args[i].size() + 1]; #if defined(_MSC_VER) && (_MSC_VER >= 1400) strcpy_s(argv[i], args[i].size() + 1, args[i].c_str()); #else strcpy(argv[i], args[i].c_str()); #endif } argv[argc] = 0; communicator = initialize(argc, argv, initializationData, version); args = argsToStringSeq(argc, argv); for(i = 0; i < origArgc; ++i) { delete[] argv[i]; } delete[] argv; } catch(...) { for(int i = 0; i < origArgc; ++i) { delete[] argv[i]; } delete[] argv; throw; } return communicator; } CommunicatorPtr Ice::initialize(const InitializationData& initData, Int version) { // // We can't simply call the other initialize() because this one does NOT read // the config file, while the other one always does. // checkIceVersion(version); CommunicatorI* communicatorI = new CommunicatorI(initData); CommunicatorPtr result = communicatorI; // For exception safety. int argc = 0; char* argv[] = { 0 }; communicatorI->finishSetup(argc, argv); return result; } CommunicatorPtr Ice::initializeWithProperties(int& argc, char* argv[], const PropertiesPtr& properties, Int version) { InitializationData initData; initData.properties = properties; return initialize(argc, argv, initData, version); } CommunicatorPtr Ice::initializeWithLogger(int& argc, char* argv[], const LoggerPtr& logger, Int version) { InitializationData initData; initData.logger = logger; return initialize(argc, argv, initData, version); } CommunicatorPtr Ice::initializeWithPropertiesAndLogger(int& argc, char* argv[], const PropertiesPtr& properties, const LoggerPtr& logger, Int version) { InitializationData initData; initData.properties = properties; initData.logger = logger; return initialize(argc, argv, initData, version); } InputStreamPtr Ice::createInputStream(const CommunicatorPtr& communicator, const vector& bytes) { return new InputStreamI(communicator, bytes); } OutputStreamPtr Ice::createOutputStream(const CommunicatorPtr& communicator) { return new OutputStreamI(communicator); } static IceUtil::StaticMutex processLoggerMutex = ICE_STATIC_MUTEX_INITIALIZER; static Ice::LoggerPtr processLogger; LoggerPtr Ice::getProcessLogger() { IceUtil::StaticMutex::Lock lock(processLoggerMutex); if(processLogger == 0) { // // TODO: Would be nice to be able to use process name as prefix by default. // processLogger = new Ice::LoggerI(""); } return processLogger; } void Ice::setProcessLogger(const LoggerPtr& logger) { IceUtil::StaticMutex::Lock lock(processLoggerMutex); processLogger = logger; } InstancePtr IceInternal::getInstance(const CommunicatorPtr& communicator) { CommunicatorI* p = dynamic_cast(communicator.get()); assert(p); return p->_instance; }