2012-08-29 00:32:15 +04:00
|
|
|
#include "event_manager.hh"
|
|
|
|
|
2014-12-03 16:56:02 +03:00
|
|
|
#include <unistd.h>
|
2012-08-29 00:32:15 +04:00
|
|
|
|
|
|
|
namespace Kakoune
|
|
|
|
{
|
|
|
|
|
2013-01-10 21:54:40 +04:00
|
|
|
FDWatcher::FDWatcher(int fd, Callback callback)
|
|
|
|
: m_fd{fd}, m_callback{std::move(callback)}
|
2012-12-03 21:49:09 +04:00
|
|
|
{
|
2013-01-31 21:58:25 +04:00
|
|
|
EventManager::instance().m_fd_watchers.insert(this);
|
2012-12-03 21:49:09 +04:00
|
|
|
}
|
|
|
|
|
2013-01-10 21:54:40 +04:00
|
|
|
FDWatcher::~FDWatcher()
|
2012-08-29 00:32:15 +04:00
|
|
|
{
|
2013-01-31 21:58:25 +04:00
|
|
|
EventManager::instance().m_fd_watchers.erase(this);
|
2012-08-29 00:32:15 +04:00
|
|
|
}
|
|
|
|
|
2014-11-25 04:00:18 +03:00
|
|
|
void FDWatcher::run(EventMode mode)
|
|
|
|
{
|
|
|
|
m_callback(*this, mode);
|
|
|
|
}
|
|
|
|
|
2014-12-03 16:56:02 +03:00
|
|
|
void FDWatcher::close_fd()
|
|
|
|
{
|
|
|
|
close(m_fd);
|
|
|
|
m_fd = -1;
|
|
|
|
}
|
|
|
|
|
2014-11-25 04:00:18 +03:00
|
|
|
Timer::Timer(TimePoint date, Callback callback, EventMode mode)
|
|
|
|
: m_date{date}, m_callback{std::move(callback)}, m_mode(mode)
|
2013-01-14 22:07:38 +04:00
|
|
|
{
|
2014-08-15 02:51:24 +04:00
|
|
|
if (EventManager::has_instance())
|
|
|
|
EventManager::instance().m_timers.insert(this);
|
2013-01-14 22:07:38 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
Timer::~Timer()
|
|
|
|
{
|
2014-08-15 02:51:24 +04:00
|
|
|
if (EventManager::has_instance())
|
|
|
|
EventManager::instance().m_timers.erase(this);
|
2013-01-14 22:07:38 +04:00
|
|
|
}
|
|
|
|
|
2014-11-25 04:00:18 +03:00
|
|
|
void Timer::run(EventMode mode)
|
2013-01-14 22:07:38 +04:00
|
|
|
{
|
2014-11-29 23:14:52 +03:00
|
|
|
if (mode == m_mode)
|
2014-11-25 04:00:18 +03:00
|
|
|
{
|
|
|
|
m_date = TimePoint::max();
|
|
|
|
m_callback(*this);
|
|
|
|
}
|
|
|
|
else // try again a little later
|
|
|
|
m_date = Clock::now() + std::chrono::milliseconds{10};
|
2013-01-14 22:07:38 +04:00
|
|
|
}
|
|
|
|
|
2013-01-10 21:54:40 +04:00
|
|
|
EventManager::EventManager()
|
2012-08-29 00:32:15 +04:00
|
|
|
{
|
2014-12-03 16:56:02 +03:00
|
|
|
FD_ZERO(&m_forced_fd);
|
2013-01-10 21:54:40 +04:00
|
|
|
}
|
2012-11-27 16:57:03 +04:00
|
|
|
|
2013-01-10 21:54:40 +04:00
|
|
|
EventManager::~EventManager()
|
|
|
|
{
|
2013-04-09 22:04:11 +04:00
|
|
|
kak_assert(m_fd_watchers.empty());
|
|
|
|
kak_assert(m_timers.empty());
|
2012-08-29 00:32:15 +04:00
|
|
|
}
|
|
|
|
|
2014-11-25 04:00:18 +03:00
|
|
|
void EventManager::handle_next_events(EventMode mode)
|
2012-08-29 00:32:15 +04:00
|
|
|
{
|
2014-12-03 16:56:02 +03:00
|
|
|
int max_fd = 0;
|
|
|
|
fd_set rfds;
|
|
|
|
FD_ZERO(&rfds);
|
2013-01-10 21:54:40 +04:00
|
|
|
for (auto& watcher : m_fd_watchers)
|
2014-12-03 16:56:02 +03:00
|
|
|
{
|
|
|
|
const int fd = watcher->fd();
|
|
|
|
if (fd != -1)
|
|
|
|
{
|
|
|
|
max_fd = std::max(fd, max_fd);
|
|
|
|
FD_SET(fd, &rfds);
|
|
|
|
}
|
|
|
|
}
|
2014-01-30 16:41:34 +04:00
|
|
|
|
2014-12-04 23:01:34 +03:00
|
|
|
timeval tv{};
|
|
|
|
if (not m_timers.empty())
|
2014-01-30 16:41:34 +04:00
|
|
|
{
|
2014-12-04 23:01:34 +03:00
|
|
|
auto next = std::min_element(
|
|
|
|
m_timers.begin(), m_timers.end(), [](Timer* lhs, Timer* rhs) {
|
|
|
|
return lhs->next_date() < rhs->next_date();
|
|
|
|
});
|
|
|
|
using namespace std::chrono; using us = std::chrono::microseconds;
|
|
|
|
auto usecs = std::max(us(0), duration_cast<us>((*next)->next_date() - Clock::now()));
|
|
|
|
auto secs = duration_cast<seconds>(usecs);
|
|
|
|
tv = timeval{ (time_t)secs.count(), (suseconds_t)(usecs - secs).count() };
|
2014-01-30 16:41:34 +04:00
|
|
|
}
|
2014-12-04 23:01:34 +03:00
|
|
|
int res = select(max_fd + 1, &rfds, nullptr, nullptr,
|
|
|
|
m_timers.empty() ? nullptr : &tv);
|
2014-08-17 18:58:42 +04:00
|
|
|
|
2014-12-03 16:56:02 +03:00
|
|
|
// copy forced fds *after* poll, so that signal handlers can write to
|
2014-08-17 18:58:42 +04:00
|
|
|
// m_forced_fd, interupt poll, and directly be serviced.
|
2014-12-03 16:56:02 +03:00
|
|
|
fd_set forced = m_forced_fd;
|
|
|
|
FD_ZERO(&m_forced_fd);
|
2014-08-17 18:58:42 +04:00
|
|
|
|
2014-12-03 16:56:02 +03:00
|
|
|
for (int fd = 0; fd < max_fd + 1; ++fd)
|
2012-11-26 17:08:27 +04:00
|
|
|
{
|
2014-12-03 16:56:02 +03:00
|
|
|
if ((res > 0 and FD_ISSET(fd, &rfds)) or FD_ISSET(fd, &forced))
|
2013-01-10 21:54:40 +04:00
|
|
|
{
|
2013-01-11 17:28:13 +04:00
|
|
|
auto it = find_if(m_fd_watchers,
|
2014-12-03 16:56:02 +03:00
|
|
|
[fd](const FDWatcher* w){return w->fd() == fd; });
|
2013-01-10 21:54:40 +04:00
|
|
|
if (it != m_fd_watchers.end())
|
2014-11-25 04:00:18 +03:00
|
|
|
(*it)->run(mode);
|
2013-01-10 21:54:40 +04:00
|
|
|
}
|
2012-11-26 17:08:27 +04:00
|
|
|
}
|
2013-01-14 22:07:38 +04:00
|
|
|
|
|
|
|
TimePoint now = Clock::now();
|
|
|
|
for (auto& timer : m_timers)
|
|
|
|
{
|
|
|
|
if (timer->next_date() <= now)
|
2014-11-25 04:00:18 +03:00
|
|
|
timer->run(mode);
|
2013-01-14 22:07:38 +04:00
|
|
|
}
|
2012-08-29 00:32:15 +04:00
|
|
|
}
|
|
|
|
|
2012-10-27 17:01:13 +04:00
|
|
|
void EventManager::force_signal(int fd)
|
|
|
|
{
|
2014-12-03 16:56:02 +03:00
|
|
|
FD_SET(fd, &m_forced_fd);
|
2013-01-10 21:54:40 +04:00
|
|
|
}
|
2012-10-27 17:01:13 +04:00
|
|
|
|
2012-08-29 00:32:15 +04:00
|
|
|
}
|