stk-code_catmod/src/utils/synchronised.hpp

141 lines
4.5 KiB
C++
Raw Normal View History

//
// SuperTuxKart - a fun racing game with go-kart
// Copyright (C) 2010-2013 Joerg Henrichs
//
// This program is free software; you can redistribute it and/or
// modify it under the terms of the GNU General Public License
// as published by the Free Software Foundation; either version 3
// of the License, or (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program; if not, write to the Free Software
// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
#ifndef HEADER_SYNCHRONISED_HPP
#define HEADER_SYNCHRONISED_HPP
#include <pthread.h>
class ISynchronised
{
public :
virtual ~ISynchronised() {}
virtual void lock() const = 0 ;
virtual void unlock() const = 0;
};
/** A variable that is automatically synchronised using pthreads mutex.
*/
template<typename TYPE>
class Synchronised : public ISynchronised
{
private:
/** The mutex to protect this variable with. */
mutable pthread_mutex_t m_mutex;
/** The actual data to be used. */
TYPE m_data;
public:
// ------------------------------------------------------------------------
/** Initialise the data and the mutex with default constructors. */
Synchronised() : m_data(TYPE())
{
pthread_mutex_init(&m_mutex, NULL);
} // Synchronised()
// ------------------------------------------------------------------------
/** Initialise the data and the mutex. */
Synchronised(const TYPE &v)
{
m_data = v;
pthread_mutex_init(&m_mutex, NULL);
} // Synchronised
// ------------------------------------------------------------------------
/** Destroy this mutex.
*/
~Synchronised()
{
pthread_mutex_destroy(&m_mutex);
} // ~Synchronised
// ------------------------------------------------------------------------
/** Sets the value of this variable using a mutex.
* \param v Value to be set.
*/
void setAtomic(const TYPE &v)
{
pthread_mutex_lock(&m_mutex);
m_data = v;
pthread_mutex_unlock(&m_mutex);
} // set
// ------------------------------------------------------------------------
/** Returns a copy of this variable.
*/
TYPE getAtomic() const
{
TYPE v;
pthread_mutex_lock(&m_mutex);
v = m_data;
pthread_mutex_unlock(&m_mutex);
return v;
} // get
// ------------------------------------------------------------------------
/** Returns a reference to the original data file. NOTE: all access to
* the data files happen without mutex protection, so calls to lock
* and unlock are necessary. This method is useful in cases that several
* operations on the class must happen atomic.
*/
TYPE &getData()
{
return m_data;
} // getData
// ------------------------------------------------------------------------
const TYPE &getData() const
{
return m_data;
} // getData
// ------------------------------------------------------------------------
/** Locks the mutex. Note that calls to get() or set() will fail, since
* they will try to lock the mutex as well!
*/
void lock() const { pthread_mutex_lock(&m_mutex); }
// ------------------------------------------------------------------------
/** Unlocks the mutex.
*/
void unlock() const {pthread_mutex_unlock(&m_mutex); }
// ------------------------------------------------------------------------
/** Gives access to the mutex, which can then be used in other pthread
* calls (e.g. pthread_cond_wait).
*/
pthread_mutex_t* getMutex() { return &m_mutex; }
private:
// Make sure that no actual copying is taking place
// ------------------------------------------------------------------------
void operator=(const Synchronised<TYPE>& v) {}
};
#define MutexLocker(x) MutexLockerHelper __dummy(x);
class MutexLockerHelper
{
const ISynchronised * m_synchronised;
public:
MutexLockerHelper(const ISynchronised & synchronised){
m_synchronised = &synchronised;
m_synchronised->lock();
}
~MutexLockerHelper(){
m_synchronised->unlock();
}
};
#endif