diff options
| author | David Faure <faure@kde.org> | 2014-06-14 11:18:26 +0200 | 
|---|---|---|
| committer | David Faure <faure@kde.org> | 2014-06-22 10:32:57 +0200 | 
| commit | fdbcede38d1226b6dcfcf7dbebaecffcb0e962d5 (patch) | |
| tree | b35bf781d7539d2b59e40a279919b3bbfac8c3d0 | |
| parent | e782f0adc99dbcfedabb5f1c90f192151b545b00 (diff) | |
| download | kconfig-fdbcede38d1226b6dcfcf7dbebaecffcb0e962d5.tar.gz kconfig-fdbcede38d1226b6dcfcf7dbebaecffcb0e962d5.tar.bz2 | |
Make KSharedConfig thread-safe
... by having a different list of shareable objects per thread.
REVIEW: 118739
| -rw-r--r-- | src/core/ksharedconfig.cpp | 38 | ||||
| -rw-r--r-- | src/core/ksharedconfig.h | 8 | 
2 files changed, 33 insertions, 13 deletions
| diff --git a/src/core/ksharedconfig.cpp b/src/core/ksharedconfig.cpp index f4b4c766..b7d155d5 100644 --- a/src/core/ksharedconfig.cpp +++ b/src/core/ksharedconfig.cpp @@ -24,6 +24,8 @@  #include "kconfiggroup.h"  #include "kconfig_p.h"  #include <QCoreApplication> +#include <QThread> +#include <QThreadStorage>  void _k_globalMainConfigSync(); @@ -32,11 +34,13 @@ class GlobalSharedConfigList : public QList<KSharedConfig *>  public:      GlobalSharedConfigList()      { -        // We want to force the sync() before the QCoreApplication -        // instance is gone. Otherwise we trigger a QLockFile::lock() -        // after QCoreApplication is gone, calling qAppName() for a non -        // existent app... -        qAddPostRoutine(&_k_globalMainConfigSync); +        if (!qApp || QThread::currentThread() == qApp->thread()) { +            // We want to force the sync() before the QCoreApplication +            // instance is gone. Otherwise we trigger a QLockFile::lock() +            // after QCoreApplication is gone, calling qAppName() for a non +            // existent app... +            qAddPostRoutine(&_k_globalMainConfigSync); +        }      }      // in addition to the list, we need to hold the main config, @@ -44,12 +48,24 @@ public:      KSharedConfigPtr mainConfig;  }; -Q_GLOBAL_STATIC(GlobalSharedConfigList, globalSharedConfigList) +static QThreadStorage<GlobalSharedConfigList *> s_storage; +template <typename T> +T * perThreadGlobalStatic() +{ +    if (!s_storage.hasLocalData()) { +        s_storage.setLocalData(new T); +    } +    return s_storage.localData(); +}; + +// Q_GLOBAL_STATIC(GlobalSharedConfigList, globalSharedConfigList), but per thread: +static GlobalSharedConfigList *globalSharedConfigList() { return perThreadGlobalStatic<GlobalSharedConfigList>(); }  void _k_globalMainConfigSync()  { -    if (globalSharedConfigList->mainConfig) { -        globalSharedConfigList->mainConfig->sync(); +    KSharedConfigPtr mainConfig = globalSharedConfigList()->mainConfig; +    if (mainConfig) { +        mainConfig->sync();      }  } @@ -64,7 +80,7 @@ KSharedConfigPtr KSharedConfig::openConfig(const QString &_fileName,          fileName = KConfig::mainConfigName();      } -    static bool wasTestModeEnabled = false; +    static QBasicAtomicInt wasTestModeEnabled = Q_BASIC_ATOMIC_INITIALIZER(false);      if (!wasTestModeEnabled && QStandardPaths::isTestModeEnabled()) {          wasTestModeEnabled = true;          list->clear(); @@ -86,7 +102,7 @@ KSharedConfigPtr KSharedConfig::openConfig(const QString &_fileName,      if (_fileName.isEmpty() && flags == FullConfig && resType == QStandardPaths::GenericConfigLocation) {          list->mainConfig = ptr; -        static bool userWarned = false; +        static QBasicAtomicInt userWarned = Q_BASIC_ATOMIC_INITIALIZER(false);          if (!userWarned) {              userWarned = true;              QByteArray readOnly = qgetenv("KDE_HOME_READONLY"); @@ -111,7 +127,7 @@ KSharedConfig::KSharedConfig(const QString &fileName,  KSharedConfig::~KSharedConfig()  { -    if (!globalSharedConfigList.isDestroyed()) { +    if (s_storage.hasLocalData()) {          globalSharedConfigList()->removeAll(this);      }  } diff --git a/src/core/ksharedconfig.h b/src/core/ksharedconfig.h index 03f05b08..b2317afd 100644 --- a/src/core/ksharedconfig.h +++ b/src/core/ksharedconfig.h @@ -30,12 +30,16 @@   *   * KConfig variant using shared memory   * - * KSharedConfig provides a reference counted, shared memory variant + * KSharedConfig provides a shared (reference counted) variant   * of KConfig.  This allows you to use manipulate the same configuration   * files from different places in your code without worrying about   * accidentally overwriting changes.   * - * Note that, as with most of kdelibs, this is @b NOT threadsafe. + * The openConfig() method is threadsafe: every thread gets a separate repository + * of shared KConfig objects. This means, however, that you'll be responsible for + * synchronizing the instances of KConfig for the same filename between threads, + * using reparseConfiguration after a manual change notification, just like you have + * to do between processes.   */  class KCONFIGCORE_EXPORT KSharedConfig : public KConfig, public QSharedData //krazy:exclude=dpointer (only for refcounting)  { | 
