lokinet/llarp/ev.cpp

270 lines
5.7 KiB
C++
Raw Normal View History

#include <ev.h>
#include <logic.hpp>
#include <mem.hpp>
#include <string_view.hpp>
#include <stddef.h>
2018-08-10 03:51:38 +00:00
#define EV_TICK_INTERVAL 100
2018-08-15 02:42:33 +00:00
// apparently current Solaris will emulate epoll.
#if __linux__ || __sun__
2018-06-06 12:46:26 +00:00
#include "ev_epoll.hpp"
#elif defined(__FreeBSD__) || defined(__OpenBSD__) || defined(__NetBSD__) \
|| (__APPLE__ && __MACH__)
2018-06-06 12:46:26 +00:00
#include "ev_kqueue.hpp"
#elif defined(_WIN32) || defined(_WIN64) || defined(__NT__)
#include "ev_win32.hpp"
#else
#error No async event loop for your platform, subclass llarp_ev_loop
2018-04-30 13:18:57 +00:00
#endif
void
llarp_ev_loop_alloc(struct llarp_ev_loop **ev)
{
#if __linux__ || __sun__
2018-04-30 13:18:57 +00:00
*ev = new llarp_epoll_loop;
#elif defined(__FreeBSD__) || defined(__OpenBSD__) || defined(__NetBSD__) \
|| (__APPLE__ && __MACH__)
2018-05-29 12:14:50 +00:00
*ev = new llarp_kqueue_loop;
#elif defined(_WIN32) || defined(_WIN64) || defined(__NT__)
*ev = new llarp_win32_loop;
#else
#error no event loop subclass
2018-04-30 13:18:57 +00:00
#endif
2018-05-18 13:17:58 +00:00
(*ev)->init();
2018-11-19 22:45:37 +00:00
(*ev)->_now = llarp::time_now_ms();
2018-01-29 14:27:24 +00:00
}
2018-01-08 13:49:05 +00:00
void
llarp_ev_loop_free(struct llarp_ev_loop **ev)
{
2018-04-30 13:18:57 +00:00
delete *ev;
2018-01-29 14:27:24 +00:00
*ev = nullptr;
}
int
2018-12-10 14:14:55 +00:00
llarp_ev_loop_run(struct llarp_ev_loop *ev, llarp::Logic *logic)
{
2018-08-29 20:40:26 +00:00
while(ev->running())
{
2018-11-19 22:45:37 +00:00
ev->_now = llarp::time_now_ms();
2018-09-21 12:30:57 +00:00
ev->tick(EV_TICK_INTERVAL);
if(ev->running())
2018-12-10 14:14:55 +00:00
logic->tick(ev->_now);
}
return 0;
}
2018-01-29 14:27:24 +00:00
int
llarp_fd_promise_wait_for_value(struct llarp_fd_promise *p)
{
return p->Get();
}
2018-06-06 12:46:26 +00:00
void
llarp_ev_loop_run_single_process(struct llarp_ev_loop *ev,
struct llarp_threadpool *tp,
2018-12-10 14:14:55 +00:00
llarp::Logic *logic)
2018-06-06 12:46:26 +00:00
{
2018-08-10 03:51:38 +00:00
while(ev->running())
2018-06-06 12:46:26 +00:00
{
2018-11-19 22:45:37 +00:00
ev->_now = llarp::time_now_ms();
2018-08-10 03:51:38 +00:00
ev->tick(EV_TICK_INTERVAL);
2018-09-21 12:30:57 +00:00
if(ev->running())
{
2018-12-10 14:14:55 +00:00
logic->tick_async(ev->_now);
2018-09-21 12:30:57 +00:00
llarp_threadpool_tick(tp);
}
2018-06-06 12:46:26 +00:00
}
}
int
2018-05-23 13:49:00 +00:00
llarp_ev_add_udp(struct llarp_ev_loop *ev, struct llarp_udp_io *udp,
const struct sockaddr *src)
{
2018-05-16 16:41:20 +00:00
udp->parent = ev;
2018-05-23 13:49:00 +00:00
if(ev->udp_listen(udp, src))
return 0;
2018-05-16 16:41:20 +00:00
return -1;
2018-01-29 14:27:24 +00:00
}
2018-01-19 16:51:27 +00:00
int
2018-05-22 19:19:06 +00:00
llarp_ev_close_udp(struct llarp_udp_io *udp)
{
if(udp->parent->udp_close(udp))
return 0;
2018-05-16 16:41:20 +00:00
return -1;
2018-01-29 14:27:24 +00:00
}
2018-01-29 14:19:00 +00:00
2018-10-29 16:48:36 +00:00
llarp_time_t
llarp_ev_loop_time_now_ms(struct llarp_ev_loop *loop)
{
2018-11-15 21:47:05 +00:00
if(loop)
return loop->_now;
2018-11-20 12:44:18 +00:00
return llarp::time_now_ms();
2018-10-29 16:48:36 +00:00
}
void
llarp_ev_loop_stop(struct llarp_ev_loop *loop)
2018-05-18 17:10:48 +00:00
{
loop->stop();
2018-05-18 17:10:48 +00:00
}
int
llarp_ev_udp_sendto(struct llarp_udp_io *udp, const sockaddr *to,
2018-11-23 14:37:26 +00:00
llarp_buffer_t buf)
{
2018-11-26 13:30:03 +00:00
auto ret =
static_cast< llarp::ev_io * >(udp->impl)->sendto(to, buf.base, buf.sz);
#ifndef _WIN32
2018-10-26 03:56:48 +00:00
if(ret == -1 && errno != 0)
2018-08-02 04:34:46 +00:00
{
#else
if(ret == -1 && WSAGetLastError())
{
#endif
#ifndef _WIN32
2018-08-02 04:34:46 +00:00
llarp::LogWarn("sendto failed ", strerror(errno));
errno = 0;
}
#else
char ebuf[1024];
int err = WSAGetLastError();
FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM, nullptr, err, LANG_NEUTRAL, ebuf,
1024, nullptr);
llarp::LogWarn("sendto failed: ", ebuf);
WSASetLastError(0);
}
#endif
2018-08-02 04:34:46 +00:00
return ret;
}
2018-08-15 15:36:34 +00:00
bool
llarp_ev_add_tun(struct llarp_ev_loop *loop, struct llarp_tun_io *tun)
{
auto dev = loop->create_tun(tun);
tun->impl = dev;
2018-08-20 19:12:12 +00:00
if(dev)
return loop->add_ev(dev, false);
2018-08-20 19:12:12 +00:00
return false;
2018-08-15 15:36:34 +00:00
}
bool
2018-11-23 14:37:26 +00:00
llarp_tcp_conn_async_write(struct llarp_tcp_conn *conn, llarp_buffer_t buf)
2018-08-15 15:36:34 +00:00
{
2018-10-29 16:48:36 +00:00
llarp::tcp_conn *impl = static_cast< llarp::tcp_conn * >(conn->impl);
2018-10-25 12:39:32 +00:00
if(impl->_shouldClose)
{
llarp::LogError("write on closed connection");
2018-10-25 12:39:32 +00:00
return false;
}
2018-11-23 14:37:26 +00:00
size_t sz = buf.sz;
2018-11-26 13:30:03 +00:00
buf.cur = buf.base;
2018-10-27 18:26:08 +00:00
while(sz > EV_WRITE_BUF_SZ)
2018-10-25 12:09:29 +00:00
{
2018-11-23 14:37:26 +00:00
if(!impl->queue_write(buf.cur, EV_WRITE_BUF_SZ))
2018-10-25 12:09:29 +00:00
return false;
2018-11-23 14:37:26 +00:00
buf.cur += EV_WRITE_BUF_SZ;
2018-10-27 18:26:08 +00:00
sz -= EV_WRITE_BUF_SZ;
2018-10-25 12:09:29 +00:00
}
2018-11-23 14:37:26 +00:00
return impl->queue_write(buf.cur, sz);
}
2018-10-09 12:06:30 +00:00
2018-11-01 12:47:14 +00:00
void
llarp_tcp_async_try_connect(struct llarp_ev_loop *loop,
struct llarp_tcp_connecter *tcp)
{
tcp->loop = loop;
llarp::string_view addr_str, port_str;
// try parsing address
const char *begin = tcp->remote;
const char *ptr = strstr(tcp->remote, ":");
// get end of address
if(ptr == nullptr)
{
llarp::LogError("bad address: ", tcp->remote);
if(tcp->error)
tcp->error(tcp);
return;
}
const char *end = ptr;
2018-11-08 12:31:50 +00:00
while(*end && ((end - begin) < static_cast< ptrdiff_t >(sizeof tcp->remote)))
2018-11-01 12:47:14 +00:00
{
++end;
}
addr_str = llarp::string_view(begin, ptr - begin);
++ptr;
port_str = llarp::string_view(ptr, end - ptr);
// actually parse address
llarp::Addr addr(addr_str, port_str);
if(!loop->tcp_connect(tcp, addr))
2018-11-01 12:47:14 +00:00
{
llarp::LogError("async connect failed");
if(tcp->error)
tcp->error(tcp);
2018-11-01 12:47:14 +00:00
}
}
2018-10-09 12:06:30 +00:00
bool
2018-10-23 11:29:37 +00:00
llarp_tcp_serve(struct llarp_ev_loop *loop, struct llarp_tcp_acceptor *tcp,
const struct sockaddr *bindaddr)
2018-10-09 12:06:30 +00:00
{
tcp->loop = loop;
llarp::ev_io *impl = loop->bind_tcp(tcp, bindaddr);
if(impl)
{
return loop->add_ev(impl, false);
}
2018-10-09 12:06:30 +00:00
return false;
}
2018-10-19 11:41:36 +00:00
void
llarp_tcp_acceptor_close(struct llarp_tcp_acceptor *tcp)
{
llarp::ev_io *impl = static_cast< llarp::ev_io * >(tcp->user);
tcp->impl = nullptr;
tcp->loop->close_ev(impl);
if(tcp->closed)
tcp->closed(tcp);
// dont free acceptor because it may be stack allocated
}
bool
2018-11-23 14:37:26 +00:00
llarp_ev_tun_async_write(struct llarp_tun_io *tun, llarp_buffer_t buf)
{
2018-11-23 14:37:26 +00:00
if(buf.sz > EV_WRITE_BUF_SZ)
2018-10-27 18:26:08 +00:00
{
2018-11-23 14:37:26 +00:00
llarp::LogWarn("packet too big, ", buf.sz, " > ", EV_WRITE_BUF_SZ);
2018-10-25 12:09:29 +00:00
return false;
2018-10-27 18:26:08 +00:00
}
2018-11-26 13:30:03 +00:00
return static_cast< llarp::tun * >(tun->impl)->queue_write(buf.base, buf.sz);
2018-10-19 11:41:36 +00:00
}
2018-10-09 12:06:30 +00:00
void
llarp_tcp_conn_close(struct llarp_tcp_conn *conn)
{
2018-10-25 12:39:32 +00:00
static_cast< llarp::tcp_conn * >(conn->impl)->_shouldClose = true;
2018-10-09 12:06:30 +00:00
}
namespace llarp
{
2018-10-25 12:39:32 +00:00
bool
tcp_conn::tick()
{
if(_shouldClose)
{
2018-11-01 12:47:14 +00:00
if(tcp.closed)
tcp.closed(&tcp);
2018-10-25 12:39:32 +00:00
return false;
}
2018-11-01 12:47:14 +00:00
else if(tcp.tick)
tcp.tick(&tcp);
2018-10-25 12:39:32 +00:00
return true;
}
} // namespace llarp