You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
concurrentqueue/benchmarks/dlib/threads/threads_kernel_abstract.h

303 lines
11 KiB
C++

// Copyright (C) 2003 Davis E. King (davis@dlib.net)
// License: Boost Software License See LICENSE.txt for the full license.
#undef DLIB_THREADS_KERNEl_ABSTRACT_
#ifdef DLIB_THREADS_KERNEl_ABSTRACT_
namespace dlib
{
// ----------------------------------------------------------------------------------------
/*!
THREAD POOLING
When threads end they go into a global thread pool and each waits there
for 30 seconds before timing out and having its resources returned to the
operating system. When create_new_thread() is called it first looks in the
thread pool to see if there are any threads it can snatch from the pool, if
not then it makes a new one.
Note that whenever I say something happens when a thread "terminates" or "ends"
I mean "when it returns to the thread pool." From the client programmer point
of view a thread terminates/ends when it returns to the dlib thread pool and you
shouldn't and indeed don't need to know when it actually gets its resources
reclaimed by the operating system.
If you want to change the timeout to a different value you can #define
DLIB_THREAD_POOL_TIMEOUT to whatever value (in milliseconds) that you like.
EXCEPTIONS
Unless specified otherwise, nothing in this file throws exceptions.
!*/
// ----------------------------------------------------------------------------------------
thread_id_type get_thread_id (
);
/*!
ensures
- returns a unique id for the calling thread. Note that while the id is unique
among all currently existing threads it may have been used by a previous
thread that has terminated.
!*/
// ----------------------------------------------------------------------------------------
bool is_dlib_thread (
thread_id_type id = get_thread_id()
);
/*!
ensures
- if (the thread with the given id was spawned by a call to
dlib::create_new_thread) then
- returns true
- else
- returns false
!*/
// ----------------------------------------------------------------------------------------
template <
typename T
>
void register_thread_end_handler (
T& obj,
void (T::*handler)()
);
/*!
requires
- handler == a valid member function pointer for class T
- handler does not throw
- handler does not call register_thread_end_handler()
- handler does not block
- is_dlib_thread() == true (i.e. the calling thread was spawned by dlib::create_new_thread())
ensures
- let ID == the thread id for the thread calling register_thread_end_handler()
- (obj.*handler)() will be called when the thread with thread id ID is
terminating and it will be called from within that terminating thread.
(i.e. inside the handler function get_thread_id() == ID == the id of the
thread that is terminating. )
- each call to this function adds another handler that will be called when
the given thread terminates. This means that if you call it a bunch of
times then you will end up registering multiple handlers (or single
handlers multiple times) that will be called when the thread ends.
throws
- std::bad_alloc
If this exception is thrown then the call to this function had no effect.
!*/
// ----------------------------------------------------------------------------------------
template <
typename T
>
void unregister_thread_end_handler (
T& obj,
void (T::*handler)()
);
/*!
requires
- handler == a valid member function pointer for class T
ensures
- Undoes all previous calls to register_thread_end_handler(obj,handler).
So the given handler won't be called when any threads end.
throws
- std::bad_alloc
If this exception is thrown then the call to this function had no effect.
!*/
// ----------------------------------------------------------------------------------------
bool create_new_thread (
void (*funct)(void*),
void* param
);
/*!
ensures
- creates a new thread for the function pointed to by funct
- passes it param as its parameter. (i.e. calls funct(param) from the new thread)
- returns true upon success and false upon failure to create the new thread
!*/
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
// mutex object
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
class mutex
{
/*!
INITIAL VALUE
mutex is in the unlocked state
WHAT THIS OBJECT REPRESENTS
This object represents a mutex intended to be used for synchronous
thread control of shared data. When a thread wants to access some
shared data it locks out other threads by calling lock() and calls
unlock() when it is finished.
!*/
public:
mutex (
);
/*!
ensures
- #*this is properly initialized
throws
- dlib::thread_error
the constructor may throw this exception if there is a problem
gathering resources to create the mutex.
!*/
~mutex (
);
/*!
requires
- *this is not locked
ensures
- all resources allocated by *this have been freed
!*/
void lock (
) const;
/*!
requires
- the thread calling lock() does not already have a lock on *this
ensures
- if (*this is currently locked by another thread) then
- the thread that called lock() on *this is put to sleep until
it becomes available
- if (*this is currently unlocked) then
- #*this becomes locked and the current thread is NOT put to sleep
but now "owns" #*this
!*/
void unlock (
) const;
/*!
requires
- the thread calling unlock() already has a lock on *this
ensures
- #*this is unlocked (i.e. other threads may now lock this object)
!*/
private:
// restricted functions
mutex(mutex&); // copy constructor
mutex& operator=(mutex&); // assignment operator
};
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
// signaler object
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
class signaler
{
/*!
WHAT THIS OBJECT REPRESENTS
This object represents an event signaling system for threads. It gives
a thread the ability to wake up other threads that are waiting for a
particular signal.
Each signaler object is associated with one and only one mutex object.
More than one signaler object may be associated with a single mutex
but a signaler object may only be associated with a single mutex.
NOTE:
You must guard against spurious wakeups. This means that a thread
might return from a call to wait even if no other thread called
signal. This is rare but must be guarded against.
!*/
public:
signaler (
const mutex& associated_mutex
);
/*!
ensures
- #*this is properly initialized
- #get_mutex() == associated_mutex
throws
- dlib::thread_error
the constructor may throw this exception if there is a problem
gathering resources to create the signaler.
!*/
~signaler (
);
/*!
ensures
- all resources allocated by *this have been freed
!*/
void wait (
) const;
/*!
requires
- get_mutex() is locked and owned by the calling thread
ensures
- atomically unlocks get_mutex() and blocks the calling thread
- calling thread may wake if another thread calls signal() or broadcast()
on *this
- when wait() returns the calling thread again has a lock on get_mutex()
!*/
bool wait_or_timeout (
unsigned long milliseconds
) const;
/*!
requires
- get_mutex() is locked and owned by the calling thread
ensures
- atomically unlocks get_mutex() and blocks the calling thread
- calling thread may wake if another thread calls signal() or broadcast()
on *this
- after the specified number of milliseconds has elapsed the calling thread
will wake once get_mutex() is free
- when wait returns the calling thread again has a lock on get_mutex()
- returns false if the call to wait_or_timeout timed out
- returns true if the call did not time out
!*/
void signal (
) const;
/*!
ensures
- if (at least one thread is waiting on *this) then
- at least one of the waiting threads will wake
!*/
void broadcast (
) const;
/*!
ensures
- any and all threads waiting on *this will wake
!*/
const mutex& get_mutex (
) const;
/*!
ensures
- returns a const reference to the mutex associated with *this
!*/
private:
// restricted functions
signaler(signaler&); // copy constructor
signaler& operator=(signaler&); // assignment operator
};
// ----------------------------------------------------------------------------------------
}
#endif // DLIB_THREADS_KERNEl_ABSTRACT_