mirror of https://github.com/commaai/cereal.git
237 lines
6.3 KiB
C++
237 lines
6.3 KiB
C++
#include <cassert>
|
|
#include <cstring>
|
|
#include <cstdlib>
|
|
#include <iostream>
|
|
#include <string>
|
|
#include <exception>
|
|
#include <filesystem>
|
|
|
|
#include <unistd.h>
|
|
#include <poll.h>
|
|
#include <signal.h>
|
|
#include <fcntl.h>
|
|
#include <sys/mman.h>
|
|
#include <sys/stat.h>
|
|
|
|
#include "msgq/event.h"
|
|
|
|
#ifndef __APPLE__
|
|
#include <sys/eventfd.h>
|
|
|
|
void event_state_shm_mmap(std::string endpoint, std::string identifier, char **shm_mem, std::string *shm_path) {
|
|
const char* op_prefix = std::getenv("OPENPILOT_PREFIX");
|
|
|
|
std::string full_path = "/dev/shm/";
|
|
if (op_prefix) {
|
|
full_path += std::string(op_prefix) + "/";
|
|
}
|
|
full_path += CEREAL_EVENTS_PREFIX + "/";
|
|
if (identifier.size() > 0) {
|
|
full_path += identifier + "/";
|
|
}
|
|
std::filesystem::create_directories(full_path);
|
|
full_path += endpoint;
|
|
|
|
int shm_fd = open(full_path.c_str(), O_RDWR | O_CREAT, 0664);
|
|
if (shm_fd < 0) {
|
|
throw std::runtime_error("Could not open shared memory file.");
|
|
}
|
|
|
|
int rc = ftruncate(shm_fd, sizeof(EventState));
|
|
if (rc < 0){
|
|
close(shm_fd);
|
|
throw std::runtime_error("Could not truncate shared memory file.");
|
|
}
|
|
|
|
char * mem = (char*)mmap(NULL, sizeof(EventState), PROT_READ | PROT_WRITE, MAP_SHARED, shm_fd, 0);
|
|
close(shm_fd);
|
|
if (mem == nullptr) {
|
|
throw std::runtime_error("Could not map shared memory file.");
|
|
}
|
|
|
|
if (shm_mem != nullptr)
|
|
*shm_mem = mem;
|
|
if (shm_path != nullptr)
|
|
*shm_path = full_path;
|
|
}
|
|
|
|
SocketEventHandle::SocketEventHandle(std::string endpoint, std::string identifier, bool override) {
|
|
char *mem;
|
|
event_state_shm_mmap(endpoint, identifier, &mem, &this->shm_path);
|
|
|
|
this->state = (EventState*)mem;
|
|
if (override) {
|
|
this->state->fds[0] = eventfd(0, EFD_NONBLOCK);
|
|
this->state->fds[1] = eventfd(0, EFD_NONBLOCK);
|
|
}
|
|
}
|
|
|
|
SocketEventHandle::~SocketEventHandle() {
|
|
close(this->state->fds[0]);
|
|
close(this->state->fds[1]);
|
|
munmap(this->state, sizeof(EventState));
|
|
unlink(this->shm_path.c_str());
|
|
}
|
|
|
|
bool SocketEventHandle::is_enabled() {
|
|
return this->state->enabled;
|
|
}
|
|
|
|
void SocketEventHandle::set_enabled(bool enabled) {
|
|
this->state->enabled = enabled;
|
|
}
|
|
|
|
Event SocketEventHandle::recv_called() {
|
|
return Event(this->state->fds[0]);
|
|
}
|
|
|
|
Event SocketEventHandle::recv_ready() {
|
|
return Event(this->state->fds[1]);
|
|
}
|
|
|
|
void SocketEventHandle::toggle_fake_events(bool enabled) {
|
|
if (enabled)
|
|
setenv("CEREAL_FAKE", "1", true);
|
|
else
|
|
unsetenv("CEREAL_FAKE");
|
|
}
|
|
|
|
void SocketEventHandle::set_fake_prefix(std::string prefix) {
|
|
if (prefix.size() == 0) {
|
|
unsetenv("CEREAL_FAKE_PREFIX");
|
|
} else {
|
|
setenv("CEREAL_FAKE_PREFIX", prefix.c_str(), true);
|
|
}
|
|
}
|
|
|
|
std::string SocketEventHandle::fake_prefix() {
|
|
const char* prefix = std::getenv("CEREAL_FAKE_PREFIX");
|
|
if (prefix == nullptr) {
|
|
return "";
|
|
} else {
|
|
return std::string(prefix);
|
|
}
|
|
}
|
|
|
|
Event::Event(int fd): event_fd(fd) {}
|
|
|
|
void Event::set() const {
|
|
throw_if_invalid();
|
|
|
|
uint64_t val = 1;
|
|
size_t count = write(this->event_fd, &val, sizeof(uint64_t));
|
|
assert(count == sizeof(uint64_t));
|
|
}
|
|
|
|
int Event::clear() const {
|
|
throw_if_invalid();
|
|
|
|
uint64_t val = 0;
|
|
// read the eventfd to clear it
|
|
read(this->event_fd, &val, sizeof(uint64_t));
|
|
|
|
return val;
|
|
}
|
|
|
|
void Event::wait(int timeout_sec) const {
|
|
throw_if_invalid();
|
|
|
|
int event_count;
|
|
struct pollfd fds = { this->event_fd, POLLIN, 0 };
|
|
struct timespec timeout = { timeout_sec, 0 };;
|
|
|
|
sigset_t signals;
|
|
sigfillset(&signals);
|
|
sigdelset(&signals, SIGALRM);
|
|
sigdelset(&signals, SIGINT);
|
|
sigdelset(&signals, SIGTERM);
|
|
sigdelset(&signals, SIGQUIT);
|
|
|
|
event_count = ppoll(&fds, 1, timeout_sec < 0 ? nullptr : &timeout, &signals);
|
|
|
|
if (event_count == 0) {
|
|
throw std::runtime_error("Event timed out pid: " + std::to_string(getpid()));
|
|
} else if (event_count < 0) {
|
|
throw std::runtime_error("Event poll failed, errno: " + std::to_string(errno) + " pid: " + std::to_string(getpid()));
|
|
}
|
|
}
|
|
|
|
bool Event::peek() const {
|
|
throw_if_invalid();
|
|
|
|
int event_count;
|
|
|
|
struct pollfd fds = { this->event_fd, POLLIN, 0 };
|
|
|
|
// poll with timeout zero to return status immediately
|
|
event_count = poll(&fds, 1, 0);
|
|
|
|
return event_count != 0;
|
|
}
|
|
|
|
bool Event::is_valid() const {
|
|
return event_fd != -1;
|
|
}
|
|
|
|
int Event::fd() const {
|
|
return event_fd;
|
|
}
|
|
|
|
int Event::wait_for_one(const std::vector<Event>& events, int timeout_sec) {
|
|
struct pollfd fds[events.size()];
|
|
for (size_t i = 0; i < events.size(); i++) {
|
|
fds[i] = { events[i].fd(), POLLIN, 0 };
|
|
}
|
|
|
|
struct timespec timeout = { timeout_sec, 0 };
|
|
|
|
sigset_t signals;
|
|
sigfillset(&signals);
|
|
sigdelset(&signals, SIGALRM);
|
|
sigdelset(&signals, SIGINT);
|
|
sigdelset(&signals, SIGTERM);
|
|
sigdelset(&signals, SIGQUIT);
|
|
|
|
int event_count = ppoll(fds, events.size(), timeout_sec < 0 ? nullptr : &timeout, &signals);
|
|
|
|
if (event_count == 0) {
|
|
throw std::runtime_error("Event timed out pid: " + std::to_string(getpid()));
|
|
} else if (event_count < 0) {
|
|
throw std::runtime_error("Event poll failed, errno: " + std::to_string(errno) + " pid: " + std::to_string(getpid()));
|
|
}
|
|
|
|
for (size_t i = 0; i < events.size(); i++) {
|
|
if (fds[i].revents & POLLIN) {
|
|
return i;
|
|
}
|
|
}
|
|
|
|
throw std::runtime_error("Event poll failed, no events ready");
|
|
}
|
|
#else
|
|
// Stub implementation for Darwin, which does not support eventfd
|
|
void event_state_shm_mmap(std::string endpoint, std::string identifier, char **shm_mem, std::string *shm_path) {}
|
|
|
|
SocketEventHandle::SocketEventHandle(std::string endpoint, std::string identifier, bool override) {
|
|
std::cerr << "SocketEventHandle not supported on macOS" << std::endl;
|
|
assert(false);
|
|
}
|
|
SocketEventHandle::~SocketEventHandle() {}
|
|
bool SocketEventHandle::is_enabled() { return this->state->enabled; }
|
|
void SocketEventHandle::set_enabled(bool enabled) {}
|
|
Event SocketEventHandle::recv_called() { return Event(); }
|
|
Event SocketEventHandle::recv_ready() { return Event(); }
|
|
void SocketEventHandle::toggle_fake_events(bool enabled) {}
|
|
void SocketEventHandle::set_fake_prefix(std::string prefix) {}
|
|
std::string SocketEventHandle::fake_prefix() { return ""; }
|
|
|
|
Event::Event(int fd): event_fd(fd) {}
|
|
void Event::set() const {}
|
|
int Event::clear() const { return 0; }
|
|
void Event::wait(int timeout_sec) const {}
|
|
bool Event::peek() const { return false; }
|
|
bool Event::is_valid() const { return false; }
|
|
int Event::fd() const { return this->event_fd; }
|
|
int Event::wait_for_one(const std::vector<Event>& events, int timeout_sec) { return -1; }
|
|
#endif
|