254 lines
6.9 KiB
Plaintext
254 lines
6.9 KiB
Plaintext
|
//////////////////////////////////////////////////////////////////////////////
|
||
|
//
|
||
|
// (C) Copyright Ion Gaztanaga 2005-2012. Distributed under the Boost
|
||
|
// Software License, Version 1.0. (See accompanying file
|
||
|
// LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
||
|
//
|
||
|
// See http://www.boost.org/libs/interprocess for documentation.
|
||
|
//
|
||
|
//////////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
#ifndef BOOST_INTERPROCESS_POSIX_SEMAPHORE_WRAPPER_HPP
|
||
|
#define BOOST_INTERPROCESS_POSIX_SEMAPHORE_WRAPPER_HPP
|
||
|
|
||
|
#ifndef BOOST_CONFIG_HPP
|
||
|
# include <boost/config.hpp>
|
||
|
#endif
|
||
|
#
|
||
|
#if defined(BOOST_HAS_PRAGMA_ONCE)
|
||
|
# pragma once
|
||
|
#endif
|
||
|
|
||
|
#include <boost/interprocess/detail/posix_time_types_wrk.hpp>
|
||
|
#include <boost/interprocess/exceptions.hpp>
|
||
|
#include <boost/interprocess/creation_tags.hpp>
|
||
|
#include <boost/interprocess/detail/os_file_functions.hpp>
|
||
|
#include <boost/interprocess/detail/shared_dir_helpers.hpp>
|
||
|
#include <boost/interprocess/permissions.hpp>
|
||
|
|
||
|
#include <fcntl.h> //O_CREAT, O_*...
|
||
|
#include <unistd.h> //close
|
||
|
#include <string> //std::string
|
||
|
#include <semaphore.h> //sem_* family, SEM_VALUE_MAX
|
||
|
#include <sys/stat.h> //mode_t, S_IRWXG, S_IRWXO, S_IRWXU,
|
||
|
#include <boost/assert.hpp>
|
||
|
|
||
|
#ifdef SEM_FAILED
|
||
|
#define BOOST_INTERPROCESS_POSIX_SEM_FAILED (reinterpret_cast<sem_t*>(SEM_FAILED))
|
||
|
#else
|
||
|
#define BOOST_INTERPROCESS_POSIX_SEM_FAILED (reinterpret_cast<sem_t*>(-1))
|
||
|
#endif
|
||
|
|
||
|
#ifdef BOOST_INTERPROCESS_POSIX_TIMEOUTS
|
||
|
#include <boost/interprocess/sync/posix/ptime_to_timespec.hpp>
|
||
|
#else
|
||
|
#include <boost/interprocess/detail/os_thread_functions.hpp>
|
||
|
#include <boost/interprocess/sync/detail/locks.hpp>
|
||
|
#include <boost/interprocess/sync/detail/common_algorithms.hpp>
|
||
|
#endif
|
||
|
|
||
|
namespace boost {
|
||
|
namespace interprocess {
|
||
|
namespace ipcdetail {
|
||
|
|
||
|
#ifdef BOOST_INTERPROCESS_POSIX_NAMED_SEMAPHORES
|
||
|
|
||
|
inline bool semaphore_open
|
||
|
(sem_t *&handle, create_enum_t type, const char *origname,
|
||
|
unsigned int count = 0, const permissions &perm = permissions())
|
||
|
{
|
||
|
std::string name;
|
||
|
#ifndef BOOST_INTERPROCESS_FILESYSTEM_BASED_POSIX_SEMAPHORES
|
||
|
add_leading_slash(origname, name);
|
||
|
#else
|
||
|
create_shared_dir_cleaning_old_and_get_filepath(origname, name);
|
||
|
#endif
|
||
|
|
||
|
//Create new mapping
|
||
|
int oflag = 0;
|
||
|
switch(type){
|
||
|
case DoOpen:
|
||
|
{
|
||
|
//No addition
|
||
|
handle = ::sem_open(name.c_str(), oflag);
|
||
|
}
|
||
|
break;
|
||
|
case DoOpenOrCreate:
|
||
|
case DoCreate:
|
||
|
{
|
||
|
while(1){
|
||
|
oflag = (O_CREAT | O_EXCL);
|
||
|
handle = ::sem_open(name.c_str(), oflag, perm.get_permissions(), count);
|
||
|
if(handle != BOOST_INTERPROCESS_POSIX_SEM_FAILED){
|
||
|
//We can't change semaphore permissions!
|
||
|
//::fchmod(handle, perm.get_permissions());
|
||
|
break;
|
||
|
}
|
||
|
else if(errno == EEXIST && type == DoOpenOrCreate){
|
||
|
oflag = 0;
|
||
|
if( (handle = ::sem_open(name.c_str(), oflag)) != BOOST_INTERPROCESS_POSIX_SEM_FAILED
|
||
|
|| (errno != ENOENT) ){
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
else{
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
default:
|
||
|
{
|
||
|
error_info err(other_error);
|
||
|
throw interprocess_exception(err);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//Check for error
|
||
|
if(handle == BOOST_INTERPROCESS_POSIX_SEM_FAILED){
|
||
|
throw interprocess_exception(error_info(errno));
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
inline void semaphore_close(sem_t *handle)
|
||
|
{
|
||
|
int ret = sem_close(handle);
|
||
|
if(ret != 0){
|
||
|
BOOST_ASSERT(0);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
inline bool semaphore_unlink(const char *semname)
|
||
|
{
|
||
|
try{
|
||
|
std::string sem_str;
|
||
|
#ifndef BOOST_INTERPROCESS_FILESYSTEM_BASED_POSIX_SEMAPHORES
|
||
|
add_leading_slash(semname, sem_str);
|
||
|
#else
|
||
|
shared_filepath(semname, sem_str);
|
||
|
#endif
|
||
|
return 0 == sem_unlink(sem_str.c_str());
|
||
|
}
|
||
|
catch(...){
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#endif //BOOST_INTERPROCESS_POSIX_NAMED_SEMAPHORES
|
||
|
|
||
|
#ifdef BOOST_INTERPROCESS_POSIX_UNNAMED_SEMAPHORES
|
||
|
|
||
|
inline void semaphore_init(sem_t *handle, unsigned int initialCount)
|
||
|
{
|
||
|
int ret = sem_init(handle, 1, initialCount);
|
||
|
//According to SUSV3 version 2003 edition, the return value of a successful
|
||
|
//sem_init call is not defined, but -1 is returned on failure.
|
||
|
//In the future, a successful call might be required to return 0.
|
||
|
if(ret == -1){
|
||
|
error_info err = system_error_code();
|
||
|
throw interprocess_exception(err);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
inline void semaphore_destroy(sem_t *handle)
|
||
|
{
|
||
|
int ret = sem_destroy(handle);
|
||
|
if(ret != 0){
|
||
|
BOOST_ASSERT(0);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#endif //BOOST_INTERPROCESS_POSIX_UNNAMED_SEMAPHORES
|
||
|
|
||
|
inline void semaphore_post(sem_t *handle)
|
||
|
{
|
||
|
int ret = sem_post(handle);
|
||
|
if(ret != 0){
|
||
|
error_info err = system_error_code();
|
||
|
throw interprocess_exception(err);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
inline void semaphore_wait(sem_t *handle)
|
||
|
{
|
||
|
int ret = sem_wait(handle);
|
||
|
if(ret != 0){
|
||
|
error_info err = system_error_code();
|
||
|
throw interprocess_exception(err);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
inline bool semaphore_try_wait(sem_t *handle)
|
||
|
{
|
||
|
int res = sem_trywait(handle);
|
||
|
if(res == 0)
|
||
|
return true;
|
||
|
if(system_error_code() == EAGAIN){
|
||
|
return false;
|
||
|
}
|
||
|
error_info err = system_error_code();
|
||
|
throw interprocess_exception(err);
|
||
|
}
|
||
|
|
||
|
#ifndef BOOST_INTERPROCESS_POSIX_TIMEOUTS
|
||
|
|
||
|
struct semaphore_wrapper_try_wrapper
|
||
|
{
|
||
|
explicit semaphore_wrapper_try_wrapper(sem_t *handle)
|
||
|
: m_handle(handle)
|
||
|
{}
|
||
|
|
||
|
void wait()
|
||
|
{ semaphore_wait(m_handle); }
|
||
|
|
||
|
bool try_wait()
|
||
|
{ return semaphore_try_wait(m_handle); }
|
||
|
|
||
|
private:
|
||
|
sem_t *m_handle;
|
||
|
};
|
||
|
|
||
|
#endif
|
||
|
|
||
|
inline bool semaphore_timed_wait(sem_t *handle, const boost::posix_time::ptime &abs_time)
|
||
|
{
|
||
|
#ifdef BOOST_INTERPROCESS_POSIX_TIMEOUTS
|
||
|
//Posix does not support infinity absolute time so handle it here
|
||
|
if(abs_time == boost::posix_time::pos_infin){
|
||
|
semaphore_wait(handle);
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
timespec tspec = ptime_to_timespec(abs_time);
|
||
|
for (;;){
|
||
|
int res = sem_timedwait(handle, &tspec);
|
||
|
if(res == 0)
|
||
|
return true;
|
||
|
if (res > 0){
|
||
|
//buggy glibc, copy the returned error code to errno
|
||
|
errno = res;
|
||
|
}
|
||
|
if(system_error_code() == ETIMEDOUT){
|
||
|
return false;
|
||
|
}
|
||
|
error_info err = system_error_code();
|
||
|
throw interprocess_exception(err);
|
||
|
}
|
||
|
return false;
|
||
|
#else //#ifdef BOOST_INTERPROCESS_POSIX_TIMEOUTS
|
||
|
|
||
|
semaphore_wrapper_try_wrapper swtw(handle);
|
||
|
ipcdetail::lock_to_wait<semaphore_wrapper_try_wrapper> lw(swtw);
|
||
|
return ipcdetail::try_based_timed_lock(lw, abs_time);
|
||
|
|
||
|
#endif //#ifdef BOOST_INTERPROCESS_POSIX_TIMEOUTS
|
||
|
}
|
||
|
|
||
|
} //namespace ipcdetail {
|
||
|
} //namespace interprocess {
|
||
|
} //namespace boost {
|
||
|
|
||
|
#endif //#ifndef BOOST_INTERPROCESS_POSIX_SEMAPHORE_WRAPPER_HPP
|