повысить рекурсию asio вызывает segfault в deadline_timer - PullRequest
0 голосов
/ 17 января 2019

Я программирую UDP-клиент. Протокол записывается в виде одного ответа на один запрос. У меня есть класс транзакции, который отправляет сообщение, а затем запускает таймер для ожидания ответа. Если ответ получен в ожидаемое время, таймер останавливается и вызывается успешный обратный вызов. В противном случае вызывается ошибка обратного вызова. Наконец, обратный вызов завершения вызывается безоговорочно в обоих случаях.

template <typename RequestType, typename ResponseType>
struct transaction{
    boost::asio::io_service&                _service;
    boost::asio::deadline_timer             _timer;
    typename ResponseType::buffer_type      _buff;
    RequestType                             _request;

    transaction(boost::asio::io_service& service, const RequestType& req): _service(service), _timer(service), _request(req){}
    virtual ~transaction(){}
    template <typename SuccessCallback, typename FailureCallback, typename FinishCallback>
    void exec(boost::asio::ip::udp::endpoint& endpoint, boost::asio::ip::udp::socket& socket, unsigned wait_seconds, SuccessCallback success, FailureCallback failure, FinishCallback finish){
        _timer.expires_from_now(boost::posix_time::seconds(wait_seconds));
        _timer.async_wait([this, &failure, &finish](const boost::system::error_code& error){
            if(!error){
                std::cout << "failed to receive reply within deadline" << std::endl;
                failure(_request);
                finish(this);
            }
        });
        socket.async_receive_from(boost::asio::buffer(_buff), endpoint, [this, &success, finish](const boost::system::error_code& error, std::size_t bytes_transferred){
            if(!error && bytes_transferred > 0){
                ResponseType reply;
                reply.parse(_buff);
                _timer.cancel();
                success(reply);
                finish(this);
            }
        });
        socket.send_to(boost::asio::buffer(_request.buffer()), endpoint);
    }
};

транзакция работает нормально на этот раз. Однако я хочу повторить попытку, если это не удастся. Ниже приводится функция детонации, которая работает нормально в первый раз. Но при сбое обратного вызова он снова вызывает knock, что приводит к segfault.

typedef transaction<request_knock, reply_knock> transaction_knock;
void knock(boost::asio::io_service& service, boost::asio::ip::udp::endpoint& endpoint, boost::asio::ip::udp::socket& socket){
    request_knock knock_request(0);
    transaction_knock* knock_transaction = new transaction_knock(service, knock_request);
    knock_transaction->exec(endpoint, socket, 5, [](const reply_knock& reply){
        std::cout << "success " << reply._data.id << std::endl;
    }, [&service, &endpoint, &socket](const request_knock& req){
        // FAILURE
        knock(service, endpoint, socket);
    }, [](transaction_knock* transaction){
        delete transaction;
    });
}

Ниже приводится обратный след

#0  0x00007ffff7f0f244 in pthread_mutex_lock () from /usr/lib/libpthread.so.0
#1  0x000055555559f3c8 in boost::asio::detail::posix_mutex::lock (this=0xfbad2a8c) at /usr/include/boost/asio/detail/posix_mutex.hpp:52
#2  0x00005555555a7cb4 in boost::asio::detail::scoped_lock<boost::asio::detail::posix_mutex>::scoped_lock (this=0x7fffffffe2a0, m=...)
    at /usr/include/boost/asio/detail/scoped_lock.hpp:46
#3  0x00005555555a02f5 in boost::asio::detail::service_registry::do_use_service (this=0xfbad2a84, key=..., 
    factory=0x5555555b4666 <boost::asio::detail::service_registry::create<boost::asio::detail::deadline_timer_service<boost::asio::time_traits<boost::posix_time::ptime> >, boost::asio::io_context>(void*)>, owner=0x7ffff7bbe5c0 <_IO_2_1_stdout_>)
    at /usr/include/boost/asio/detail/impl/service_registry.ipp:117
#4  0x00005555555b39da in boost::asio::detail::service_registry::use_service<boost::asio::detail::deadline_timer_service<boost::asio::time_traits<boost::posix_time::ptime> > > (this=0xfbad2a84, owner=...) at /usr/include/boost/asio/detail/impl/service_registry.hpp:39
#5  0x00005555555b2219 in boost::asio::use_service<boost::asio::detail::deadline_timer_service<boost::asio::time_traits<boost::posix_time::ptime> > > (ioc=...) at /usr/include/boost/asio/impl/io_context.hpp:39
#6  0x00005555555b021d in boost::asio::basic_io_object<boost::asio::detail::deadline_timer_service<boost::asio::time_traits<boost::posix_time::ptime> >, true>::basic_io_object (this=0x5555555e2f20, io_context=...) at /usr/include/boost/asio/basic_io_object.hpp:224
#7  0x00005555555ad795 in boost::asio::basic_deadline_timer<boost::posix_time::ptime, boost::asio::time_traits<boost::posix_time::ptime> >::basic_deadline_timer (this=0x5555555e2f20, io_context=...) at /usr/include/boost/asio/basic_deadline_timer.hpp:159
#8  0x00005555555aa4ab in transaction<request_knock, reply_knock>::transaction (this=0x5555555e2f10, service=..., req=...)
    at /home/else/Projects/comet/main.cpp:111
#9  0x000055555559a90c in knock (service=..., endpoint=..., socket=...) at /home/else/Projects/comet/main.cpp:192
#10 0x000055555559a8a2 in <lambda(const request_knock&)>::operator()(const request_knock &) const (__closure=0x7fffffffe3d0, req=...)

Я пытаюсь выяснить причину этого сбоя?

...