/* * Carla semaphore utils * Copyright (C) 2013-2015 Filipe Coelho * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License as * published by the Free Software Foundation; either version 2 of * the License, or any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * For a full copy of the GNU General Public License see the doc/GPL.txt file. */ #ifndef CARLA_SEM_UTILS_HPP_INCLUDED #define CARLA_SEM_UTILS_HPP_INCLUDED #include "CarlaUtils.hpp" #include #ifdef CARLA_OS_WIN struct carla_sem_t { HANDLE handle; }; #elif defined(CARLA_OS_MAC) // TODO struct carla_sem_t { char dummy; }; #else # include # include # include struct carla_sem_t { sem_t sem; }; #endif /* * Create a new semaphore, pre-allocated version. */ static inline bool carla_sem_create2(carla_sem_t& sem) noexcept { #if defined(CARLA_OS_WIN) SECURITY_ATTRIBUTES sa; carla_zeroStruct(sa); sa.nLength = sizeof(SECURITY_ATTRIBUTES); sa.bInheritHandle = TRUE; sem.handle = ::CreateSemaphore(&sa, 0, 1, nullptr); return (sem.handle != INVALID_HANDLE_VALUE); #elif defined(CARLA_OS_MAC) return false; // TODO #else return (::sem_init(&sem.sem, 1, 0) == 0); #endif } /* * Create a new semaphore. */ static inline carla_sem_t* carla_sem_create() noexcept { if (carla_sem_t* const sem = (carla_sem_t*)std::malloc(sizeof(carla_sem_t))) { if (carla_sem_create2(*sem)) return sem; std::free(sem); } return nullptr; } /* * Destroy a semaphore, pre-allocated version. */ static inline void carla_sem_destroy2(carla_sem_t& sem) noexcept { #if defined(CARLA_OS_WIN) ::CloseHandle(sem.handle); #elif defined(CARLA_OS_MAC) // TODO #else ::sem_destroy(&sem.sem); #endif } /* * Destroy a semaphore. */ static inline void carla_sem_destroy(carla_sem_t* const sem) noexcept { CARLA_SAFE_ASSERT_RETURN(sem != nullptr,); carla_sem_destroy2(*sem); std::free(sem); } /* * Post semaphore (unlock). */ static inline void carla_sem_post(carla_sem_t& sem) noexcept { #ifdef CARLA_OS_WIN ::ReleaseSemaphore(sem.handle, 1, nullptr); #elif defined(CARLA_OS_MAC) // TODO #else ::sem_post(&sem.sem); #endif } /* * Wait for a semaphore (lock). */ static inline bool carla_sem_timedwait(carla_sem_t& sem, const uint secs) noexcept { CARLA_SAFE_ASSERT_RETURN(secs > 0, false); #if defined(CARLA_OS_WIN) return (::WaitForSingleObject(sem.handle, secs*1000) == WAIT_OBJECT_0); #elif defined(CARLA_OS_MAC) // TODO #else if (::sem_trywait(&sem.sem) == 0) return true; # if 1 /* NOTE: * This is very ugly! * While trying to fix 32bit-on-64bit issues I noticed this somehow gets a signal, * while that timedwait does not. * I'll keep this code for the time being, but only for TESTING! * It should revert to the proper code (below) when we know more info about the bug. */ timespec timeout, now, step; step.tv_sec = 0; step.tv_nsec = 2; ::clock_gettime(CLOCK_REALTIME, &timeout); timeout.tv_sec += static_cast(secs); for (;;) { ::nanosleep(&step, NULL); ::clock_gettime(CLOCK_REALTIME, &now); if (now.tv_sec >= timeout.tv_sec) return false; if (::sem_trywait(&sem.sem) == 0) return true; } //timeval now; //::gettimeofday(&now, nullptr); //timeout.tv_sec = now.tv_sec; //timeout.tv_nsec = now.tv_usec * 1000; # else timespec timeout; ::clock_gettime(CLOCK_REALTIME, &timeout); timeout.tv_sec += static_cast(secs); try { return (::sem_timedwait(&sem.sem, &timeout) == 0); } CARLA_SAFE_EXCEPTION_RETURN("sem_timedwait", false); # endif #endif } // ----------------------------------------------------------------------- #endif // CARLA_SEM_UTILS_HPP_INCLUDED