diff options
author | Matthew Newhook <matthew@zeroc.com> | 2006-12-22 18:52:53 +0000 |
---|---|---|
committer | Matthew Newhook <matthew@zeroc.com> | 2006-12-22 18:52:53 +0000 |
commit | 84d9f5c369fceccfa16401ed50783ed2ad209559 (patch) | |
tree | 689d30c8707629f1db8155edadb47ce1bce363ad /cpp/demo/Ice/session/Client.cpp | |
parent | Added autoflushing of batches (diff) | |
download | ice-84d9f5c369fceccfa16401ed50783ed2ad209559.tar.bz2 ice-84d9f5c369fceccfa16401ed50783ed2ad209559.tar.xz ice-84d9f5c369fceccfa16401ed50783ed2ad209559.zip |
http://bugzilla.zeroc.com/bugzilla/show_bug.cgi?id=1391.
Diffstat (limited to 'cpp/demo/Ice/session/Client.cpp')
-rwxr-xr-x | cpp/demo/Ice/session/Client.cpp | 96 |
1 files changed, 75 insertions, 21 deletions
diff --git a/cpp/demo/Ice/session/Client.cpp b/cpp/demo/Ice/session/Client.cpp index fc78b4dee0c..e0be25f7d61 100755 --- a/cpp/demo/Ice/session/Client.cpp +++ b/cpp/demo/Ice/session/Client.cpp @@ -72,10 +72,20 @@ class SessionClient : public Ice::Application public: virtual int run(int, char*[]); + virtual void interruptCallback(int); private: void menu(); + void cleanup(bool); + + // + // The interrupt callback and main can run concurrently with one + // another so shared variables must be mutex protected. + // + IceUtil::Mutex _mutex; + SessionRefreshThreadPtr _refresh; + SessionPrx _session; }; int @@ -88,6 +98,12 @@ main(int argc, char* argv[]) int SessionClient::run(int argc, char* argv[]) { + // + // Since this is an interactive demo we want the custom interrupt + // callback to be called when the process is interrupted. + // + userCallbackOnInterrupt(); + string name; cout << "Please enter your name ==> "; cin >> name; @@ -104,11 +120,14 @@ SessionClient::run(int argc, char* argv[]) return EXIT_FAILURE; } - SessionPrx session = factory->create(name); - - SessionRefreshThreadPtr refresh = new SessionRefreshThread( - communicator()->getLogger(), IceUtil::Time::seconds(5), session); - refresh->start(); + { + IceUtil::Mutex::Lock sync(_mutex); + _session = factory->create(name); + + _refresh = new SessionRefreshThread( + communicator()->getLogger(), IceUtil::Time::seconds(5), _session); + _refresh->start(); + } vector<HelloPrx> hellos; @@ -144,7 +163,7 @@ SessionClient::run(int argc, char* argv[]) } else if(c == 'c') { - hellos.push_back(session->createHello()); + hellos.push_back(_session->createHello()); cout << "Created hello object " << hellos.size() - 1 << endl; } else if(c == 's') @@ -179,14 +198,7 @@ SessionClient::run(int argc, char* argv[]) // is set to 0 so that if session->destroy() raises an exception // the thread will not be re-terminated and re-joined. // - refresh->terminate(); - refresh->getThreadControl().join(); - refresh = 0; - - if(destroy) - { - session->destroy(); - } + cleanup(destroy); if(shutdown) { factory->shutdown(); @@ -194,14 +206,12 @@ SessionClient::run(int argc, char* argv[]) } catch(...) { - // - // The refresher thread must be terminated in the event of a - // failure. - // - if(refresh) + try + { + cleanup(true); + } + catch(...) { - refresh->terminate(); - refresh->getThreadControl().join(); } throw; } @@ -210,6 +220,50 @@ SessionClient::run(int argc, char* argv[]) } void +SessionClient::interruptCallback(int) +{ + // + // Terminate the refresh thread, destroy the session and then + // destroy the communicator, followed by an exit. We have to call + // exit because main may be blocked in a cin >> s call which + // cannot be interrupted portably. + // + cleanup(true); + + try + { + communicator()->destroy(); + } + catch(const IceUtil::Exception& ex) + { + cerr << appName() << ": " << ex << endl; + } + catch(...) + { + cerr << appName() << ": unknown exception" << endl; + } + exit(EXIT_SUCCESS); +} + +void +SessionClient::cleanup(bool destroy) +{ + IceUtil::Mutex::Lock sync(_mutex); + if(_refresh) + { + _refresh->terminate(); + _refresh->getThreadControl().join(); + _refresh = 0; + } + + if(destroy && _session) + { + _session->destroy(); + _session = 0; + } +} + +void SessionClient::menu() { cout << |