/* * Copyright (C) 2016 Alexander Makarov * * Source: * https://wiki.qt.io/Qt_thread-safe_singleton * * This file is a part of Breakpad-qt library. * * 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. * */ #ifndef CALL_ONCE #define CALL_ONCE #include #include #include #include #include #include namespace CallOnce { enum ECallOnce { CO_Request, CO_InProgress, CO_Finished }; Q_GLOBAL_STATIC(QThreadStorage, once_flag) } template inline static void qCallOnce(Function func, QBasicAtomicInt& flag) { using namespace CallOnce; #if QT_VERSION < QT_VERSION_CHECK(5, 0, 0) int protectFlag = flag.fetchAndStoreAcquire(flag); #elif QT_VERSION >= QT_VERSION_CHECK(5, 14, 0) int protectFlag = flag.fetchAndStoreAcquire(flag.loadRelaxed()); #else int protectFlag = flag.fetchAndStoreAcquire(flag.load()); #endif if (protectFlag == CO_Finished) return; if (protectFlag == CO_Request && flag.testAndSetRelaxed(protectFlag, CO_InProgress)) { func(); flag.fetchAndStoreRelease(CO_Finished); } else { do { QThread::yieldCurrentThread(); } while (!flag.testAndSetAcquire(CO_Finished, CO_Finished)); } } template inline static void qCallOncePerThread(Function func) { using namespace CallOnce; if (!once_flag()->hasLocalData()) { once_flag()->setLocalData(new QAtomicInt(CO_Request)); qCallOnce(func, *once_flag()->localData()); } } #endif // CALL_ONCE