/************************************************************************** ** ** This file is part of Qt Creator ** ** Copyright (c) 2011 Nokia Corporation and/or its subsidiary(-ies). ** ** Contact: Nokia Corporation (info@qt.nokia.com) ** ** ** GNU Lesser General Public License Usage ** ** 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, Nokia gives you certain additional ** rights. These rights are described in the Nokia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ** Other Usage ** ** Alternatively, this file may be used in accordance with the terms and ** conditions contained in a signed written agreement between you and Nokia. ** ** If you have questions regarding the use of this file, please contact ** Nokia at info@qt.nokia.com. ** **************************************************************************/ #include "toolchainmanager.h" #include "abi.h" #include "toolchain.h" #include #include #include #include #include #include #include static const char TOOLCHAIN_DATA_KEY[] = "ToolChain."; static const char TOOLCHAIN_COUNT_KEY[] = "ToolChain.Count"; static const char TOOLCHAIN_FILE_VERSION_KEY[] = "Version"; static const char DEFAULT_DEBUGGER_COUNT_KEY[] = "DefaultDebugger.Count"; static const char DEFAULT_DEBUGGER_ABI_KEY[] = "DefaultDebugger.Abi."; static const char DEFAULT_DEBUGGER_PATH_KEY[] = "DefaultDebugger.Path."; static const char TOOLCHAIN_FILENAME[] = "/toolChains.xml"; static QString settingsFileName() { ExtensionSystem::PluginManager *pm = ExtensionSystem::PluginManager::instance(); QFileInfo settingsLocation(pm->settings()->fileName()); return settingsLocation.absolutePath() + QLatin1String(TOOLCHAIN_FILENAME); } namespace ProjectExplorer { ToolChainManager *ToolChainManager::m_instance = 0; namespace Internal { // -------------------------------------------------------------------------- // ToolChainManagerPrivate // -------------------------------------------------------------------------- class ToolChainManagerPrivate { public: QList m_toolChains; QMap m_abiToDebugger; }; } // namespace Internal // -------------------------------------------------------------------------- // ToolChainManager // -------------------------------------------------------------------------- ToolChainManager *ToolChainManager::instance() { Q_ASSERT(m_instance); return m_instance; } ToolChainManager::ToolChainManager(QObject *parent) : QObject(parent), d(new Internal::ToolChainManagerPrivate) { Q_ASSERT(!m_instance); m_instance = this; connect(Core::ICore::instance(), SIGNAL(saveSettingsRequested()), this, SLOT(saveToolChains())); connect(this, SIGNAL(toolChainAdded(ProjectExplorer::ToolChain*)), this, SIGNAL(toolChainsChanged())); connect(this, SIGNAL(toolChainRemoved(ProjectExplorer::ToolChain*)), this, SIGNAL(toolChainsChanged())); connect(this, SIGNAL(toolChainUpdated(ProjectExplorer::ToolChain*)), this, SIGNAL(toolChainsChanged())); } void ToolChainManager::restoreToolChains() { // Restore SDK settings first restoreToolChains(Core::ICore::instance()->resourcePath() + QLatin1String("/Nokia") + QLatin1String(TOOLCHAIN_FILENAME), true); // Then auto detect ExtensionSystem::PluginManager *pm = ExtensionSystem::PluginManager::instance(); QList factories = pm->getObjects(); // Autodetect tool chains: foreach (ToolChainFactory *f, factories) { QList tcs = f->autoDetect(); foreach (ToolChain *tc, tcs) registerToolChain(tc); } // Then restore user settings restoreToolChains(settingsFileName(), false); } ToolChainManager::~ToolChainManager() { // Deregister tool chains QList copy = d->m_toolChains; foreach (ToolChain *tc, copy) deregisterToolChain(tc); delete d; m_instance = 0; } void ToolChainManager::saveToolChains() { PersistentSettingsWriter writer; writer.saveValue(QLatin1String(TOOLCHAIN_FILE_VERSION_KEY), 1); int count = 0; foreach (ToolChain *tc, d->m_toolChains) { if (!tc->isAutoDetected() && tc->isValid()) { QVariantMap tmp = tc->toMap(); if (tmp.isEmpty()) continue; writer.saveValue(QString::fromLatin1(TOOLCHAIN_DATA_KEY) + QString::number(count), tmp); ++count; } } writer.saveValue(QLatin1String(TOOLCHAIN_COUNT_KEY), count); writer.save(settingsFileName(), "QtCreatorToolChains", Core::ICore::instance()->mainWindow()); // Do not save default debuggers! Those are set by the SDK! } void ToolChainManager::restoreToolChains(const QString &fileName, bool autoDetected) { PersistentSettingsReader reader; if (!reader.load(fileName)) return; QVariantMap data = reader.restoreValues(); // Check version: int version = data.value(QLatin1String(TOOLCHAIN_FILE_VERSION_KEY), 0).toInt(); if (version < 1) return; // Read default debugger settings (if any) int count = data.value(QLatin1String(DEFAULT_DEBUGGER_COUNT_KEY)).toInt(); for (int i = 0; i < count; ++i) { const QString abiKey = QString::fromLatin1(DEFAULT_DEBUGGER_ABI_KEY) + QString::number(i); if (!data.contains(abiKey)) continue; const QString pathKey = QString::fromLatin1(DEFAULT_DEBUGGER_PATH_KEY) + QString::number(i); if (!data.contains(pathKey)) continue; d->m_abiToDebugger.insert(data.value(abiKey).toString(), data.value(pathKey).toString()); } ExtensionSystem::PluginManager *pm = ExtensionSystem::PluginManager::instance(); QList factories = pm->getObjects(); count = data.value(QLatin1String(TOOLCHAIN_COUNT_KEY), 0).toInt(); for (int i = 0; i < count; ++i) { const QString key = QString::fromLatin1(TOOLCHAIN_DATA_KEY) + QString::number(i); if (!data.contains(key)) break; const QVariantMap tcMap = data.value(key).toMap(); bool restored = false; foreach (ToolChainFactory *f, factories) { if (f->canRestore(tcMap)) { if (ToolChain *tc = f->restore(tcMap)) { tc->setAutoDetected(autoDetected); registerToolChain(tc); restored = true; break; } } } if (!restored) qWarning("Warning: Unable to restore manual tool chain '%s' stored in %s.", qPrintable(ToolChainFactory::idFromMap(tcMap)), qPrintable(QDir::toNativeSeparators(fileName))); } } QList ToolChainManager::toolChains() const { return d->m_toolChains; } QList ToolChainManager::findToolChains(const Abi &abi) const { QList result; foreach (ToolChain *tc, d->m_toolChains) { Abi targetAbi = tc->targetAbi(); if (targetAbi.isCompatibleWith(abi)) result.append(tc); } return result; } ToolChain *ToolChainManager::findToolChain(const QString &id) const { foreach (ToolChain *tc, d->m_toolChains) { if (tc->id() == id) return tc; } return 0; } QString ToolChainManager::defaultDebugger(const Abi &abi) const { return d->m_abiToDebugger.value(abi.toString()); } void ToolChainManager::notifyAboutUpdate(ProjectExplorer::ToolChain *tc) { if (!tc || !d->m_toolChains.contains(tc)) return; emit toolChainUpdated(tc); } bool ToolChainManager::registerToolChain(ToolChain *tc) { if (!tc || d->m_toolChains.contains(tc)) return true; foreach (ToolChain *current, d->m_toolChains) { if (*tc == *current) return false; } d->m_toolChains.append(tc); emit toolChainAdded(tc); return true; } void ToolChainManager::deregisterToolChain(ToolChain *tc) { if (!tc || !d->m_toolChains.contains(tc)) return; d->m_toolChains.removeOne(tc); emit toolChainRemoved(tc); delete tc; } } // namespace ProjectExplorer