poco/Foundation/include/Poco/Condition.h

138 lines
3.2 KiB
C
Raw Normal View History

2012-04-29 18:52:25 +00:00
//
// Condition.h
//
// Library: Foundation
// Package: Threading
// Module: Condition
//
// Definition of the Condition class template.
//
// Copyright (c) 2007, Applied Informatics Software Engineering GmbH.
// and Contributors.
//
// SPDX-License-Identifier: BSL-1.0
2012-04-29 18:52:25 +00:00
//
#ifndef Foundation_Condition_INCLUDED
#define Foundation_Condition_INCLUDED
#include "Poco/Foundation.h"
#include "Poco/Mutex.h"
#include "Poco/ScopedUnlock.h"
#include "Poco/Event.h"
#include "Poco/Exception.h"
#include <deque>
namespace Poco {
class Foundation_API Condition
2022-07-07 17:18:20 +08:00
/// A Condition is a synchronization object used to block a thread
/// until a particular condition is met.
2012-04-29 18:52:25 +00:00
/// A Condition object is always used in conjunction with
/// a Mutex (or FastMutex) object.
///
/// Condition objects are similar to POSIX condition variables, which the
/// difference that Condition is not subject to spurious wakeups.
///
/// Threads waiting on a Condition are resumed in FIFO order.
{
public:
Condition();
/// Creates the Condition.
2022-07-07 17:18:20 +08:00
2012-04-29 18:52:25 +00:00
~Condition();
/// Destroys the Condition.
2022-07-07 17:18:20 +08:00
2012-04-29 18:52:25 +00:00
template <class Mtx>
void wait(Mtx& mutex)
/// Unlocks the mutex (which must be locked upon calling
/// wait()) and waits until the Condition is signalled.
///
2022-07-07 17:18:20 +08:00
/// The given mutex will be locked again upon
2012-04-29 18:52:25 +00:00
/// leaving the function, even in case of an exception.
{
ScopedUnlock<Mtx> unlock(mutex, false);
Event event;
{
FastMutex::ScopedLock lock(_mutex);
mutex.unlock();
enqueue(event);
}
event.wait();
}
2022-07-07 17:18:20 +08:00
2012-04-29 18:52:25 +00:00
template <class Mtx>
void wait(Mtx& mutex, long milliseconds)
/// Unlocks the mutex (which must be locked upon calling
/// wait()) and waits for the given time until the Condition is signalled.
///
2022-07-07 17:18:20 +08:00
/// The given mutex will be locked again upon successfully leaving the
2012-04-29 18:52:25 +00:00
/// function, even in case of an exception.
///
/// Throws a TimeoutException if the Condition is not signalled
/// within the given time interval.
{
if (!tryWait(mutex, milliseconds))
throw TimeoutException();
}
2022-07-07 17:18:20 +08:00
2012-04-29 18:52:25 +00:00
template <class Mtx>
bool tryWait(Mtx& mutex, long milliseconds)
/// Unlocks the mutex (which must be locked upon calling
/// tryWait()) and waits for the given time until the Condition is signalled.
///
2022-07-07 17:18:20 +08:00
/// The given mutex will be locked again upon leaving the
2012-04-29 18:52:25 +00:00
/// function, even in case of an exception.
///
/// Returns true if the Condition has been signalled
/// within the given time interval, otherwise false.
{
ScopedUnlock<Mtx> unlock(mutex, false);
Event event;
{
FastMutex::ScopedLock lock(_mutex);
mutex.unlock();
enqueue(event);
}
if (!event.tryWait(milliseconds))
{
FastMutex::ScopedLock lock(_mutex);
dequeue(event);
return false;
}
return true;
}
2022-07-07 17:18:20 +08:00
2012-04-29 18:52:25 +00:00
void signal();
/// Signals the Condition and allows one waiting thread
/// to continue execution.
void broadcast();
/// Signals the Condition and allows all waiting
/// threads to continue their execution.
protected:
void enqueue(Event& event);
void dequeue();
void dequeue(Event& event);
2022-07-07 17:18:20 +08:00
2012-04-29 18:52:25 +00:00
private:
Condition(const Condition&);
Condition& operator = (const Condition&);
2022-07-07 17:18:20 +08:00
2012-04-29 18:52:25 +00:00
typedef std::deque<Event*> WaitQueue;
2022-07-07 17:18:20 +08:00
2012-04-29 18:52:25 +00:00
FastMutex _mutex;
WaitQueue _waitQueue;
};
} // namespace Poco
#endif // Foundation_Condition_INCLUDED