/**************************************************************************** ** ** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** 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 Digia. For licensing terms and ** conditions see http://qt.digia.com/licensing. For further information ** use the contact form at http://qt.digia.com/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 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 2.1 requirements ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, Digia gives you certain additional ** rights. These rights are described in the Digia Qt 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 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