//%///////////////////////////////////////////////////////////////////////////// // // Copyright (c) 2000, 2001, 2002 BMC Software, Hewlett-Packard Company, IBM, // The Open Group, Tivoli Systems // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to // deal in the Software without restriction, including without limitation the // rights to use, copy, modify, merge, publish, distribute, sublicense, and/or // sell copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // THE ABOVE COPYRIGHT NOTICE AND THIS PERMISSION NOTICE SHALL BE INCLUDED IN // ALL COPIES OR SUBSTANTIAL PORTIONS OF THE SOFTWARE. THE SOFTWARE IS PROVIDED // "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT // LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR // PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT // HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN // ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. // //============================================================================== // // Author: Jenny Yu, Hewlett-Packard Company (jenny_yu@hp.com) // // Modified By: // //%//////////////////////////////////////////////////////////////////////////// ///////////////////////////////////////////////////////////////////////////// // ShutdownService ///////////////////////////////////////////////////////////////////////////// #include #include #include #include #include #include #include #include #include #include #include #include PEGASUS_USING_STD; PEGASUS_NAMESPACE_BEGIN /** The constant representing the shutdown timeout property name */ static String SHUTDOWN_TIMEOUT_PROPERTY = "shutdownTimeout"; /** Initialize ShutdownService instance */ ShutdownService* ShutdownService::_instance = 0; /** Initialize all other class variables */ CIMServer* ShutdownService::_cimserver = 0; Uint32 ShutdownService::_shutdownTimeout = 0; pegasus_internal_identity ShutdownService::_id = 0; ModuleController* ShutdownService::_controller = 0; ModuleController::client_handle* ShutdownService::_client_handle = 0; /** Constructor. */ ShutdownService::ShutdownService(CIMServer* cimserver) { _cimserver = cimserver; // // get client identify // _id = peg_credential_types::MODULE; // // get module controller // _controller = &(ModuleController::get_client_handle(_id, &_client_handle)); if((_client_handle == NULL)) { throw UninitializedObjectException(); } } /** Destructor. */ ShutdownService::~ShutdownService() { } /** return a pointer to the ShutdownService instance. */ ShutdownService* ShutdownService::getInstance(CIMServer* cimserver) { if (!_instance) { _instance = new ShutdownService(cimserver); } return _instance; } /** The shutdown method to be called by the ShutdownProvider to process a shutdown request from the CLI client. */ void ShutdownService::shutdown(Boolean force, Uint32 timeout) { PEG_METHOD_ENTER(TRC_REPOSITORY, "ShutdownService::shutdown"); // // Initialize variables // Boolean timeoutExpired = false; Boolean noMoreRequests = false; _shutdownTimeout = timeout; try { // // set CIMServer state to TERMINATING // _cimserver->setState(CIMServerState::TERMINATING); // // Tell the CIMServer to stop accepting new client connection requests. // _cimserver->stopClientConnection(); // // Determine if there are any outstanding CIM operation requests // (take into account that one of the request is the shutdown request). // Uint32 requestCount = _cimserver->getOutstandingRequestCount(); if (requestCount > 1) { noMoreRequests = _waitUntilNoMoreRequests(); } else { noMoreRequests = true; } // // proceed to shutdown the CIMServer // _shutdownCIMServer(); } catch(CIMException & e) { PEG_TRACE_STRING(TRC_PROVIDERMANAGER, Tracer::LEVEL4, "Error occurred during CIMServer shutdown: " + e.getMessage()); } catch(Exception & e) { PEG_TRACE_STRING(TRC_PROVIDERMANAGER, Tracer::LEVEL4, "Error occurred during CIMServer shutdown: " + e.getMessage()); } catch(...) { PEG_TRACE_STRING(TRC_PROVIDERMANAGER, Tracer::LEVEL4, "Unexpected error occured during CIMServer shutdown. "); } // // All done // PEG_METHOD_EXIT(); return; } /**********************************************************/ /* callback */ /**********************************************************/ void ShutdownService::async_callback(Uint32 user_data, Message *reply, void *parm) { callback_data *cb_data = reinterpret_cast(parm); cb_data->reply = reply; cb_data->client_sem.signal(); } /**********************************************************/ /* private methods */ /**********************************************************/ void ShutdownService::_shutdownCIMServer() { PEG_METHOD_ENTER(TRC_REPOSITORY, "ShutdownService::_shutdownCIMServer"); // // Shutdown the providers // _shutdownProviders(); // // Shutdown the Cimom services // _shutdownCimomServices(); // // Tell CIMServer to shutdown completely. // _cimserver->shutdown(); PEG_METHOD_EXIT(); return; } void ShutdownService::_shutdownCimomServices() { PEG_METHOD_ENTER(TRC_REPOSITORY, "ShutdownService::_shutdownCimomServices"); // // Shutdown the Indication Service // _sendShutdownRequestToService(PEGASUS_QUEUENAME_INDICATIONSERVICE); // Shutdown the Indication Handler Service _sendShutdownRequestToService(PEGASUS_QUEUENAME_INDHANDLERMANAGER); // // shutdown Authenticator Delegator Service // _sendShutdownRequestToService(PEGASUS_QUEUENAME_HTTPAUTHDELEGATOR); // // shutdown CIM Operation Request Authorizer Service // _sendShutdownRequestToService(PEGASUS_QUEUENAME_OPREQAUTHORIZER); // // shutdown CIM Operation Request Decoder Service // _sendShutdownRequestToService(PEGASUS_QUEUENAME_OPREQDECODER); // // shutdown CIM Operation Request Dispatcher Service // _sendShutdownRequestToService(PEGASUS_QUEUENAME_OPREQDISPATCHER); // // shutdown CIM Export Request Decoder Service // _sendShutdownRequestToService(PEGASUS_QUEUENAME_EXPORTREQDECODER); // // shutdown CIM Export Request Dispatcher Service // _sendShutdownRequestToService(PEGASUS_QUEUENAME_EXPORTREQDISPATCHER); PEG_METHOD_EXIT(); return; } void ShutdownService::_sendShutdownRequestToService(const char * serviceName) { MessageQueue* queue = MessageQueue::lookup(serviceName); if (queue == 0) { // service not found, just return return; } MessageQueueService * _service = dynamic_cast(queue); Uint32 _queueId = _service->getQueueId(); // // create stop request // CimServiceStop *stopRequest = new CimServiceStop( _service->get_next_xid(), // routing NULL, // operation _queueId, // destination _controller->getQueueId(), // response true); // blocking // ATTN-JY-P2-05162002: call ClientSendWait, until asyn_callback is fixed // // Now send Stop request to service // AsyncMessage *reply = _controller->ClientSendWait(*_client_handle, _queueId, stopRequest); //delete stopRequest; return; } void ShutdownService::_shutdownProviders() { PEG_METHOD_ENTER(TRC_REPOSITORY, "ShutdownService::_shutdownProviders"); // // get provider manager service // MessageQueue * queue = MessageQueue::lookup(PEGASUS_QUEUENAME_PROVIDERMANAGER_CPP); if (queue == 0) { PEG_METHOD_EXIT(); return; } MessageQueueService * _service = dynamic_cast(queue); Uint32 _queueId = _service->getQueueId(); // // create stop all providers request // CIMStopAllProvidersRequestMessage * stopRequest = new CIMStopAllProvidersRequestMessage( XmlWriter::getNextMessageId (), QueueIdStack(_queueId)); // // create async request message // AsyncLegacyOperationStart * asyncRequest = new AsyncLegacyOperationStart ( _service->get_next_xid(), NULL, _queueId, stopRequest, _queueId); // ATTN-JY-P2-05162002: call ClientSendWait, until asyn_callback is fixed AsyncReply * asyncReply = _controller->ClientSendWait(*_client_handle, _queueId, asyncRequest); CIMStopAllProvidersResponseMessage * response = reinterpret_cast( (static_cast(asyncReply))->get_result()); if (response->cimException.getCode() != CIM_ERR_SUCCESS) { CIMException e = response->cimException; delete stopRequest; delete asyncRequest; delete asyncReply; delete response; PEG_METHOD_EXIT(); throw (e); } //delete stopRequest; delete asyncRequest; delete asyncReply; delete response; PEG_METHOD_EXIT(); return; } Boolean ShutdownService::_waitUntilNoMoreRequests() { Uint32 maxWaitTime = _shutdownTimeout; // maximum wait time in seconds Uint32 waitInterval = 1; // one second wait interval Uint32 requestCount = _cimserver->getOutstandingRequestCount(); // create a mutex Mutex _mutex; // create a condition Condition _cond(_mutex); // // Loop and wait one second until either there is no more requests // or until timeout expires. // while ( requestCount > 1 && maxWaitTime > 0) { // lock the mutex _mutex.lock(pegasus_thread_self()); try { _cond.unlocked_timed_wait(waitInterval*1000, pegasus_thread_self()); } catch (TimeOut to) { requestCount = _cimserver->getOutstandingRequestCount(); maxWaitTime = maxWaitTime - waitInterval; } catch (...) { maxWaitTime = 0; } // unlock the mutex _mutex.unlock(); } if (requestCount > 1) return false; else return true; } PEGASUS_NAMESPACE_END