IXWebSocket/ixwebsocket/IXSelectInterruptEventFd.cpp

116 lines
2.8 KiB
C++
Raw Normal View History

2018-11-07 21:24:08 +01:00
/*
2019-03-15 02:37:38 +01:00
* IXSelectInterruptEventFd.cpp
2018-11-07 21:24:08 +01:00
* Author: Benjamin Sergeant
2019-03-15 02:37:38 +01:00
* Copyright (c) 2018-2019 Machine Zone, Inc. All rights reserved.
2018-11-07 21:24:08 +01:00
*/
2019-03-15 02:37:38 +01:00
//
// On Linux we use eventd to wake up select.
//
2018-11-07 21:24:08 +01:00
//
// Linux/Android has a special type of virtual files. select(2) will react
// when reading/writing to those files, unlike closing sockets.
//
// https://linux.die.net/man/2/eventfd
// http://www.sourcexr.com/articles/2013/10/26/lightweight-inter-process-signaling-with-eventfd
//
// eventfd was added in Linux kernel 2.x, and our oldest Android (Kitkat 4.4)
// is on Kernel 3.x
//
// cf Android/Kernel table here
2018-11-07 21:24:08 +01:00
// https://android.stackexchange.com/questions/51651/which-android-runs-which-linux-kernel
//
// On macOS we use UNIX pipes to wake up select.
//
2018-11-07 21:24:08 +01:00
2019-03-15 02:54:47 +01:00
#include "IXSelectInterruptEventFd.h"
2018-11-07 21:24:08 +01:00
2019-03-15 19:43:27 +01:00
#include <assert.h>
2019-09-23 19:25:23 +02:00
#include <errno.h>
#include <fcntl.h>
2019-03-15 02:37:38 +01:00
#include <sstream>
2019-09-23 19:25:23 +02:00
#include <string.h> // for strerror
#include <sys/eventfd.h>
#include <unistd.h> // for write
2018-11-07 21:24:08 +01:00
namespace ix
2018-11-07 21:24:08 +01:00
{
2019-03-15 02:37:38 +01:00
SelectInterruptEventFd::SelectInterruptEventFd()
2018-11-07 21:24:08 +01:00
{
2019-03-15 19:41:57 +01:00
_eventfd = -1;
2018-11-07 21:24:08 +01:00
}
2019-03-15 02:37:38 +01:00
SelectInterruptEventFd::~SelectInterruptEventFd()
2018-11-07 21:24:08 +01:00
{
::close(_eventfd);
}
2019-03-15 02:37:38 +01:00
bool SelectInterruptEventFd::init(std::string& errorMsg)
2018-11-07 21:24:08 +01:00
{
2019-03-15 19:41:57 +01:00
// calling init twice is a programming error
assert(_eventfd == -1);
2019-03-15 02:37:38 +01:00
_eventfd = eventfd(0, 0);
if (_eventfd < 0)
{
std::stringstream ss;
ss << "SelectInterruptEventFd::init() failed in eventfd()"
<< " : " << strerror(errno);
errorMsg = ss.str();
_eventfd = -1;
return false;
}
if (fcntl(_eventfd, F_SETFL, O_NONBLOCK) == -1)
{
std::stringstream ss;
ss << "SelectInterruptEventFd::init() failed in fcntl() call"
<< " : " << strerror(errno);
errorMsg = ss.str();
_eventfd = -1;
return false;
}
return true;
}
bool SelectInterruptEventFd::notify(uint64_t value)
{
int fd = _eventfd;
2018-11-07 21:24:08 +01:00
if (fd == -1) return false;
2018-11-07 21:24:08 +01:00
// we should write 8 bytes for an uint64_t
return write(fd, &value, sizeof(value)) == 8;
}
// TODO: return max uint64_t for errors ?
2019-03-15 02:37:38 +01:00
uint64_t SelectInterruptEventFd::read()
{
2019-03-15 02:37:38 +01:00
int fd = _eventfd;
uint64_t value = 0;
::read(fd, &value, sizeof(value));
return value;
2018-11-07 21:24:08 +01:00
}
2019-03-15 02:37:38 +01:00
bool SelectInterruptEventFd::clear()
2018-11-07 21:24:08 +01:00
{
if (_eventfd == -1) return false;
// 0 is a special value ; select will not wake up
2018-11-07 21:24:08 +01:00
uint64_t value = 0;
// we should write 8 bytes for an uint64_t
return write(_eventfd, &value, sizeof(value)) == 8;
}
2019-03-15 19:41:57 +01:00
int SelectInterruptEventFd::getFd() const
2018-11-07 21:24:08 +01:00
{
return _eventfd;
}
2019-09-23 19:25:23 +02:00
} // namespace ix