X-Git-Url: https://main.carlh.net/gitweb/?a=blobdiff_plain;f=src%2Flib%2Fencode_server_finder.cc;h=52c8c89496bdec9b46bb5de1487b06d48907eed8;hb=0b15f493754d1b630338b17f2a6dfc66ccc4f6b7;hp=1234dcd529daa847445f8be7df3bdc2d51d48fbb;hpb=ed0b3ee0c5a0ba11d3a1a1dfee8e71238bcab4bd;p=dcpomatic.git diff --git a/src/lib/encode_server_finder.cc b/src/lib/encode_server_finder.cc index 1234dcd52..52c8c8949 100644 --- a/src/lib/encode_server_finder.cc +++ b/src/lib/encode_server_finder.cc @@ -39,14 +39,13 @@ using std::cout; using boost::shared_ptr; using boost::scoped_array; using boost::weak_ptr; +using boost::optional; using dcp::raw_convert; EncodeServerFinder* EncodeServerFinder::_instance = 0; EncodeServerFinder::EncodeServerFinder () - : _search_thread (0) - , _listen_thread (0) - , _stop (false) + : _stop (false) { Config::instance()->Changed.connect (boost::bind (&EncodeServerFinder::config_changed, this, _1)); } @@ -54,11 +53,11 @@ EncodeServerFinder::EncodeServerFinder () void EncodeServerFinder::start () { - _search_thread = new boost::thread (boost::bind (&EncodeServerFinder::search_thread, this)); - _listen_thread = new boost::thread (boost::bind (&EncodeServerFinder::listen_thread, this)); + _search_thread = boost::thread (boost::bind(&EncodeServerFinder::search_thread, this)); + _listen_thread = boost::thread (boost::bind(&EncodeServerFinder::listen_thread, this)); #ifdef DCPOMATIC_LINUX - pthread_setname_np (_search_thread->native_handle(), "encode-server-search"); - pthread_setname_np (_listen_thread->native_handle(), "encode-server-listen"); + pthread_setname_np (_search_thread.native_handle(), "encode-server-search"); + pthread_setname_np (_listen_thread.native_handle(), "encode-server-listen"); #endif } @@ -74,32 +73,25 @@ EncodeServerFinder::stop () _stop = true; _search_condition.notify_all (); - if (_search_thread) { - /* Ideally this would be a DCPOMATIC_ASSERT(_search_thread->joinable()) but we - can't throw exceptions from a destructor. - */ - if (_search_thread->joinable ()) { - _search_thread->join (); + if (_search_thread.joinable()) { + try { + _search_thread.join(); + } catch (...) { + } } - delete _search_thread; - _search_thread = 0; _listen_io_service.stop (); - if (_listen_thread) { - /* Ideally this would be a DCPOMATIC_ASSERT(_listen_thread->joinable()) but we - can't throw exceptions from a destructor. - */ - if (_listen_thread->joinable ()) { - _listen_thread->join (); + if (_listen_thread.joinable()) { + try { + _listen_thread.join (); + } catch (...) { + } } - delete _listen_thread; - _listen_thread = 0; boost::mutex::scoped_lock lm (_servers_mutex); - _good_servers.clear (); - _bad_servers.clear (); + _servers.clear (); } void @@ -118,6 +110,7 @@ try socket.set_option (boost::asio::socket_base::broadcast (true)); string const data = DCPOMATIC_HELLO; + int const interval = 10; while (!_stop) { if (Config::instance()->use_any_servers ()) { @@ -147,8 +140,31 @@ try } } + /* Discard servers that we haven't seen for a while */ + bool removed = false; + { + boost::mutex::scoped_lock lm (_servers_mutex); + + list::iterator i = _servers.begin(); + while (i != _servers.end()) { + if (i->last_seen_seconds() > 2 * interval) { + list::iterator j = i; + ++j; + _servers.erase (i); + i = j; + removed = true; + } else { + ++i; + } + } + } + + if (removed) { + emit (boost::bind (boost::ref (ServersListChanged))); + } + boost::mutex::scoped_lock lm (_search_condition_mutex); - _search_condition.timed_wait (lm, boost::get_system_time() + boost::posix_time::seconds (10)); + _search_condition.timed_wait (lm, boost::get_system_time() + boost::posix_time::seconds (interval)); } } catch (...) @@ -207,14 +223,14 @@ EncodeServerFinder::handle_accept (boost::system::error_code ec, shared_ptrread_string (s); string const ip = socket->socket().remote_endpoint().address().to_string (); - if (!server_found (ip)) { + optional::iterator> found = server_found (ip); + if (found) { + (*found)->set_seen (); + } else { EncodeServerDescription sd (ip, xml->number_child("Threads"), xml->optional_number_child("Version").get_value_or(0)); - if (sd.link_version() == SERVER_LINK_VERSION) { - boost::mutex::scoped_lock lm (_servers_mutex); - _good_servers.push_back (sd); - } else { + { boost::mutex::scoped_lock lm (_servers_mutex); - _bad_servers.push_back (sd); + _servers.push_back (sd); } emit (boost::bind (boost::ref (ServersListChanged))); } @@ -222,25 +238,20 @@ EncodeServerFinder::handle_accept (boost::system::error_code ec, shared_ptr::iterator> +EncodeServerFinder::server_found (string ip) { boost::mutex::scoped_lock lm (_servers_mutex); - list::const_iterator i = _good_servers.begin(); - while (i != _good_servers.end() && i->host_name() != ip) { + list::iterator i = _servers.begin(); + while (i != _servers.end() && i->host_name() != ip) { ++i; } - if (i != _good_servers.end()) { - return true; - } - - i = _bad_servers.begin(); - while (i != _bad_servers.end() && i->host_name() != ip) { - ++i; + if (i != _servers.end()) { + return i; } - return i != _bad_servers.end (); + return optional::iterator>(); } EncodeServerFinder* @@ -262,17 +273,10 @@ EncodeServerFinder::drop () } list -EncodeServerFinder::good_servers () const -{ - boost::mutex::scoped_lock lm (_servers_mutex); - return _good_servers; -} - -list -EncodeServerFinder::bad_servers () const +EncodeServerFinder::servers () const { boost::mutex::scoped_lock lm (_servers_mutex); - return _bad_servers; + return _servers; } void @@ -281,8 +285,7 @@ EncodeServerFinder::config_changed (Config::Property what) if (what == Config::USE_ANY_SERVERS || what == Config::SERVERS) { { boost::mutex::scoped_lock lm (_servers_mutex); - _good_servers.clear (); - _bad_servers.clear (); + _servers.clear (); } ServersListChanged (); _search_condition.notify_all ();