#if defined(__linux__) || defined(__APPLE__) #include "platform.h" #include "utils.h" #include #include #include #include #include #include #include #include #include #include #include #include #include /* For O_* constants */ #include /* For mode constants */ #include #include #include /* For mode constants */ #include /* For O_* constants */ struct PlatformMutexLinux : public PlatformMutex { sem_t* sem_ = nullptr; PlatformMutexLinux(const std::string& name) { std::cerr << "PlatformMutexLinux name=" << name << std::endl; sem_ = sem_open(name.c_str(), O_CREAT, 0666 /*permission*/, 1 /*initial_value*/); } ~PlatformMutexLinux() override { sem_close(sem_); } }; struct PlatformScopedMutexLockLinux : public PlatformScopedMutexLock { sem_t* sem_ = nullptr; PlatformScopedMutexLockLinux(sem_t* sem) : sem_(sem) { sem_wait(sem_); } ~PlatformScopedMutexLockLinux() override { sem_post(sem_); } }; void* checked(void* result, const char* expr) { if (!result) { std::cerr << "FAIL errno=" << errno << " in |" << expr << "|" << std::endl; exit(1); } return result; } int checked(int result, const char* expr) { if (result == -1) { std::cerr << "FAIL errno=" << errno << " in |" << expr << "|" << std::endl; exit(1); } return result; } #define CHECKED(expr) checked(expr, #expr) struct PlatformSharedMemoryLinux : public PlatformSharedMemory { std::string name_; size_t size_; int fd_; PlatformSharedMemoryLinux(const std::string& name, size_t size) : name_(name), size_(size) { std::cerr << "PlatformSharedMemoryLinux name=" << name << ", size=" << size << std::endl; // Try to create shared memory but only if it does not already exist. Since // we created the memory, we need to initialize it. fd_ = shm_open(name_.c_str(), O_RDWR | O_CREAT | O_EXCL, S_IRUSR | S_IWUSR); if (fd_ >= 0) { std::cerr << "Calling ftruncate fd_=" << fd_ << std::endl; CHECKED(ftruncate(fd_, size)); } // Otherwise, we just open existing shared memory. We don't need to // create or initialize it. else { fd_ = CHECKED(shm_open( name_.c_str(), O_RDWR, /* memory is read/write, create if needed */ S_IRUSR | S_IWUSR /* user read/write */)); } // Map the shared memory to an address. data = CHECKED(mmap(nullptr /*kernel assigned starting address*/, size, PROT_READ | PROT_WRITE, MAP_SHARED, fd_, 0 /*offset*/)); capacity = size; std::cout << "Open shared memory name=" << name << ", fd=" << fd_ << ", shared=" << data << ", capacity=" << capacity << std::endl; } ~PlatformSharedMemoryLinux() override { CHECKED(munmap(data, size_)); CHECKED(shm_unlink(name_.c_str())); data = nullptr; } }; #undef CHECKED std::unique_ptr CreatePlatformMutex(const std::string& name) { std::string name2 = "/" + name; return MakeUnique(name2); } std::unique_ptr CreatePlatformScopedMutexLock( PlatformMutex* mutex) { return MakeUnique( static_cast(mutex)->sem_); } void PlatformInit() { } std::unique_ptr CreatePlatformSharedMemory( const std::string& name, size_t size) { std::string name2 = "/" + name; return MakeUnique(name2, size); } #endif