forked from mirror/openmw-tes3mp
Potential fix for inability to cancel Boost.Asio blocking accept on Linux
This commit is contained in:
parent
5fe4313b95
commit
22c1314738
2 changed files with 39 additions and 8 deletions
|
@ -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());
|
||||||
|
|
|
@ -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;
|
||||||
|
|
Loading…
Reference in a new issue