2017-11-04 18:43:02 +01:00
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
// MuldeR's Utilities for Qt
|
2022-06-16 16:44:01 +02:00
|
|
|
// Copyright (C) 2004-2022 LoRd_MuldeR <MuldeR2@GMX.de>
|
2017-11-04 18:43:02 +01:00
|
|
|
//
|
|
|
|
// This library is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU Lesser General Public
|
|
|
|
// License as published by the Free Software Foundation; either
|
|
|
|
// version 2.1 of the License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This library 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
|
|
|
|
// Lesser General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Lesser General Public
|
|
|
|
// License along with this library; if not, write to the Free Software
|
|
|
|
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
//
|
|
|
|
// http://www.gnu.org/licenses/lgpl-2.1.txt
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @file
|
|
|
|
* @brief This file contains a template class for lazy initialization
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
//MUtils
|
|
|
|
#include <MUtils/Global.h>
|
|
|
|
#include <MUtils/Exception.h>
|
|
|
|
|
|
|
|
//Qt
|
2018-12-08 15:17:33 +01:00
|
|
|
#include <QThread>
|
2017-11-04 18:43:02 +01:00
|
|
|
#include <QAtomicPointer>
|
2018-12-08 15:17:33 +01:00
|
|
|
#include <QAtomicInt>
|
2017-11-04 18:43:02 +01:00
|
|
|
|
2017-11-04 21:10:32 +01:00
|
|
|
//CRT
|
|
|
|
#include <functional>
|
|
|
|
|
2017-11-04 18:43:02 +01:00
|
|
|
namespace MUtils
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* \brief Lazy initialization template class
|
|
|
|
*
|
2017-11-04 21:10:32 +01:00
|
|
|
* The lazy-initialized value of type T can be obtained from a `Lazy<T>` instance by using the `operator*()`. Initialization of the value happens when the `operator*()` is called for the very first time, by invoking the `initializer` lambda-function that was passed to the constructor. The return value of the `initializer` lambda-function is then stored internally, so that any subsequent call to the `operator*()` *immediately* returns the previously created value.
|
2017-11-04 18:43:02 +01:00
|
|
|
*
|
2018-12-08 15:17:33 +01:00
|
|
|
* **Note on thread-saftey:** This class is thread-safe in the sense that all calls to `operator*()` on the same `Lazy<T>` instance, regardless from which thread, are guaranteed to return the exactly same value/object. The *first* thread trying to access the value will invoke the `initializer` lambda-function; concurrent threads may need to busy-wait until the initialization is completed. The `initializer` lambda-function is invoked at most once.
|
2017-11-04 18:43:02 +01:00
|
|
|
*/
|
|
|
|
template<typename T> class Lazy
|
|
|
|
{
|
|
|
|
public:
|
2017-11-04 21:10:32 +01:00
|
|
|
Lazy(std::function<T*(void)> &&initializer) : m_initializer(initializer) { }
|
|
|
|
|
2017-11-04 18:43:02 +01:00
|
|
|
T& operator*(void)
|
|
|
|
{
|
2018-12-08 15:17:33 +01:00
|
|
|
return (*getValue());
|
|
|
|
}
|
|
|
|
|
|
|
|
T* operator->(void)
|
|
|
|
{
|
|
|
|
return getValue();
|
|
|
|
}
|
|
|
|
|
2018-12-08 15:35:32 +01:00
|
|
|
bool initialized()
|
|
|
|
{
|
|
|
|
return (m_state > 1);
|
|
|
|
}
|
|
|
|
|
2018-12-08 15:17:33 +01:00
|
|
|
~Lazy(void)
|
|
|
|
{
|
|
|
|
if(T *const value = m_value)
|
2017-11-04 18:43:02 +01:00
|
|
|
{
|
2018-04-15 15:12:13 +02:00
|
|
|
delete value;
|
2017-11-04 18:43:02 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-08 15:17:33 +01:00
|
|
|
protected:
|
|
|
|
__forceinline T* getValue()
|
2018-04-15 01:27:07 +02:00
|
|
|
{
|
2018-12-08 15:17:33 +01:00
|
|
|
T *value;
|
|
|
|
while (!(value = m_value))
|
2018-04-15 01:27:07 +02:00
|
|
|
{
|
2018-12-08 15:35:32 +01:00
|
|
|
if (m_state.testAndSetOrdered(0, 1))
|
2018-12-08 15:17:33 +01:00
|
|
|
{
|
|
|
|
if (value = m_initializer())
|
|
|
|
{
|
|
|
|
m_value.fetchAndStoreOrdered(value);
|
2018-12-08 15:35:32 +01:00
|
|
|
m_state.fetchAndStoreOrdered(2);
|
2018-12-08 15:17:33 +01:00
|
|
|
break; /*success*/
|
|
|
|
}
|
2018-12-08 15:35:32 +01:00
|
|
|
m_state.fetchAndStoreOrdered(0);
|
2018-12-08 15:17:33 +01:00
|
|
|
MUTILS_THROW("Initializer returned NULL pointer!");
|
|
|
|
}
|
|
|
|
QThread::yieldCurrentThread();
|
2018-04-15 01:27:07 +02:00
|
|
|
}
|
2018-12-08 15:17:33 +01:00
|
|
|
return value;
|
2018-04-15 01:27:07 +02:00
|
|
|
}
|
|
|
|
|
2017-11-04 18:43:02 +01:00
|
|
|
private:
|
2017-11-04 21:10:32 +01:00
|
|
|
QAtomicPointer<T> m_value;
|
2018-12-08 15:35:32 +01:00
|
|
|
QAtomicInt m_state;
|
2017-11-04 21:10:32 +01:00
|
|
|
const std::function<T*(void)> m_initializer;
|
2017-11-04 18:43:02 +01:00
|
|
|
};
|
|
|
|
}
|