// Copyright (C) 2010  Davis E. King (davis@dlib.net)
// License: Boost Software License   See LICENSE.txt for the full license.
#undef DLIB_READWRITE_MUTEX_EXTENSIOn_ABSTRACT_
#ifdef DLIB_READWRITE_MUTEX_EXTENSIOn_ABSTRACT_

#include "threads_kernel_abstract.h"

namespace dlib
{

// ----------------------------------------------------------------------------------------

    class read_write_mutex
    {
        /*!
            INITIAL VALUE
                read_write_mutex is in the fully 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.   

                This mutex also has the additional ability to distinguish between
                a lock for the purposes of modifying some shared data, a write lock,
                and a lock for the purposes of only reading shared data, a readonly
                lock.  The lock() and unlock() functions are used for write locks while
                the lock_readonly() and unlock_readonly() are for readonly locks.  

                The difference between a readonly and write lock can be understood as 
                follows.  The read_write_mutex will allow many threads to obtain simultaneous
                readonly locks but will only allow a single thread to obtain a write lock.
                Moreover, while the write lock is obtained no other threads are allowed
                to have readonly locks.  
        !*/
    public:

        read_write_mutex (
        );
        /*!
            ensures
                - #*this is properly initialized
                - max_readonly_locks() == 0xFFFFFFFF
                  (i.e. about 4 billion)
            throws
                - dlib::thread_error
                    the constructor may throw this exception if there is a problem 
                    gathering resources to create the read_write_mutex.
        !*/

        explicit read_write_mutex (
            unsigned long max_locks
        );
        /*!
            requires
                - max_locks > 0
            ensures
                - #*this is properly initialized
                - max_readonly_locks() == max_locks
            throws
                - dlib::thread_error
                    the constructor may throw this exception if there is a problem 
                    gathering resources to create the read_write_mutex.
        !*/

        ~read_write_mutex (
        );
        /*!
            requires
                - *this is not locked
            ensures
                - all resources allocated by *this have been freed
        !*/

        void lock (
        ) const;
        /*!
            requires
                - The thread calling this function does not have any kind of lock on this 
                  object
            ensures
                - if (there is any kind of lock on *this) then 
                    - the calling thread is put to sleep until a write lock becomes available. 
                      Once available, a write lock is obtained on this mutex and this function 
                      terminates.
                - else  
                    - a write lock is obtained on this mutex and the calling thread is not put to sleep 
        !*/

        void unlock (
        ) const;
        /*!
            ensures
                - if (there is a write lock on *this) then
                    - #*this is unlocked (i.e. other threads may now lock this object)
                - else
                    - the call to unlock() has no effect
        !*/

        unsigned long max_readonly_locks (
        ) const;
        /*!
            ensures
                - returns the maximum number of concurrent readonly locks this object will allow.
        !*/

        void lock_readonly (
        ) const;
        /*!
            requires
                - The thread calling this function does not already have a write
                  lock on this object
            ensures
                - if (there is a write lock on *this or there are no free readonly locks) then
                    - the calling thread is put to sleep until there is no longer a write lock
                      and a free readonly lock is available.  Once this is the case, a readonly 
                      lock is obtained and this function terminates.
                - else 
                    - a readonly lock is obtained on *this and the calling thread is not put
                      to sleep.  Note that multiple readonly locks can be obtained at once.
        !*/

        void unlock_readonly (
        ) const;
        /*!
            ensures
                - if (there is a readonly lock on *this) then
                    - one readonly lock is removed from *this.  
                - else
                    - the call to unlock_readonly() has no effect.
        !*/

    private:
        // restricted functions
        read_write_mutex(read_write_mutex&);        // copy constructor
        read_write_mutex& operator=(read_write_mutex&);    // assignment operator
    };

// ----------------------------------------------------------------------------------------

}

#endif // DLIB_READWRITE_MUTEX_EXTENSIOn_ABSTRACT_