/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of Qt Creator. ** ** 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 https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3 as published by the Free Software ** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-3.0.html. ** ****************************************************************************/ #include "kitmanager.h" #include "devicesupport/idevicefactory.h" #include "kit.h" #include "kitfeatureprovider.h" #include "kitmanagerconfigwidget.h" #include "project.h" #include "projectexplorerconstants.h" #include "task.h" #include #include #include #include #include #include #include #include using namespace Core; using namespace Utils; using namespace ProjectExplorer::Internal; namespace ProjectExplorer { namespace Internal { const char KIT_DATA_KEY[] = "Profile."; const char KIT_COUNT_KEY[] = "Profile.Count"; const char KIT_FILE_VERSION_KEY[] = "Version"; const char KIT_DEFAULT_KEY[] = "Profile.Default"; const char KIT_FILENAME[] = "/qtcreator/profiles.xml"; static FileName settingsFileName() { QFileInfo settingsLocation(ICore::settings()->fileName()); return FileName::fromString(settingsLocation.absolutePath() + QLatin1String(KIT_FILENAME)); } // -------------------------------------------------------------------------- // KitManagerPrivate: // -------------------------------------------------------------------------- class KitManagerPrivate { public: KitManagerPrivate(); ~KitManagerPrivate(); Kit *m_defaultKit; bool m_initialized; QList m_informationList; QList m_kitList; PersistentSettingsWriter *m_writer; }; KitManagerPrivate::KitManagerPrivate() : m_defaultKit(0), m_initialized(false), m_writer(0) { } KitManagerPrivate::~KitManagerPrivate() { qDeleteAll(m_informationList); delete m_writer; } } // namespace Internal // -------------------------------------------------------------------------- // KitManager: // -------------------------------------------------------------------------- static Internal::KitManagerPrivate *d; static KitManager *m_instance; KitManager *KitManager::instance() { return m_instance; } KitManager::KitManager(QObject *parent) : QObject(parent) { d = new KitManagerPrivate; QTC_CHECK(!m_instance); m_instance = this; connect(ICore::instance(), &ICore::saveSettingsRequested, this, &KitManager::saveKits); connect(this, &KitManager::kitAdded, this, &KitManager::kitsChanged); connect(this, &KitManager::kitRemoved, this, &KitManager::kitsChanged); connect(this, &KitManager::kitUpdated, this, &KitManager::kitsChanged); } void KitManager::restoreKits() { QTC_ASSERT(!d->m_initialized, return); static bool initializing = false; if (initializing) // kits will call kits() to check their display names, which will trigger another // call to restoreKits, which ... return; initializing = true; QList kitsToRegister; QList kitsToValidate; QList kitsToCheck; QList sdkKits; // read all kits from SDK QFileInfo systemSettingsFile(ICore::settings(QSettings::SystemScope)->fileName()); QFileInfo kitFile(systemSettingsFile.absolutePath() + QLatin1String(KIT_FILENAME)); if (kitFile.exists()) { KitList system = restoreKits(FileName(kitFile)); // make sure we mark these as autodetected and run additional setup logic foreach (Kit *k, system.kits) { k->setAutoDetected(true); k->setSdkProvided(true); k->makeSticky(); } // SDK kits are always considered to be up for validation since they might have been // extended with additional information by creator in the meantime: kitsToValidate = system.kits; } // read all kits from user file KitList userKits; FileName userSettingsFile(settingsFileName()); if (userSettingsFile.exists()) userKits = restoreKits(userSettingsFile); foreach (Kit *k, userKits.kits) { if (k->isSdkProvided()) kitsToCheck.append(k); else kitsToRegister.append(k); } Kit *toStore = 0; foreach (Kit *current, kitsToValidate) { toStore = current; toStore->setup(); // Make sure all kitinformation are properly set up before merging them // with the information from the user settings file // Check whether we had this kit stored and prefer the stored one: for (int i = 0; i < kitsToCheck.count(); ++i) { if (kitsToCheck.at(i)->id() == current->id()) { toStore = kitsToCheck.at(i); kitsToCheck.removeAt(i); // Overwrite settings that the SDK sets to those values: foreach (const KitInformation *ki, kitInformation()) { // Copy sticky settings over: if (current->isSticky(ki->id())) { toStore->setValue(ki->id(), current->value(ki->id())); toStore->setSticky(ki->id(), true); } } delete current; break; } } addKit(toStore); sdkKits << toStore; } // Delete all loaded autodetected kits that were not rediscovered: foreach (Kit *k, kitsToCheck) delete k; kitsToCheck.clear(); // Store manual kits foreach (Kit *k, kitsToRegister) addKit(k); if (kits().isEmpty()) { Kit *defaultKit = new Kit; // One kit using default values defaultKit->setUnexpandedDisplayName(tr("Desktop")); defaultKit->setSdkProvided(false); defaultKit->setAutoDetected(false); defaultKit->setIconPath(FileName::fromLatin1(ProjectExplorer::Constants::DESKTOP_DEVICE_ICON)); defaultKit->setup(); addKit(defaultKit); setDefaultKit(defaultKit); } Kit *k = find(userKits.defaultKit); if (!k && !defaultKit()) k = Utils::findOrDefault(kitsToRegister + sdkKits, &Kit::isValid); if (k) setDefaultKit(k); d->m_writer = new PersistentSettingsWriter(settingsFileName(), QLatin1String("QtCreatorProfiles")); d->m_initialized = true; emit kitsLoaded(); emit kitsChanged(); } KitManager::~KitManager() { foreach (Kit *k, d->m_kitList) delete k; d->m_kitList.clear(); delete d; m_instance = 0; } void KitManager::saveKits() { if (!d->m_writer) // ignore save requests while we are not initialized. return; QVariantMap data; data.insert(QLatin1String(KIT_FILE_VERSION_KEY), 1); int count = 0; foreach (Kit *k, kits()) { QVariantMap tmp = k->toMap(); if (tmp.isEmpty()) continue; data.insert(QString::fromLatin1(KIT_DATA_KEY) + QString::number(count), tmp); ++count; } data.insert(QLatin1String(KIT_COUNT_KEY), count); data.insert(QLatin1String(KIT_DEFAULT_KEY), d->m_defaultKit ? QString::fromLatin1(d->m_defaultKit->id().name()) : QString()); d->m_writer->save(data, ICore::mainWindow()); } static bool isLoaded() { return d->m_initialized; } static bool greaterPriority(KitInformation *a, KitInformation *b) { return a->priority() > b->priority(); } void KitManager::registerKitInformation(KitInformation *ki) { QTC_CHECK(!isLoaded()); QTC_ASSERT(!d->m_informationList.contains(ki), return); QList::iterator it = qLowerBound(d->m_informationList.begin(), d->m_informationList.end(), ki, greaterPriority); d->m_informationList.insert(it, ki); if (!isLoaded()) return; foreach (Kit *k, kits()) { if (!k->hasValue(ki->id())) k->setValue(ki->id(), ki->defaultValue(k)); else ki->fix(k); } return; } void KitManager::deregisterKitInformation(KitInformation *ki) { QTC_CHECK(d->m_informationList.contains(ki)); d->m_informationList.removeOne(ki); delete ki; } QSet KitManager::supportedPlatforms() { QSet platforms; foreach (const Kit *k, kits()) platforms.unite(k->supportedPlatforms()); return platforms; } QSet KitManager::availableFeatures(Core::Id platformId) { QSet features; foreach (const Kit *k, kits()) { if (!k->supportedPlatforms().contains(platformId)) continue; features.unite(k->availableFeatures()); } return features; } QList KitManager::sortKits(const QList kits) { // This method was added to delay the sorting of kits as long as possible. // Since the displayName can contain variables it can be costly (e.g. involve // calling executables to find version information, etc.) to call that // method! // Avoid lots of potentially expensive calls to Kit::displayName(): QList > sortList = Utils::transform(kits, [](Kit *k) { return qMakePair(k->displayName(), k); }); Utils::sort(sortList, [](const QPair &a, const QPair &b) -> bool { if (a.first == b.first) return a.second < b.second; return a. first < b.first; }); return Utils::transform(sortList, [](const QPair &a) { return a.second; }); } KitManager::KitList KitManager::restoreKits(const FileName &fileName) { KitList result; PersistentSettingsReader reader; if (!reader.load(fileName)) { qWarning("Warning: Failed to read \"%s\", cannot restore kits!", qPrintable(fileName.toUserOutput())); return result; } QVariantMap data = reader.restoreValues(); // Check version: int version = data.value(QLatin1String(KIT_FILE_VERSION_KEY), 0).toInt(); if (version < 1) { qWarning("Warning: Kit file version %d not supported, cannot restore kits!", version); return result; } const int count = data.value(QLatin1String(KIT_COUNT_KEY), 0).toInt(); for (int i = 0; i < count; ++i) { const QString key = QString::fromLatin1(KIT_DATA_KEY) + QString::number(i); if (!data.contains(key)) break; const QVariantMap stMap = data.value(key).toMap(); Kit *k = new Kit(stMap); if (k->id().isValid()) { result.kits.append(k); } else { // If the Id is broken, then do not trust the rest of the data either. delete k; qWarning("Warning: Unable to restore kits stored in %s at position %d.", qPrintable(fileName.toUserOutput()), i); } } const Id id = Id::fromSetting(data.value(QLatin1String(KIT_DEFAULT_KEY))); if (!id.isValid()) return result; foreach (Kit *k, result.kits) { if (k->id() == id) { result.defaultKit = id; break; } } return result; } QList KitManager::kits() { return d->m_kitList; } QList KitManager::matchingKits(const KitMatcher &matcher) { QList result; foreach (Kit *k, d->m_kitList) if (matcher.matches(k)) result.append(k); return result; } Kit *KitManager::find(Id id) { if (!id.isValid()) return 0; return Utils::findOrDefault(kits(), Utils::equal(&Kit::id, id)); } Kit *KitManager::find(const KitMatcher &matcher) { return Utils::findOrDefault(d->m_kitList, [&matcher](Kit *k) { return matcher.matches(k); }); } Kit *KitManager::defaultKit() { return d->m_defaultKit; } QList KitManager::kitInformation() { return d->m_informationList; } KitManagerConfigWidget *KitManager::createConfigWidget(Kit *k) { KitManagerConfigWidget *result = new KitManagerConfigWidget(k); foreach (KitInformation *ki, kitInformation()) result->addConfigWidget(ki->createConfigWidget(result->workingCopy())); result->updateVisibility(); return result; } void KitManager::deleteKit(Kit *k) { QTC_ASSERT(!KitManager::kits().contains(k), return); delete k; } void KitManager::notifyAboutUpdate(Kit *k) { if (!k || !isLoaded()) return; if (d->m_kitList.contains(k)) emit m_instance->kitUpdated(k); else emit m_instance->unmanagedKitUpdated(k); } bool KitManager::registerKit(Kit *k) { QTC_ASSERT(isLoaded(), return false); if (!k) return true; QTC_ASSERT(k->id().isValid(), return false); if (kits().contains(k)) return false; // make sure we have all the information in our kits: m_instance->addKit(k); if (!d->m_defaultKit || (!d->m_defaultKit->isValid() && k->isValid())) setDefaultKit(k); emit m_instance->kitAdded(k); return true; } void KitManager::deregisterKit(Kit *k) { if (!k || !kits().contains(k)) return; d->m_kitList.removeOne(k); if (defaultKit() == k) { QList stList = kits(); Kit *newDefault = 0; foreach (Kit *cur, stList) { if (cur->isValid()) { newDefault = cur; break; } } setDefaultKit(newDefault); } emit m_instance->kitRemoved(k); delete k; } void KitManager::setDefaultKit(Kit *k) { if (defaultKit() == k) return; if (k && !kits().contains(k)) return; d->m_defaultKit = k; emit m_instance->defaultkitChanged(); } void KitManager::addKit(Kit *k) { if (!k) return; { KitGuard g(k); foreach (KitInformation *ki, d->m_informationList) { if (!k->hasValue(ki->id())) k->setValue(ki->id(), ki->defaultValue(k)); else ki->fix(k); } } d->m_kitList.append(k); } void KitInformation::addToEnvironment(const Kit *k, Environment &env) const { Q_UNUSED(k); Q_UNUSED(env); } IOutputParser *KitInformation::createOutputParser(const Kit *k) const { Q_UNUSED(k); return 0; } QString KitInformation::displayNamePostfix(const Kit *k) const { Q_UNUSED(k); return QString(); } QSet KitInformation::supportedPlatforms(const Kit *k) const { Q_UNUSED(k); return QSet(); } QSet KitInformation::availableFeatures(const Kit *k) const { Q_UNUSED(k); return QSet(); } void KitInformation::addToMacroExpander(Kit *k, MacroExpander *expander) const { Q_UNUSED(k); Q_UNUSED(expander); } void KitInformation::notifyAboutUpdate(Kit *k) { if (k) k->kitUpdated(); } // -------------------------------------------------------------------- // KitFeatureProvider: // -------------------------------------------------------------------- // This FeatureProvider maps the platforms onto the device types. QSet KitFeatureProvider::availableFeatures(Id id) const { return KitManager::availableFeatures(id); } QSet KitFeatureProvider::availablePlatforms() const { return KitManager::supportedPlatforms(); } QString KitFeatureProvider::displayNameForPlatform(Id id) const { foreach (IDeviceFactory *f, ExtensionSystem::PluginManager::getObjects()) { const QString dn = f->displayNameForId(id); if (!dn.isEmpty()) return dn; } return QString(); } } // namespace ProjectExplorer