2007-08-09 20:03:17 +00:00
|
|
|
/*
|
2011-01-10 21:54:38 +00:00
|
|
|
* Copyright 2007-2011, Ingo Weinhold, ingo_weinhold@gmx.de.
|
2007-08-09 20:03:17 +00:00
|
|
|
* Distributed under the terms of the MIT License.
|
|
|
|
*/
|
|
|
|
#ifndef _KERNEL_CONDITION_VARIABLE_H
|
|
|
|
#define _KERNEL_CONDITION_VARIABLE_H
|
|
|
|
|
|
|
|
|
|
|
|
#include <OS.h>
|
|
|
|
|
2007-09-02 22:21:26 +00:00
|
|
|
#include <debug.h>
|
|
|
|
|
2007-08-09 20:03:17 +00:00
|
|
|
#ifdef __cplusplus
|
|
|
|
|
2008-04-20 15:19:48 +00:00
|
|
|
#include <util/DoublyLinkedList.h>
|
2007-08-09 20:03:17 +00:00
|
|
|
#include <util/OpenHashTable.h>
|
|
|
|
|
|
|
|
|
2009-06-22 23:49:05 +00:00
|
|
|
struct ConditionVariable;
|
2007-08-09 20:03:17 +00:00
|
|
|
|
2007-08-26 23:53:12 +00:00
|
|
|
|
2008-04-22 18:32:15 +00:00
|
|
|
struct ConditionVariableEntry
|
|
|
|
: DoublyLinkedListLinkImpl<ConditionVariableEntry> {
|
2007-08-26 23:53:12 +00:00
|
|
|
public:
|
2007-09-02 22:21:26 +00:00
|
|
|
#if KDEBUG
|
2008-04-22 18:32:15 +00:00
|
|
|
inline ConditionVariableEntry();
|
|
|
|
inline ~ConditionVariableEntry();
|
2007-09-02 22:21:26 +00:00
|
|
|
#endif
|
|
|
|
|
2008-05-17 10:21:37 +00:00
|
|
|
bool Add(const void* object);
|
|
|
|
status_t Wait(uint32 flags = 0, bigtime_t timeout = 0);
|
2008-04-22 21:46:23 +00:00
|
|
|
status_t Wait(const void* object, uint32 flags = 0,
|
|
|
|
bigtime_t timeout = 0);
|
2007-08-26 23:53:12 +00:00
|
|
|
|
2009-10-21 23:44:59 +00:00
|
|
|
inline status_t WaitStatus() const { return fWaitStatus; }
|
|
|
|
|
|
|
|
inline ConditionVariable* Variable() const { return fVariable; }
|
2008-04-22 18:32:15 +00:00
|
|
|
|
2008-04-22 21:46:23 +00:00
|
|
|
private:
|
2008-07-16 22:43:50 +00:00
|
|
|
inline void AddToVariable(ConditionVariable* variable);
|
2008-04-22 21:46:23 +00:00
|
|
|
|
2008-04-22 18:32:15 +00:00
|
|
|
private:
|
|
|
|
ConditionVariable* fVariable;
|
2011-01-10 21:54:38 +00:00
|
|
|
Thread* fThread;
|
2008-05-17 10:21:37 +00:00
|
|
|
status_t fWaitStatus;
|
2007-08-26 23:53:12 +00:00
|
|
|
|
2009-06-22 23:49:05 +00:00
|
|
|
friend struct ConditionVariable;
|
2007-08-09 20:03:17 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2009-07-27 00:39:12 +00:00
|
|
|
struct ConditionVariable {
|
2007-08-09 20:03:17 +00:00
|
|
|
public:
|
2008-04-22 21:46:23 +00:00
|
|
|
void Init(const void* object,
|
|
|
|
const char* objectType);
|
|
|
|
// for anonymous (unpublished) cvars
|
|
|
|
|
2007-08-09 20:03:17 +00:00
|
|
|
void Publish(const void* object,
|
|
|
|
const char* objectType);
|
2013-11-08 02:41:26 +01:00
|
|
|
void Unpublish();
|
|
|
|
|
|
|
|
inline void NotifyOne(status_t result = B_OK);
|
|
|
|
inline void NotifyAll(status_t result = B_OK);
|
|
|
|
|
|
|
|
static void NotifyOne(const void* object, status_t result);
|
|
|
|
static void NotifyAll(const void* object, status_t result);
|
2009-12-03 12:47:29 +00:00
|
|
|
// (both methods) caller must ensure that
|
|
|
|
// the variable is not unpublished
|
|
|
|
// concurrently
|
2009-12-03 12:24:17 +00:00
|
|
|
|
2008-07-16 22:43:50 +00:00
|
|
|
void Add(ConditionVariableEntry* entry);
|
2008-04-22 21:46:23 +00:00
|
|
|
|
2008-07-22 20:36:32 +00:00
|
|
|
status_t Wait(uint32 flags = 0, bigtime_t timeout = 0);
|
|
|
|
// all-in one, i.e. doesn't need a
|
|
|
|
// ConditionVariableEntry
|
|
|
|
|
2008-09-03 14:53:01 +00:00
|
|
|
const void* Object() const { return fObject; }
|
|
|
|
const char* ObjectType() const { return fObjectType; }
|
2008-04-22 18:32:15 +00:00
|
|
|
|
|
|
|
static void ListAll();
|
|
|
|
void Dump() const;
|
2007-08-09 20:03:17 +00:00
|
|
|
|
|
|
|
private:
|
2013-11-08 02:41:26 +01:00
|
|
|
void _Notify(bool all, status_t result);
|
2009-10-21 23:44:59 +00:00
|
|
|
void _NotifyLocked(bool all, status_t result);
|
2007-08-09 20:03:17 +00:00
|
|
|
|
|
|
|
protected:
|
2008-04-22 18:32:15 +00:00
|
|
|
typedef DoublyLinkedList<ConditionVariableEntry> EntryList;
|
2008-04-20 15:19:48 +00:00
|
|
|
|
2007-08-09 20:03:17 +00:00
|
|
|
const void* fObject;
|
|
|
|
const char* fObjectType;
|
2008-04-20 15:19:48 +00:00
|
|
|
EntryList fEntries;
|
2009-07-27 00:39:12 +00:00
|
|
|
ConditionVariable* fNext;
|
2007-08-09 20:03:17 +00:00
|
|
|
|
2009-06-22 23:49:05 +00:00
|
|
|
friend struct ConditionVariableEntry;
|
|
|
|
friend struct ConditionVariableHashDefinition;
|
2007-08-09 20:03:17 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2008-04-22 18:32:15 +00:00
|
|
|
#if KDEBUG
|
2007-08-09 20:03:17 +00:00
|
|
|
|
2008-04-22 18:32:15 +00:00
|
|
|
inline
|
|
|
|
ConditionVariableEntry::ConditionVariableEntry()
|
|
|
|
: fVariable(NULL)
|
2007-08-09 20:03:17 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2008-04-22 18:32:15 +00:00
|
|
|
inline
|
|
|
|
ConditionVariableEntry::~ConditionVariableEntry()
|
2007-08-09 20:03:17 +00:00
|
|
|
{
|
2008-04-22 18:32:15 +00:00
|
|
|
if (fVariable != NULL) {
|
|
|
|
panic("Destroying condition variable entry %p, but it's still "
|
|
|
|
"attached to variable %p\n", this, fVariable);
|
|
|
|
}
|
2007-08-09 20:03:17 +00:00
|
|
|
}
|
|
|
|
|
2008-04-22 18:32:15 +00:00
|
|
|
#endif
|
2007-08-09 20:03:17 +00:00
|
|
|
|
|
|
|
|
2007-08-26 23:53:12 +00:00
|
|
|
inline void
|
2013-11-08 02:41:26 +01:00
|
|
|
ConditionVariable::NotifyOne(status_t result)
|
2007-08-09 20:03:17 +00:00
|
|
|
{
|
2013-11-08 02:41:26 +01:00
|
|
|
_Notify(false, result);
|
2007-08-09 20:03:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-04-22 18:32:15 +00:00
|
|
|
inline void
|
2013-11-08 02:41:26 +01:00
|
|
|
ConditionVariable::NotifyAll(status_t result)
|
2007-08-26 23:53:12 +00:00
|
|
|
{
|
2013-11-08 02:41:26 +01:00
|
|
|
_Notify(true, result);
|
2007-08-26 23:53:12 +00:00
|
|
|
}
|
2007-08-09 20:03:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
extern "C" {
|
|
|
|
#endif // __cplusplus
|
|
|
|
|
|
|
|
extern void condition_variable_init();
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
} // extern "C"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif /* _KERNEL_CONDITION_VARIABLE_H */
|