Potential fix for inability to cancel Boost.Asio blocking accept on Linux

This commit is contained in:
athile 2010-07-06 10:03:49 -07:00
parent 5fe4313b95
commit 22c1314738
2 changed files with 39 additions and 8 deletions

View file

@ -1,10 +1,8 @@
#include "server.hpp" #include "server.hpp"
#include "libs/platform/strings.h"
using boost::asio::ip::tcp; using boost::asio::ip::tcp;
#include <libs/mangle/tools/str_exception.hpp>
// //
// Namespace for containing implementation details that the // Namespace for containing implementation details that the
// rest of OpenMW doesn't need to worry about // rest of OpenMW doesn't need to worry about
@ -105,7 +103,7 @@ namespace OMW { namespace CommandServer { namespace Detail {
bDone = true; bDone = true;
} }
else else
throw str_exception("Unexpected header!"); throw std::runtime_error("Unexpected header!");
} }
else else
bDone = true; bDone = true;
@ -120,21 +118,54 @@ namespace OMW { namespace CommandServer {
using namespace Detail; using namespace Detail;
Server::Server (Deque* pCommandQueue, const int port) Server::Server (Deque* pCommandQueue, const int port)
: mAcceptor (mIOService, tcp::endpoint(tcp::v4(), port)) : mPort (port)
, mAcceptor (mIOService, tcp::endpoint(tcp::v4(), mPort))
, mbStopping (false) , mbStopping (false)
, mpCommandQueue (pCommandQueue) , mpCommandQueue (pCommandQueue)
{ {
} }
void Server::start() void Server::start()
{ {
mIOService.run(); mIOService.run();
mpThread = new boost::thread(boost::bind(&Server::threadMain, this)); mpThread = new boost::thread(boost::bind(&Server::threadMain, this));
} }
//
// Helper function - see Server::stop()
//
static void connectAndDisconnect (int port)
{
char portString[64];
snprintf(portString, 64, "%d", port);
boost::asio::io_service ioService;
tcp::resolver resolver(ioService);
tcp::resolver::query query("localhost", portString);
tcp::resolver::iterator endpoint_iterator = resolver.resolve(query);
tcp::resolver::iterator end;
tcp::socket socket(ioService);
boost::system::error_code error = boost::asio::error::host_not_found;
while (error && endpoint_iterator != end)
{
socket.close();
socket.connect(*endpoint_iterator++, error);
}
socket.close();
ioService.stop();
}
void Server::stop() void Server::stop()
{ {
// Boost.Asio doesn't have a way to cancel the blocking accept() call
// in the listening thread. Therefore, set an internal flag to let
// the server know it's stopping and then unblock it via a do-nothing
// connect/disconnect.
mbStopping = true;
connectAndDisconnect(mPort);
// (1) Stop accepting new connections // (1) Stop accepting new connections
// (2) Wait for the listener thread to finish // (2) Wait for the listener thread to finish
mAcceptor.close(); mAcceptor.close();
@ -144,7 +175,6 @@ namespace OMW { namespace CommandServer {
// open connections // open connections
{ {
boost::mutex::scoped_lock lock(mConnectionsMutex); boost::mutex::scoped_lock lock(mConnectionsMutex);
mbStopping = true;
for (ConnectionSet::iterator it = mConnections.begin(); for (ConnectionSet::iterator it = mConnections.begin();
it != mConnections.end(); it != mConnections.end();
++it) ++it)
@ -186,7 +216,7 @@ namespace OMW { namespace CommandServer {
std::auto_ptr<Connection> spConnection(new Connection(mAcceptor.io_service(), this)); std::auto_ptr<Connection> spConnection(new Connection(mAcceptor.io_service(), this));
boost::system::error_code ec; boost::system::error_code ec;
mAcceptor.accept(spConnection->socket(), ec); mAcceptor.accept(spConnection->socket(), ec);
if (!ec) if (!ec && !mbStopping)
{ {
boost::mutex::scoped_lock lock(mConnectionsMutex); boost::mutex::scoped_lock lock(mConnectionsMutex);
mConnections.insert(spConnection.get()); mConnections.insert(spConnection.get());

View file

@ -45,6 +45,7 @@ namespace OMW { namespace CommandServer
void threadMain(); void threadMain();
// Objects used to set up the listening server // Objects used to set up the listening server
int mPort;
boost::asio::io_service mIOService; boost::asio::io_service mIOService;
boost::asio::ip::tcp::acceptor mAcceptor; boost::asio::ip::tcp::acceptor mAcceptor;
boost::thread* mpThread; boost::thread* mpThread;