| /* | 
 |  * Copyright (C) 2007 The Android Open Source Project | 
 |  * | 
 |  * Licensed 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. | 
 |  */ | 
 |  | 
 | #ifndef _LIBS_UTILS_CONDITION_H | 
 | #define _LIBS_UTILS_CONDITION_H | 
 |  | 
 | #include <stdint.h> | 
 | #include <sys/types.h> | 
 | #include <time.h> | 
 |  | 
 | #if !defined(_WIN32) | 
 | # include <pthread.h> | 
 | #endif | 
 |  | 
 | #include <utils/Errors.h> | 
 | #include <utils/Mutex.h> | 
 | #include <utils/Timers.h> | 
 |  | 
 | // --------------------------------------------------------------------------- | 
 | namespace android { | 
 | // --------------------------------------------------------------------------- | 
 |  | 
 | /* | 
 |  * Condition variable class.  The implementation is system-dependent. | 
 |  * | 
 |  * Condition variables are paired up with mutexes.  Lock the mutex, | 
 |  * call wait(), then either re-wait() if things aren't quite what you want, | 
 |  * or unlock the mutex and continue.  All threads calling wait() must | 
 |  * use the same mutex for a given Condition. | 
 |  */ | 
 | class Condition { | 
 | public: | 
 |     enum { | 
 |         PRIVATE = 0, | 
 |         SHARED = 1 | 
 |     }; | 
 |  | 
 |     enum WakeUpType { | 
 |         WAKE_UP_ONE = 0, | 
 |         WAKE_UP_ALL = 1 | 
 |     }; | 
 |  | 
 |     Condition(); | 
 |     Condition(int type); | 
 |     ~Condition(); | 
 |     // Wait on the condition variable.  Lock the mutex before calling. | 
 |     status_t wait(Mutex& mutex); | 
 |     // same with relative timeout | 
 |     status_t waitRelative(Mutex& mutex, nsecs_t reltime); | 
 |     // Signal the condition variable, allowing exactly one thread to continue. | 
 |     void signal(); | 
 |     // Signal the condition variable, allowing one or all threads to continue. | 
 |     void signal(WakeUpType type) { | 
 |         if (type == WAKE_UP_ONE) { | 
 |             signal(); | 
 |         } else { | 
 |             broadcast(); | 
 |         } | 
 |     } | 
 |     // Signal the condition variable, allowing all threads to continue. | 
 |     void broadcast(); | 
 |  | 
 | private: | 
 | #if !defined(_WIN32) | 
 |     pthread_cond_t mCond; | 
 | #else | 
 |     void*   mState; | 
 | #endif | 
 | }; | 
 |  | 
 | // --------------------------------------------------------------------------- | 
 |  | 
 | #if !defined(_WIN32) | 
 |  | 
 | inline Condition::Condition() { | 
 |     pthread_cond_init(&mCond, NULL); | 
 | } | 
 | inline Condition::Condition(int type) { | 
 |     if (type == SHARED) { | 
 |         pthread_condattr_t attr; | 
 |         pthread_condattr_init(&attr); | 
 |         pthread_condattr_setpshared(&attr, PTHREAD_PROCESS_SHARED); | 
 |         pthread_cond_init(&mCond, &attr); | 
 |         pthread_condattr_destroy(&attr); | 
 |     } else { | 
 |         pthread_cond_init(&mCond, NULL); | 
 |     } | 
 | } | 
 | inline Condition::~Condition() { | 
 |     pthread_cond_destroy(&mCond); | 
 | } | 
 | inline status_t Condition::wait(Mutex& mutex) { | 
 |     return -pthread_cond_wait(&mCond, &mutex.mMutex); | 
 | } | 
 | inline status_t Condition::waitRelative(Mutex& mutex, nsecs_t reltime) { | 
 | #if defined(HAVE_PTHREAD_COND_TIMEDWAIT_RELATIVE) | 
 |     struct timespec ts; | 
 |     ts.tv_sec  = reltime/1000000000; | 
 |     ts.tv_nsec = reltime%1000000000; | 
 |     return -pthread_cond_timedwait_relative_np(&mCond, &mutex.mMutex, &ts); | 
 | #else // HAVE_PTHREAD_COND_TIMEDWAIT_RELATIVE | 
 |     struct timespec ts; | 
 | #if defined(__linux__) | 
 |     clock_gettime(CLOCK_REALTIME, &ts); | 
 | #else // __APPLE__ | 
 |     // we don't support the clocks here. | 
 |     struct timeval t; | 
 |     gettimeofday(&t, NULL); | 
 |     ts.tv_sec = t.tv_sec; | 
 |     ts.tv_nsec= t.tv_usec*1000; | 
 | #endif | 
 |     ts.tv_sec += reltime/1000000000; | 
 |     ts.tv_nsec+= reltime%1000000000; | 
 |     if (ts.tv_nsec >= 1000000000) { | 
 |         ts.tv_nsec -= 1000000000; | 
 |         ts.tv_sec  += 1; | 
 |     } | 
 |     return -pthread_cond_timedwait(&mCond, &mutex.mMutex, &ts); | 
 | #endif // HAVE_PTHREAD_COND_TIMEDWAIT_RELATIVE | 
 | } | 
 | inline void Condition::signal() { | 
 |     /* | 
 |      * POSIX says pthread_cond_signal wakes up "one or more" waiting threads. | 
 |      * However bionic follows the glibc guarantee which wakes up "exactly one" | 
 |      * waiting thread. | 
 |      * | 
 |      * man 3 pthread_cond_signal | 
 |      *   pthread_cond_signal restarts one of the threads that are waiting on | 
 |      *   the condition variable cond. If no threads are waiting on cond, | 
 |      *   nothing happens. If several threads are waiting on cond, exactly one | 
 |      *   is restarted, but it is not specified which. | 
 |      */ | 
 |     pthread_cond_signal(&mCond); | 
 | } | 
 | inline void Condition::broadcast() { | 
 |     pthread_cond_broadcast(&mCond); | 
 | } | 
 |  | 
 | #endif // !defined(_WIN32) | 
 |  | 
 | // --------------------------------------------------------------------------- | 
 | }; // namespace android | 
 | // --------------------------------------------------------------------------- | 
 |  | 
 | #endif // _LIBS_UTILS_CONDITON_H |