Boost.Asio async_handshake无法取消

时间:2016-10-13 16:32:11

标签: c++ sockets ssl boost

当使用Boost Asio SSL流启动async_handshake,然后在握手完成之前实现取消/关闭流的截止时间计时器时,程序将因缓冲区溢出异常而崩溃(在Windows上,我没有& #39;尝试了Linux)。在套接字为close d之后崩溃,并且外部处理程序完成执行,但在run()命令完成之前。

在SSL握手进行过程中,是否存在无法关闭套接字的原因?或者这是Boost Asio中的错误?

class Connection
{
    void connect_handler(const boost::system::error_code &err)
    {
        ...
        ssock->async_handshake(boost::asio::ssl::stream_base::client,
                   boost::bind(&Connection::handle_handshake, this, _1));
        ...
    }

    void handle_handshake(const boost::system::error_code &err)
    {
        // CONNECTED SECURELY
    }

    void handle_timeout()
    {
        // HANDLE SSL SHUTDOWN IF ALREADY CONNECTED...
        ...
        ssock->shutdown(boost::asio::ip::tcp::socket::shutdown_both);
        ssock->close();
        delete ssock;
    }

    ...

    boost::asio::ssl::stream<boost::asio::ip::tcp::socket> *ssock;
};

为了澄清一点,在IO服务调用handle_timeout()之前调用handle_handshake()将在io_service::run()方法中崩溃。

2 个答案:

答案 0 :(得分:0)

问题是在异步处理程序完成之前,socket对象正在被销毁。要取消挂起/排队的处理程序,请使用io_service::stop(),如下所示:

io_service.stop(); // Stop the IO service to prevent handlers from executing
ssock->shutdown(boost::asio::ip::tcp::socket::shutdown_both);
ssock->close();
delete ssock;

所以不,它不是Boost.Asio本身的错误。

答案 1 :(得分:0)

我们也碰到了那个。我的同事(AW)发现了罪魁祸首。握手在后台使用读写操作,这些操作需要取消,并且握手操作需要正确完成(带有operation_aborted错误),然后才能安全地关闭袜子。经过测试的解决方案如下:

void handle_timeout() {
  ...

  // cancel pending read/write operations
  ssock->lowest_layer().cancel();

  // finish shutdown on next tick of event loop
  boost::asio::post([=] {
    ssock->shutdown(boost::asio::ip::tcp::socket::shutdown_both);
    ssock->close();
    delete ssock;
  });
}