/* * Carla semaphore utils * Copyright (C) 2013-2014 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 #include #include #include #if defined(CARLA_OS_MAC) # include # include extern "C" { # include "osx_sem_timedwait.c" }; #endif /* * Create a new semaphore. */ static inline sem_t* carla_sem_create() noexcept { #if defined(CARLA_OS_MAC) static ulong sCounter = 0; ++sCounter; std::srand(static_cast(std::time(nullptr))); char strBuf[0xff+1]; carla_zeroChar(strBuf, 0xff+1); std::snprintf(strBuf, 0xff, "carla-sem-%lu-%lu-%i", static_cast(::getpid()), sCounter, std::rand()); ::sem_unlink(strBuf); return ::sem_open(strBuf, O_CREAT, O_RDWR, 0); #else sem_t* const sem((sem_t*)std::malloc(sizeof(sem_t))); CARLA_SAFE_ASSERT_RETURN(sem != nullptr, nullptr); if (::sem_init(sem, 1, 0) != 0) { std::free(sem); return nullptr; } return sem; #endif } /* * Destroy a semaphore. */ static inline void carla_sem_destroy(sem_t* const sem) noexcept { CARLA_SAFE_ASSERT_RETURN(sem != nullptr,); #if defined(CARLA_OS_MAC) ::sem_close(sem); #else ::sem_destroy(sem); std::free(sem); #endif } /* * Post semaphore (unlock). */ static inline bool carla_sem_post(sem_t* const sem) noexcept { CARLA_SAFE_ASSERT_RETURN(sem != nullptr, false); return (::sem_post(sem) == 0); } /* * Wait for a semaphore (lock). */ static inline bool carla_sem_timedwait(sem_t* const sem, const uint secs) noexcept { CARLA_SAFE_ASSERT_RETURN(sem != nullptr, false); CARLA_SAFE_ASSERT_RETURN(secs > 0, false); timespec timeout; #ifdef CARLA_OS_LINUX ::clock_gettime(CLOCK_REALTIME, &timeout); #else timeval now; ::gettimeofday(&now, nullptr); timeout.tv_sec = now.tv_sec; timeout.tv_nsec = now.tv_usec * 1000; #endif timeout.tv_sec += secs; try { return (::sem_timedwait(sem, &timeout) == 0); } CARLA_SAFE_EXCEPTION_RETURN("sem_timedwait", false); } // ----------------------------------------------------------------------- #endif // CARLA_SEM_UTILS_HPP_INCLUDED