Fix lifetime issues

This commit is contained in:
pacevedom 2018-12-23 20:33:53 +01:00
parent 855f39b321
commit 39df973976
1 changed files with 59 additions and 46 deletions

View File

@ -61,7 +61,7 @@ namespace server {
template <typename socket_type> template <typename socket_type>
class connection : public std::enable_shared_from_this<connection<socket_type>>, class connection : public std::enable_shared_from_this<connection<socket_type>>,
private boost::noncopyable { private boost::noncopyable {
public: public:
/// Construct a connection with the given io_service. /// Construct a connection with the given io_service.
template <typename... SocketArgs> template <typename... SocketArgs>
explicit connection( explicit connection(
@ -69,10 +69,13 @@ public:
const boost::posix_time::time_duration &tls_handshake_timeout, const boost::posix_time::time_duration &tls_handshake_timeout,
const boost::posix_time::time_duration &read_timeout, const boost::posix_time::time_duration &read_timeout,
SocketArgs &&... args) SocketArgs &&... args)
: socket_(std::forward<SocketArgs>(args)...), mux_(mux), : socket_(std::forward<SocketArgs>(args)...),
mux_(mux),
deadline_(socket_.get_io_service()), deadline_(socket_.get_io_service()),
tls_handshake_timeout_(tls_handshake_timeout), tls_handshake_timeout_(tls_handshake_timeout),
read_timeout_(read_timeout), writing_(false), stopped_(false) {} read_timeout_(read_timeout),
writing_(false),
stopped_(false) {}
/// Start the first asynchronous operation for the connection. /// Start the first asynchronous operation for the connection.
void start() { void start() {
@ -80,11 +83,12 @@ public:
handler_ = std::make_shared<http2_handler>( handler_ = std::make_shared<http2_handler>(
socket_.get_io_service(), socket_.lowest_layer().remote_endpoint(ec), socket_.get_io_service(), socket_.lowest_layer().remote_endpoint(ec),
[ this, self = this->shared_from_this() ]() { [self = this->shared_from_this()]() {
if (!std::weak_ptr<connection>{self}.lock()) { auto ptr = std::weak_ptr<connection>{self}.lock();
if (!ptr) {
return; return;
} }
do_write(); ptr->do_write();
}, },
mux_); mux_);
if (handler_->start() != 0) { if (handler_->start() != 0) {
@ -100,9 +104,11 @@ public:
deadline_.expires_from_now(tls_handshake_timeout_); deadline_.expires_from_now(tls_handshake_timeout_);
deadline_.async_wait([self = this->shared_from_this()]( deadline_.async_wait([self = this->shared_from_this()](
const boost::system::error_code &) { const boost::system::error_code &) {
if (!std::weak_ptr<connection>{self}.lock()) auto ptr = std::weak_ptr<connection>{self}.lock();
if (!ptr) {
return; return;
return self->handle_deadline(); }
return ptr->handle_deadline();
}); });
} }
@ -110,9 +116,11 @@ public:
deadline_.expires_from_now(read_timeout_); deadline_.expires_from_now(read_timeout_);
deadline_.async_wait([self = this->shared_from_this()]( deadline_.async_wait([self = this->shared_from_this()](
const boost::system::error_code &) { const boost::system::error_code &) {
if (!std::weak_ptr<connection>{self}.lock()) auto ptr = std::weak_ptr<connection>{self}.lock();
if (!ptr) {
return; return;
return self->handle_deadline(); }
return ptr->handle_deadline();
}); });
} }
@ -130,46 +138,50 @@ public:
deadline_.async_wait([self = this->shared_from_this()]( deadline_.async_wait([self = this->shared_from_this()](
const boost::system::error_code &) { const boost::system::error_code &) {
if (!std::weak_ptr<connection>{self}.lock()) auto ptr = std::weak_ptr<connection>{self}.lock();
if (!ptr) {
return; return;
return self->handle_deadline(); }
return ptr->handle_deadline();
}); });
} }
void do_read() { void do_read() {
deadline_.expires_from_now(read_timeout_); deadline_.expires_from_now(read_timeout_);
socket_.async_read_some(boost::asio::buffer(buffer_), [ socket_.async_read_some(
this, self = this->shared_from_this() boost::asio::buffer(buffer_),
](const boost::system::error_code &e, std::size_t bytes_transferred) { [this, self=this->shared_from_this()](const boost::system::error_code &e,
if (!std::weak_ptr<connection>{self}.lock()) { std::size_t bytes_transferred) {
return; auto ptr = std::weak_ptr<connection>{self}.lock();
} if (!ptr) {
if (e) { return;
stop(); }
return; if (e) {
} ptr->stop();
return;
}
if (handler_->on_read(buffer_, bytes_transferred) != 0) { if (handler_->on_read(buffer_, bytes_transferred) != 0) {
stop(); ptr->stop();
return; return;
} }
do_write(); ptr->do_write();
if (!writing_ && handler_->should_stop()) { if (!writing_ && handler_->should_stop()) {
stop(); stop();
return; return;
} }
do_read(); ptr->do_read();
// If an error occurs then no new asynchronous operations are // If an error occurs then no new asynchronous operations are
// started. This means that all shared_ptr references to the // started. This means that all shared_ptr references to the
// connection object will disappear and the object will be // connection object will disappear and the object will be
// destroyed automatically after this handler returns. The // destroyed automatically after this handler returns. The
// connection class's destructor closes the socket. // connection class's destructor closes the socket.
}); });
} }
void do_write() { void do_write() {
@ -203,17 +215,18 @@ public:
boost::asio::async_write(socket_, boost::asio::buffer(outbuf_, nwrite), [ boost::asio::async_write(socket_, boost::asio::buffer(outbuf_, nwrite), [
this, self = this->shared_from_this() this, self = this->shared_from_this()
](const boost::system::error_code &e, std::size_t) { ](const boost::system::error_code &e, std::size_t) {
if (!std::weak_ptr<connection>{self}.lock()) { auto ptr = std::weak_ptr<connection>{self}.lock();
if (!ptr) {
return; return;
} }
if (e) { if (e) {
stop(); ptr->stop();
return; return;
} }
writing_ = false; writing_ = false;
do_write(); ptr->do_write();
}); });
// No new asynchronous operations are started. This means that all // No new asynchronous operations are started. This means that all
@ -233,7 +246,7 @@ public:
deadline_.cancel(); deadline_.cancel();
} }
private: private:
socket_type socket_; socket_type socket_;
serve_mux &mux_; serve_mux &mux_;
@ -253,10 +266,10 @@ private:
bool stopped_; bool stopped_;
}; };
} // namespace server } // namespace server
} // namespace asio_http2 } // namespace asio_http2
} // namespace nghttp2 } // namespace nghttp2
#endif // ASIO_SERVER_CONNECTION_H #endif // ASIO_SERVER_CONNECTION_H