blob: 3b8631bc79d22a4757672fc98f74b79d184d7323 [file] [log] [blame]
Mark Slee9f0c6512007-02-28 23:58:26 +00001// Copyright (c) 2006- Facebook
2// Distributed under the Thrift Software License
3//
4// See accompanying file LICENSE or visit the Thrift site at:
5// http://developers.facebook.com/thrift/
6
Mark Sleef5f2be42006-09-05 21:05:31 +00007#ifndef _THRIFT_CONCURRENCY_MUTEX_H_
8#define _THRIFT_CONCURRENCY_MUTEX_H_ 1
Marc Slemko66949872006-07-15 01:52:39 +00009
boz19cee902007-09-22 23:08:11 +000010#include <boost/shared_ptr.hpp>
boz19cee902007-09-22 23:08:11 +000011
David Reiss0c90f6f2008-02-06 22:18:40 +000012namespace facebook { namespace thrift { namespace concurrency {
Marc Slemko66949872006-07-15 01:52:39 +000013
Mark Sleef5f2be42006-09-05 21:05:31 +000014/**
15 * A simple mutex class
16 *
17 * @author marc
18 * @version $Id:$
19 */
Marc Slemko66949872006-07-15 01:52:39 +000020class Mutex {
Marc Slemko66949872006-07-15 01:52:39 +000021 public:
David Reissc6dab612008-06-10 22:55:13 +000022 typedef void (*Initializer)(void*);
23
24 Mutex(Initializer init = DEFAULT_INITIALIZER);
boz19cee902007-09-22 23:08:11 +000025 virtual ~Mutex() {}
Marc Slemko66949872006-07-15 01:52:39 +000026 virtual void lock() const;
boz5362e702007-08-15 20:55:36 +000027 virtual bool trylock() const;
Marc Slemko66949872006-07-15 01:52:39 +000028 virtual void unlock() const;
29
David Reissc6dab612008-06-10 22:55:13 +000030 static void DEFAULT_INITIALIZER(void*);
David Reissc6dab612008-06-10 22:55:13 +000031 static void ADAPTIVE_INITIALIZER(void*);
David Reissc6dab612008-06-10 22:55:13 +000032 static void RECURSIVE_INITIALIZER(void*);
David Reissc6dab612008-06-10 22:55:13 +000033
Marc Slemko66949872006-07-15 01:52:39 +000034 private:
boz19cee902007-09-22 23:08:11 +000035
Marc Slemko66949872006-07-15 01:52:39 +000036 class impl;
yunfang14542962007-10-03 22:59:41 +000037 boost::shared_ptr<impl> impl_;
Marc Slemko66949872006-07-15 01:52:39 +000038};
39
bozcce81842007-07-06 22:27:52 +000040class ReadWriteMutex {
41public:
42 ReadWriteMutex();
boz19cee902007-09-22 23:08:11 +000043 virtual ~ReadWriteMutex() {}
bozcce81842007-07-06 22:27:52 +000044
45 // these get the lock and block until it is done successfully
46 virtual void acquireRead() const;
47 virtual void acquireWrite() const;
48
49 // these attempt to get the lock, returning false immediately if they fail
50 virtual bool attemptRead() const;
51 virtual bool attemptWrite() const;
52
53 // this releases both read and write locks
54 virtual void release() const;
David Reiss0c90f6f2008-02-06 22:18:40 +000055
bozcce81842007-07-06 22:27:52 +000056private:
boz19cee902007-09-22 23:08:11 +000057
bozcce81842007-07-06 22:27:52 +000058 class impl;
yunfang14542962007-10-03 22:59:41 +000059 boost::shared_ptr<impl> impl_;
bozcce81842007-07-06 22:27:52 +000060};
61
Mark Slee85287d32007-07-09 19:50:30 +000062class Guard {
David Reiss0c90f6f2008-02-06 22:18:40 +000063 public:
bozcce81842007-07-06 22:27:52 +000064 Guard(const Mutex& value) : mutex_(value) {
Mark Slee2f6404d2006-10-10 01:37:40 +000065 mutex_.lock();
Marc Slemko66949872006-07-15 01:52:39 +000066 }
bozcce81842007-07-06 22:27:52 +000067 ~Guard() {
Mark Slee2f6404d2006-10-10 01:37:40 +000068 mutex_.unlock();
Marc Slemko66949872006-07-15 01:52:39 +000069 }
70
71 private:
Mark Slee2f6404d2006-10-10 01:37:40 +000072 const Mutex& mutex_;
Marc Slemko66949872006-07-15 01:52:39 +000073};
74
yunfanga36f5db2007-07-14 01:23:05 +000075class RWGuard {
David Reiss0c90f6f2008-02-06 22:18:40 +000076 public:
yunfanga36f5db2007-07-14 01:23:05 +000077 RWGuard(const ReadWriteMutex& value, bool write = 0) : rw_mutex_(value) {
78 if (write) {
79 rw_mutex_.acquireWrite();
David Reiss96d23882007-07-26 21:10:32 +000080 } else {
yunfanga36f5db2007-07-14 01:23:05 +000081 rw_mutex_.acquireRead();
David Reiss96d23882007-07-26 21:10:32 +000082 }
David Reiss0c90f6f2008-02-06 22:18:40 +000083 }
yunfanga36f5db2007-07-14 01:23:05 +000084 ~RWGuard() {
85 rw_mutex_.release();
David Reiss0c90f6f2008-02-06 22:18:40 +000086 }
87 private:
boz19cee902007-09-22 23:08:11 +000088 const ReadWriteMutex& rw_mutex_;
David Reiss0c90f6f2008-02-06 22:18:40 +000089};
yunfanga36f5db2007-07-14 01:23:05 +000090
Marc Slemko66949872006-07-15 01:52:39 +000091
David Reisseaa61e42007-12-20 21:42:05 +000092// A little hack to prevent someone from trying to do "Guard(m);"
93// Sorry for polluting the global namespace, but I think it's worth it.
94#define Guard(m) incorrect_use_of_Guard(m)
95#define RWGuard(m) incorrect_use_of_RWGuard(m)
96
97
Marc Slemko66949872006-07-15 01:52:39 +000098}}} // facebook::thrift::concurrency
99
Mark Sleef5f2be42006-09-05 21:05:31 +0000100#endif // #ifndef _THRIFT_CONCURRENCY_MUTEX_H_