/**************************************************************************** ** ** Copyright (C) 2015 The Qt Company Ltd. ** Contact: http://www.qt.io/licensing ** ** This file is part of the Qt Build Suite. ** ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms and ** conditions see http://www.qt.io/terms-conditions. For further information ** use the contact form at http://www.qt.io/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 2.1 or version 3 as published by the Free ** Software Foundation and appearing in the file LICENSE.LGPLv21 and ** LICENSE.LGPLv3 included in the packaging of this file. Please review the ** following information to ensure the GNU Lesser General Public License ** requirements will be met: https://www.gnu.org/licenses/lgpl.html and ** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, The Qt Company gives you certain additional ** rights. These rights are described in The Qt Company LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ****************************************************************************/ #ifndef QBS_PERSISTENCE #define QBS_PERSISTENCE #include "persistentobject.h" #include #include #include #include #include #include namespace qbs { namespace Internal { class PersistentPool { public: PersistentPool(const Logger &logger); ~PersistentPool(); class HeadData { public: QVariantMap projectConfig; }; void load(const QString &filePath); void setupWriteStream(const QString &filePath); void finalizeWriteStream(); void closeStream(); void clear(); QDataStream &stream(); template T *idLoad(); template void loadContainer(T &container); template QSharedPointer idLoadS(); template void loadContainerS(T &container); void store(const QSharedPointer &ptr) { store(ptr.data()); } void store(const PersistentObject *object); template void storeContainer(const T &container); void store(const QVariantMap &map); QVariantMap loadVariantMap(); void store(const QVariant &variant); QVariant loadVariant(); void storeString(const QString &t); QString loadString(int id); QString idLoadString(); void storeStringSet(const QSet &t); QSet loadStringSet(const QList &id); QSet idLoadStringSet(); void storeStringList(const QStringList &t); QStringList loadStringList(const QList &ids); QStringList idLoadStringList(); const HeadData &headData() const { return m_headData; } void setHeadData(const HeadData &hd) { m_headData = hd; } private: typedef int PersistentObjectId; template struct RemovePointer { typedef T Type; }; template struct RemovePointer { typedef T Type; }; template struct RemoveConst { typedef T Type; }; template struct RemoveConst { typedef T Type; }; template T *loadRaw(PersistentObjectId id); template QSharedPointer load(PersistentObjectId id); QDataStream m_stream; HeadData m_headData; QVector m_loadedRaw; QVector > m_loaded; QHash m_storageIndices; PersistentObjectId m_lastStoredObjectId; QVector m_stringStorage; QHash m_inverseStringStorage; PersistentObjectId m_lastStoredStringId; Logger m_logger; }; template inline T *PersistentPool::idLoad() { PersistentObjectId id; stream() >> id; return loadRaw(id); } template inline void PersistentPool::loadContainer(T &container) { int count; stream() >> count; container.clear(); container.reserve(count); for (int i = count; --i >= 0;) container += idLoad::Type>(); } template inline QSharedPointer PersistentPool::idLoadS() { PersistentObjectId id; m_stream >> id; return load(id); } template inline void PersistentPool::loadContainerS(T &container) { int count; stream() >> count; container.clear(); container.reserve(count); for (int i = count; --i >= 0;) container += idLoadS::Type>(); } template inline void PersistentPool::storeContainer(const T &container) { stream() << container.count(); typename T::const_iterator it = container.constBegin(); const typename T::const_iterator itEnd = container.constEnd(); for (; it != itEnd; ++it) store(*it); } template inline T *PersistentPool::loadRaw(PersistentObjectId id) { if (id < 0) return 0; if (id < m_loadedRaw.count()) { PersistentObject *obj = m_loadedRaw.value(id); return dynamic_cast(obj); } int i = m_loadedRaw.count(); m_loadedRaw.resize(id + 1); for (; i < m_loadedRaw.count(); ++i) m_loadedRaw[i] = 0; T * const t = new T; PersistentObject * const po = t; m_loadedRaw[id] = po; po->load(*this); return t; } template inline QSharedPointer PersistentPool::load(PersistentObjectId id) { if (id < 0) return QSharedPointer(); if (id < m_loaded.count()) { QSharedPointer obj = m_loaded.value(id); return obj.dynamicCast(); } m_loaded.resize(id + 1); const QSharedPointer t = T::create(); m_loaded[id] = t; PersistentObject * const po = t.data(); po->load(*this); return t; } } // namespace Internal } // namespace qbs #endif