You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							237 lines
						
					
					
						
							6.3 KiB
						
					
					
				
			
		
		
	
	
							237 lines
						
					
					
						
							6.3 KiB
						
					
					
				| #include <cassert>
 | |
| #include <cstring>
 | |
| #include <cstdlib>
 | |
| #include <iostream>
 | |
| #include <string>
 | |
| #include <exception>
 | |
| #include <filesystem>
 | |
| #include <vector>
 | |
| 
 | |
| #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
 | |
| 
 |