#ifndef _sys_Monitor_h #define _sys_Monitor_h /* * * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance * with the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, * software distributed under the License is distributed on an * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY * KIND, either express or implied. See the License for the * specific language governing permissions and limitations * under the License. * */ #include #include #include #include #ifdef USE_APR # include #endif namespace qpid { namespace sys { /** * A monitor is a condition variable and a mutex */ class Monitor : public Mutex { public: inline Monitor(); inline ~Monitor(); inline void wait(); inline bool wait(const Time& absoluteTime); inline void notify(); inline void notifyAll(); private: #ifdef USE_APR apr_thread_cond_t* condition; #else pthread_cond_t condition; #endif }; // APR ================================================================ #ifdef USE_APR Monitor::Monitor() { CHECK_APR_SUCCESS(apr_thread_cond_create(&condition, APRPool::get())); } Monitor::~Monitor() { CHECK_APR_SUCCESS(apr_thread_cond_destroy(condition)); } void Monitor::wait() { CHECK_APR_SUCCESS(apr_thread_cond_wait(condition, mutex)); } bool Monitor::wait(const Time& absoluteTime){ // APR uses microseconds. apr_status_t status = apr_thread_cond_timedwait(condition, mutex, absoluteTime/TIME_USEC); if(status != APR_TIMEUP) CHECK_APR_SUCCESS(status); return status == 0; } void Monitor::notify(){ CHECK_APR_SUCCESS(apr_thread_cond_signal(condition)); } void Monitor::notifyAll(){ CHECK_APR_SUCCESS(apr_thread_cond_broadcast(condition)); } #else // POSIX ================================================================ Monitor::Monitor() { QPID_POSIX_THROW_IF(pthread_cond_init(&condition, 0)); } Monitor::~Monitor() { QPID_POSIX_THROW_IF(pthread_cond_destroy(&condition)); } void Monitor::wait() { QPID_POSIX_THROW_IF(pthread_cond_wait(&condition, &mutex)); } bool Monitor::wait(const Time& absoluteTime){ struct timespec ts; toTimespec(ts, absoluteTime); int status = pthread_cond_timedwait(&condition, &mutex, &ts); if (status != 0) { if (status == ETIMEDOUT) return false; throw QPID_POSIX_ERROR(status); } return true; } void Monitor::notify(){ QPID_POSIX_THROW_IF(pthread_cond_signal(&condition)); } void Monitor::notifyAll(){ QPID_POSIX_THROW_IF(pthread_cond_broadcast(&condition)); } #endif /*USE_APR*/ }} #endif /*!_sys_Monitor_h*/