Перевел пример простого сервера на stackful coroutines (пока без обработки ошибок и завершения соединения)

This commit is contained in:
Selim Mustafaev 2014-11-17 23:44:42 +03:00
parent efe4b8f50f
commit 99f137ca25
6 changed files with 55 additions and 105 deletions

View File

@ -38,7 +38,7 @@ endif()
ExternalProject_Add(
boost
URL http://switch.dl.sourceforge.net/project/boost/boost/${boost_version}/boost_${boost_version_underscored}.zip
CONFIGURE_COMMAND ./bootstrap.sh --with-libraries=coroutine
CONFIGURE_COMMAND ./bootstrap.sh --with-libraries=coroutine,thread
BUILD_COMMAND ./b2 --ignore-site-config --stagedir=${CMAKE_SOURCE_DIR} link=static threading=multi cxxflags=-std=c++11 ${boost_build_type}
BUILD_IN_SOURCE 1
INSTALL_COMMAND ""
@ -62,7 +62,7 @@ project(${TEST_SERVER_PROJECT_NAME} CXX)
aux_source_directory(${SRC_DIR}/server TEST_SERVER_SRC)
add_executable(${TEST_SERVER_PROJECT_NAME} ${TEST_SERVER_SRC})
add_dependencies(${TEST_SERVER_PROJECT_NAME} boost cpputil)
target_link_libraries(${TEST_SERVER_PROJECT_NAME} cpputil ${CMAKE_SOURCE_DIR}/lib/libboost_system.a ${CMAKE_SOURCE_DIR}/lib/libboost_coroutine.a)
target_link_libraries(${TEST_SERVER_PROJECT_NAME} cpputil ${CMAKE_SOURCE_DIR}/lib/libboost_system.a ${CMAKE_SOURCE_DIR}/lib/libboost_thread.a ${CMAKE_SOURCE_DIR}/lib/libboost_coroutine.a ${CMAKE_SOURCE_DIR}/lib/libboost_context.a)
install(TARGETS ${TEST_PROJECT_NAME}
RUNTIME DESTINATION bin

View File

@ -1,10 +1,35 @@
#include "server.h"
#include "session.h"
#include <boost/asio/spawn.hpp>
#include <iostream>
using tcp = boost::asio::ip::tcp;
void start_server(std::string address, std::string port)
{
boost::asio::io_service io_service;
tcp::resolver resolver(io_service);
tcp::resolver::query query(address, port);
tcp::acceptor acceptor(io_service, *resolver.resolve(query));
boost::asio::spawn(io_service, [&](boost::asio::yield_context yield){
for (;;)
{
session* s = new session(io_service);
acceptor.async_accept(s->socket(), yield);
s->start();
}
});
io_service.run();
}
int main()
{
server srv("0.0.0.0", "12345");
srv.run();
// server srv("0.0.0.0", "12345");
// srv.run();
start_server("0.0.0.0", "12345");
return 0;
}

View File

@ -1,41 +0,0 @@
#include "server.h"
#include <boost/asio/yield.hpp>
#include <iostream>
server::server(const std::string& address, const std::string& port)
{
tcp::resolver resolver(_io_service);
tcp::resolver::query query(address, port);
_acceptor = new tcp::acceptor(_io_service, *resolver.resolve(query));
}
void server::run()
{
accept();
_io_service.run();
}
void server::accept(boost::system::error_code ec)
{
if(!ec)
{
reenter(this) for(;;)
{
_session = new session(_io_service);
yield _acceptor->async_accept(_session->_socket, std::bind(&server::accept, this, std::placeholders::_1));
std::cout << "new connection started" << std::endl;
_session->start();
}
}
else
{
std::cout << "error: " << ec.message() << std::endl;
}
}
server::~server() {
if(_socket) delete _socket;
if(_acceptor) delete _acceptor;
}
#include <boost/asio/unyield.hpp>

View File

@ -1,29 +0,0 @@
#ifndef _SERVER_H_
#define _SERVER_H_
#include "session.h"
#include <boost/asio.hpp>
class server: boost::asio::coroutine
{
private:
typedef boost::asio::ip::tcp tcp;
private:
boost::asio::io_service _io_service;
tcp::acceptor* _acceptor;
tcp::socket* _socket;
session* _session;
public:
server() = delete ;
server(server& srv) = delete;
server(const std::string& address, const std::string& port);
void run();
~server();
private:
void accept(boost::system::error_code ec = boost::system::error_code());
};
#endif // _SERVER_H_

View File

@ -1,37 +1,29 @@
#include "session.h"
#include <iostream>
#include <boost/asio/yield.hpp>
#include <boost/asio/spawn.hpp>
session::session(boost::asio::io_service &io_service): boost::asio::coroutine(), _socket(io_service)
{
}
void session::start(boost::system::error_code ec, std::size_t bytes_transferred)
void session::start()
{
if(!ec)
boost::asio::spawn(_socket.get_io_service(), [this](boost::asio::yield_context yield){
for(;;)
{
reenter(this) for(;;) // псевдолинейный код
{
std::memset(_data, 0, 128);
yield _socket.async_read_some(boost::asio::buffer(_data, 128), std::bind(&session::start, this, std::placeholders::_1, std::placeholders::_2));
std::size_t bytes = _socket.async_read_some(boost::asio::buffer(_data, 128), yield);
yield _fut = call_async<int>([this]{
int n = async_call<int>([this]{
int n = std::atoi(_data);
return n*n;
});
}, yield);
yield {
std::string res = std::to_string(_fut.get()) + "\n";
_socket.async_write_some(boost::asio::buffer(res), std::bind(&session::start, this, std::placeholders::_1, std::placeholders::_2));
}
}
}
else
{
std::cout << "error: " << ec.message() << std::endl;
delete this;
_socket.async_write_some(boost::asio::buffer(std::to_string(n) + "\n"), yield);
}
});
}
#include <boost/asio/unyield.hpp>
boost::asio::ip::tcp::socket& session::socket()
{
return _socket;
}

View File

@ -8,7 +8,6 @@ class session: boost::asio::coroutine
{
private:
typedef boost::asio::ip::tcp tcp;
friend class server;
private:
tcp::socket _socket;
@ -22,17 +21,21 @@ public:
session() = delete;
session(session&) = delete;
session(boost::asio::io_service& io_service);
void start(boost::system::error_code ec = boost::system::error_code(), std::size_t bytes_transferred = 0);
void start();
template <typename R> std::future<R> call_async(std::function<R()> func)
template <typename R, typename CompletionToken> R async_call(std::function<R()> func, CompletionToken&& token)
{
return _pool.add_task<R>([this, &func]{
R result = func();
_socket.get_io_service().dispatch(std::bind(&session::start, this, boost::system::error_code(), 0));
return result;
typename boost::asio::handler_type<CompletionToken, void(R)>::type handler(std::forward<CompletionToken>(token));
boost::asio::async_result<decltype(handler)> result(handler);
_pool.add_task<void>([this, handler, &func]{
_socket.get_io_service().dispatch(std::bind(handler, func()));
});
return result.get();
}
boost::asio::ip::tcp::socket& socket();
};
#endif // _SESSION_H_