| /* |
| * 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. |
| */ |
| |
| #ifndef _THRIFT_CONCURRENCY_MUTEX_H_ |
| #define _THRIFT_CONCURRENCY_MUTEX_H_ 1 |
| |
| #include <memory> |
| #include <boost/noncopyable.hpp> |
| #include <stdint.h> |
| |
| namespace apache { |
| namespace thrift { |
| namespace concurrency { |
| |
| #ifndef THRIFT_NO_CONTENTION_PROFILING |
| |
| /** |
| * Determines if the Thrift Mutex and ReadWriteMutex classes will attempt to |
| * profile their blocking acquire methods. If this value is set to non-zero, |
| * Thrift will attempt to invoke the callback once every profilingSampleRate |
| * times. However, as the sampling is not synchronized the rate is not |
| * guranateed, and could be subject to big bursts and swings. Please ensure |
| * your sampling callback is as performant as your application requires. |
| * |
| * The callback will get called with the wait time taken to lock the mutex in |
| * usec and a (void*) that uniquely identifies the Mutex (or ReadWriteMutex) |
| * being locked. |
| * |
| * The enableMutexProfiling() function is unsynchronized; calling this function |
| * while profiling is already enabled may result in race conditions. On |
| * architectures where a pointer assignment is atomic, this is safe but there |
| * is no guarantee threads will agree on a single callback within any |
| * particular time period. |
| */ |
| typedef void (*MutexWaitCallback)(const void* id, int64_t waitTimeMicros); |
| void enableMutexProfiling(int32_t profilingSampleRate, MutexWaitCallback callback); |
| |
| #endif |
| |
| /** |
| * NOTE: All mutex implementations throw an exception on failure. See each |
| * specific implementation to understand the exception type(s) used. |
| */ |
| |
| /** |
| * A simple mutex class |
| * |
| * @version $Id:$ |
| */ |
| class Mutex { |
| public: |
| typedef void (*Initializer)(void*); |
| |
| Mutex(Initializer init = DEFAULT_INITIALIZER); |
| virtual ~Mutex() {} |
| |
| virtual void lock() const; |
| virtual bool trylock() const; |
| virtual bool timedlock(int64_t milliseconds) const; |
| virtual void unlock() const; |
| |
| void* getUnderlyingImpl() const; |
| |
| // If you attempt to use one of these and it fails to link, it means |
| // your version of pthreads does not support it - try another one. |
| static void ADAPTIVE_INITIALIZER(void*); |
| static void DEFAULT_INITIALIZER(void*); |
| static void ERRORCHECK_INITIALIZER(void*); |
| static void RECURSIVE_INITIALIZER(void*); |
| |
| private: |
| class impl; |
| std::shared_ptr<impl> impl_; |
| }; |
| |
| class ReadWriteMutex { |
| public: |
| ReadWriteMutex(); |
| virtual ~ReadWriteMutex() {} |
| |
| // these get the lock and block until it is done successfully |
| virtual void acquireRead() const; |
| virtual void acquireWrite() const; |
| |
| // these attempt to get the lock, returning false immediately if they fail |
| virtual bool attemptRead() const; |
| virtual bool attemptWrite() const; |
| |
| // this releases both read and write locks |
| virtual void release() const; |
| |
| private: |
| class impl; |
| std::shared_ptr<impl> impl_; |
| }; |
| |
| /** |
| * A ReadWriteMutex that guarantees writers will not be starved by readers: |
| * When a writer attempts to acquire the mutex, all new readers will be |
| * blocked from acquiring the mutex until the writer has acquired and |
| * released it. In some operating systems, this may already be guaranteed |
| * by a regular ReadWriteMutex. |
| */ |
| class NoStarveReadWriteMutex : public ReadWriteMutex { |
| public: |
| NoStarveReadWriteMutex(); |
| |
| virtual void acquireRead() const; |
| virtual void acquireWrite() const; |
| |
| private: |
| Mutex mutex_; |
| mutable volatile bool writerWaiting_; |
| }; |
| |
| class Guard : boost::noncopyable { |
| public: |
| Guard(const Mutex& value, int64_t timeout = 0) : mutex_(&value) { |
| if (timeout == 0) { |
| value.lock(); |
| } else if (timeout < 0) { |
| if (!value.trylock()) { |
| mutex_ = NULL; |
| } |
| } else { |
| if (!value.timedlock(timeout)) { |
| mutex_ = NULL; |
| } |
| } |
| } |
| ~Guard() { |
| if (mutex_) { |
| mutex_->unlock(); |
| } |
| } |
| |
| operator bool() const { return (mutex_ != NULL); } |
| |
| private: |
| const Mutex* mutex_; |
| }; |
| |
| // Can be used as second argument to RWGuard to make code more readable |
| // as to whether we're doing acquireRead() or acquireWrite(). |
| enum RWGuardType { RW_READ = 0, RW_WRITE = 1 }; |
| |
| class RWGuard : boost::noncopyable { |
| public: |
| RWGuard(const ReadWriteMutex& value, bool write = false) : rw_mutex_(value) { |
| if (write) { |
| rw_mutex_.acquireWrite(); |
| } else { |
| rw_mutex_.acquireRead(); |
| } |
| } |
| |
| RWGuard(const ReadWriteMutex& value, RWGuardType type) : rw_mutex_(value) { |
| if (type == RW_WRITE) { |
| rw_mutex_.acquireWrite(); |
| } else { |
| rw_mutex_.acquireRead(); |
| } |
| } |
| ~RWGuard() { rw_mutex_.release(); } |
| |
| private: |
| const ReadWriteMutex& rw_mutex_; |
| }; |
| } |
| } |
| } // apache::thrift::concurrency |
| |
| #endif // #ifndef _THRIFT_CONCURRENCY_MUTEX_H_ |