|
- /*
- * Carla semaphore utils
- * Copyright (C) 2013-2014 Filipe Coelho <falktx@falktx.com>
- *
- * 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 <ctime>
-
- #ifdef CARLA_OS_WIN
- struct sem_t { HANDLE handle; };
- #else
- # include <sys/time.h>
- # include <sys/types.h>
- # include <semaphore.h>
- # ifdef CARLA_OS_MAC
- # include <fcntl.h>
- extern "C" {
- # include "osx_sem_timedwait.c"
- };
- # endif
- #endif
-
- /*
- * Create a new semaphore.
- */
- static inline
- sem_t* carla_sem_create() noexcept
- {
- #if defined(CARLA_OS_WIN)
- sem_t* const sem = (sem_t*)std::malloc(sizeof(sem_t));
- CARLA_SAFE_ASSERT_RETURN(sem != nullptr, nullptr);
-
- SECURITY_ATTRIBUTES sa;
- carla_zeroStruct(sa);
- sa.nLength = sizeof(SECURITY_ATTRIBUTES);
- sa.bInheritHandle = TRUE;
-
- sem->handle = ::CreateSemaphore(&sa, 0, 1, nullptr);
-
- return sem;
- #elif defined(CARLA_OS_MAC)
- static ulong sCounter = 0;
-
- if (++sCounter == 1)
- std::srand(static_cast<uint>(std::time(nullptr)));
-
- char strBuf[0xff+1];
- carla_zeroChar(strBuf, 0xff+1);
- std::snprintf(strBuf, 0xff, "carla-sem-%lu-%lu-%i", static_cast<ulong>(::getpid()), sCounter, std::rand());
-
- ::sem_unlink(strBuf);
-
- return ::sem_open(strBuf, O_CREAT, O_RDWR, 0);
- #else
- sem_t sem;
-
- if (::sem_init(&sem, 1, 0) != 0)
- return nullptr;
-
- // can't return temporary variable, so allocate a new one
- if (sem_t* const sem2 = (sem_t*)std::malloc(sizeof(sem_t)))
- {
- std::memcpy(sem2, &sem, sizeof(sem_t));
- return sem2;
- }
-
- ::sem_destroy(&sem);
- return nullptr;
- #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_WIN)
- ::CloseHandle(sem->handle);
- std::free(sem);
- #elif defined(CARLA_OS_MAC)
- ::sem_close(sem);
- #else
- // we can't call "sem_destroy(sem)" directly because it will free memory which we allocated during carla_sem_create()
- // so we create a temp variable, free our memory, and finally pass the temp variable to sem_destroy()
-
- // temp var
- sem_t sem2;
- std::memcpy(&sem2, sem, sizeof(sem_t));
-
- // destroy semaphore
- ::sem_destroy(&sem2);
-
- // free memory allocated in carla_sem_create()
- // FIXME
- //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);
-
- #ifdef CARLA_OS_WIN
- return (::ReleaseSemaphore(sem->handle, 1, nullptr) != FALSE);
- #else
- return (::sem_post(sem) == 0);
- #endif
- }
-
- /*
- * 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);
-
- #if defined(CARLA_OS_WIN)
- const DWORD result = ::WaitForSingleObject(sem->handle, secs*1000);
-
- switch (result)
- {
- case WAIT_OBJECT_0:
- return true;
- case WAIT_TIMEOUT:
- errno = ETIMEDOUT;
- return false;
- default:
- errno = EINTR;
- return false;
- }
- #else
- 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 += static_cast<time_t>(secs);
-
- try {
- return (::sem_timedwait(sem, &timeout) == 0);
- } CARLA_SAFE_EXCEPTION_RETURN("sem_timedwait", false);
- #endif
- }
-
- // -----------------------------------------------------------------------
-
- #endif // CARLA_SEM_UTILS_HPP_INCLUDED
|