/**************************************************************************** ** ** 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 "buildconfiguration.h" #include "buildinfo.h" #include "buildsteplist.h" #include "projectexplorer.h" #include "kitmanager.h" #include "target.h" #include "project.h" #include "kit.h" #include #include #include #include #include #include #include #include #include #include #include #include static const char BUILD_STEP_LIST_COUNT[] = "ProjectExplorer.BuildConfiguration.BuildStepListCount"; static const char BUILD_STEP_LIST_PREFIX[] = "ProjectExplorer.BuildConfiguration.BuildStepList."; static const char CLEAR_SYSTEM_ENVIRONMENT_KEY[] = "ProjectExplorer.BuildConfiguration.ClearSystemEnvironment"; static const char USER_ENVIRONMENT_CHANGES_KEY[] = "ProjectExplorer.BuildConfiguration.UserEnvironmentChanges"; static const char BUILDDIRECTORY_KEY[] = "ProjectExplorer.BuildConfiguration.BuildDirectory"; namespace ProjectExplorer { BuildConfiguration::BuildConfiguration(Target *target, Core::Id id) : ProjectConfiguration(target, id) { Utils::MacroExpander *expander = macroExpander(); expander->setDisplayName(tr("Build Settings")); expander->setAccumulating(true); expander->registerSubProvider([target] { return target->macroExpander(); }); expander->registerVariable("buildDir", tr("Build directory"), [this] { return buildDirectory().toUserOutput(); }); expander->registerVariable(Constants::VAR_CURRENTBUILD_NAME, tr("Name of current build"), [this] { return displayName(); }, false); expander->registerPrefix(Constants::VAR_CURRENTBUILD_ENV, tr("Variables in the current build environment"), [this](const QString &var) { return environment().value(var); }); updateCacheAndEmitEnvironmentChanged(); connect(target, &Target::kitChanged, this, &BuildConfiguration::handleKitUpdate); connect(this, &BuildConfiguration::environmentChanged, this, &BuildConfiguration::emitBuildDirectoryChanged); } Utils::FileName BuildConfiguration::buildDirectory() const { const QString path = macroExpander()->expand(QDir::cleanPath(environment().expandVariables(m_buildDirectory.toString()))); return Utils::FileName::fromString(QDir::cleanPath(QDir(target()->project()->projectDirectory().toString()).absoluteFilePath(path))); } Utils::FileName BuildConfiguration::rawBuildDirectory() const { return m_buildDirectory; } void BuildConfiguration::setBuildDirectory(const Utils::FileName &dir) { if (dir == m_buildDirectory) return; m_buildDirectory = dir; emitBuildDirectoryChanged(); } void BuildConfiguration::initialize(const BuildInfo *info) { setDisplayName(info->displayName); setDefaultDisplayName(info->displayName); setBuildDirectory(info->buildDirectory); m_stepLists.append(new BuildStepList(this, Constants::BUILDSTEPS_BUILD)); m_stepLists.append(new BuildStepList(this, Constants::BUILDSTEPS_CLEAN)); } QList BuildConfiguration::createSubConfigWidgets() { return QList() << new BuildEnvironmentWidget(this); } QList BuildConfiguration::knownStepLists() const { return Utils::transform(m_stepLists, &BuildStepList::id); } BuildStepList *BuildConfiguration::stepList(Core::Id id) const { return Utils::findOrDefault(m_stepLists, Utils::equal(&BuildStepList::id, id)); } QVariantMap BuildConfiguration::toMap() const { QVariantMap map(ProjectConfiguration::toMap()); map.insert(QLatin1String(CLEAR_SYSTEM_ENVIRONMENT_KEY), m_clearSystemEnvironment); map.insert(QLatin1String(USER_ENVIRONMENT_CHANGES_KEY), Utils::EnvironmentItem::toStringList(m_userEnvironmentChanges)); map.insert(QLatin1String(BUILDDIRECTORY_KEY), m_buildDirectory.toString()); map.insert(QLatin1String(BUILD_STEP_LIST_COUNT), m_stepLists.count()); for (int i = 0; i < m_stepLists.count(); ++i) map.insert(QLatin1String(BUILD_STEP_LIST_PREFIX) + QString::number(i), m_stepLists.at(i)->toMap()); return map; } bool BuildConfiguration::fromMap(const QVariantMap &map) { m_clearSystemEnvironment = map.value(QLatin1String(CLEAR_SYSTEM_ENVIRONMENT_KEY)).toBool(); m_userEnvironmentChanges = Utils::EnvironmentItem::fromStringList(map.value(QLatin1String(USER_ENVIRONMENT_CHANGES_KEY)).toStringList()); m_buildDirectory = Utils::FileName::fromString(map.value(QLatin1String(BUILDDIRECTORY_KEY)).toString()); updateCacheAndEmitEnvironmentChanged(); qDeleteAll(m_stepLists); m_stepLists.clear(); int maxI = map.value(QLatin1String(BUILD_STEP_LIST_COUNT), 0).toInt(); for (int i = 0; i < maxI; ++i) { QVariantMap data = map.value(QLatin1String(BUILD_STEP_LIST_PREFIX) + QString::number(i)).toMap(); if (data.isEmpty()) { qWarning() << "No data for build step list" << i << "found!"; continue; } auto list = new BuildStepList(this, idFromMap(data)); if (!list->fromMap(data)) { qWarning() << "Failed to restore build step list" << i; delete list; return false; } m_stepLists.append(list); } // We currently assume there to be at least a clean and build list! QTC_CHECK(knownStepLists().contains(Core::Id(Constants::BUILDSTEPS_BUILD))); QTC_CHECK(knownStepLists().contains(Core::Id(Constants::BUILDSTEPS_CLEAN))); return ProjectConfiguration::fromMap(map); } void BuildConfiguration::updateCacheAndEmitEnvironmentChanged() { Utils::Environment env = baseEnvironment(); env.modify(userEnvironmentChanges()); if (env == m_cachedEnvironment) return; m_cachedEnvironment = env; emit environmentChanged(); // might trigger buildDirectoryChanged signal! } void BuildConfiguration::handleKitUpdate() { updateCacheAndEmitEnvironmentChanged(); } void BuildConfiguration::emitBuildDirectoryChanged() { if (buildDirectory() != m_lastEmmitedBuildDirectory) { m_lastEmmitedBuildDirectory = buildDirectory(); emit buildDirectoryChanged(); } } Target *BuildConfiguration::target() const { return static_cast(parent()); } Project *BuildConfiguration::project() const { return target()->project(); } Utils::Environment BuildConfiguration::baseEnvironment() const { Utils::Environment result; if (useSystemEnvironment()) result = Utils::Environment::systemEnvironment(); addToEnvironment(result); target()->kit()->addToEnvironment(result); return result; } QString BuildConfiguration::baseEnvironmentText() const { if (useSystemEnvironment()) return tr("System Environment"); else return tr("Clean Environment"); } Utils::Environment BuildConfiguration::environment() const { return m_cachedEnvironment; } void BuildConfiguration::setUseSystemEnvironment(bool b) { if (useSystemEnvironment() == b) return; m_clearSystemEnvironment = !b; updateCacheAndEmitEnvironmentChanged(); } void BuildConfiguration::addToEnvironment(Utils::Environment &env) const { Q_UNUSED(env); } bool BuildConfiguration::useSystemEnvironment() const { return !m_clearSystemEnvironment; } QList BuildConfiguration::userEnvironmentChanges() const { return m_userEnvironmentChanges; } void BuildConfiguration::setUserEnvironmentChanges(const QList &diff) { if (m_userEnvironmentChanges == diff) return; m_userEnvironmentChanges = diff; updateCacheAndEmitEnvironmentChanged(); } bool BuildConfiguration::isEnabled() const { return true; } QString BuildConfiguration::disabledReason() const { return QString(); } bool BuildConfiguration::regenerateBuildFiles(Node *node) { Q_UNUSED(node); return false; } QString BuildConfiguration::buildTypeName(BuildConfiguration::BuildType type) { switch (type) { case ProjectExplorer::BuildConfiguration::Debug: return QLatin1String("debug"); case ProjectExplorer::BuildConfiguration::Profile: return QLatin1String("profile"); case ProjectExplorer::BuildConfiguration::Release: return QLatin1String("release"); case ProjectExplorer::BuildConfiguration::Unknown: // fallthrough default: return QLatin1String("unknown"); } } bool BuildConfiguration::isActive() const { return target()->isActive() && target()->activeBuildConfiguration() == this; } /*! * Helper function that prepends the directory containing the C++ toolchain to * PATH. This is used to in build configurations targeting broken build systems * to provide hints about which compiler to use. */ void BuildConfiguration::prependCompilerPathToEnvironment(Utils::Environment &env) const { return prependCompilerPathToEnvironment(target()->kit(), env); } void BuildConfiguration::prependCompilerPathToEnvironment(Kit *k, Utils::Environment &env) { const ToolChain *tc = ToolChainKitInformation::toolChain(k, ProjectExplorer::Constants::CXX_LANGUAGE_ID); if (!tc) return; const Utils::FileName compilerDir = tc->compilerCommand().parentDir(); if (!compilerDir.isEmpty()) env.prependOrSetPath(compilerDir.toString()); } /// // IBuildConfigurationFactory /// static QList g_buildConfigurationFactories; IBuildConfigurationFactory::IBuildConfigurationFactory() { g_buildConfigurationFactories.append(this); } IBuildConfigurationFactory::~IBuildConfigurationFactory() { g_buildConfigurationFactories.removeOne(this); } int IBuildConfigurationFactory::priority(const Target *parent) const { return canHandle(parent) ? m_basePriority : -1; } bool IBuildConfigurationFactory::supportsTargetDeviceType(Core::Id id) const { if (m_supportedTargetDeviceTypes.isEmpty()) return true; return m_supportedTargetDeviceTypes.contains(id); } int IBuildConfigurationFactory::priority(const Kit *k, const QString &projectPath) const { QTC_ASSERT(!m_supportedProjectMimeTypeName.isEmpty(), return -1); if (k && Utils::mimeTypeForFile(projectPath).matchesName(m_supportedProjectMimeTypeName) && supportsTargetDeviceType(DeviceTypeKitInformation::deviceTypeId(k))) { return m_basePriority; } return -1; } // setup IBuildConfigurationFactory *IBuildConfigurationFactory::find(const Kit *k, const QString &projectPath) { IBuildConfigurationFactory *factory = 0; int priority = -1; for (IBuildConfigurationFactory *i : g_buildConfigurationFactories) { int iPriority = i->priority(k, projectPath); if (iPriority > priority) { factory = i; priority = iPriority; } } return factory; } // create IBuildConfigurationFactory * IBuildConfigurationFactory::find(Target *parent) { IBuildConfigurationFactory *factory = 0; int priority = -1; for (IBuildConfigurationFactory *i : g_buildConfigurationFactories) { int iPriority = i->priority(parent); if (iPriority > priority) { factory = i; priority = iPriority; } } return factory; } void IBuildConfigurationFactory::setSupportedProjectType(Core::Id id) { m_supportedProjectType = id; } void IBuildConfigurationFactory::setSupportedProjectMimeTypeName(const QString &mimeTypeName) { m_supportedProjectMimeTypeName = mimeTypeName; } void IBuildConfigurationFactory::setSupportedTargetDeviceTypes(const QList &ids) { m_supportedTargetDeviceTypes = ids; } void IBuildConfigurationFactory::setBasePriority(int basePriority) { m_basePriority = basePriority; } bool IBuildConfigurationFactory::canHandle(const Target *target) const { if (m_supportedProjectType.isValid() && m_supportedProjectType != target->project()->id()) return false; if (containsType(target->project()->projectIssues(target->kit()), Task::TaskType::Error)) return false; if (!supportsTargetDeviceType(DeviceTypeKitInformation::deviceTypeId(target->kit()))) return false; return true; } BuildConfiguration *IBuildConfigurationFactory::create(Target *parent, const BuildInfo *info) const { if (!canHandle(parent)) return nullptr; QTC_ASSERT(m_creator, return nullptr); BuildConfiguration *bc = m_creator(parent); if (!bc) return nullptr; bc->initialize(info); return bc; } BuildConfiguration *IBuildConfigurationFactory::restore(Target *parent, const QVariantMap &map) { IBuildConfigurationFactory *factory = nullptr; int priority = -1; for (IBuildConfigurationFactory *i : g_buildConfigurationFactories) { if (!i->canHandle(parent)) continue; const Core::Id id = idFromMap(map); if (!id.name().startsWith(i->m_buildConfigId.name())) continue; int iPriority = i->priority(parent); if (iPriority > priority) { factory = i; priority = iPriority; } } if (!factory) return nullptr; QTC_ASSERT(factory->m_creator, return nullptr); BuildConfiguration *bc = factory->m_creator(parent); QTC_ASSERT(bc, return nullptr); if (!bc->fromMap(map)) { delete bc; bc = nullptr; } return bc; } BuildConfiguration *IBuildConfigurationFactory::clone(Target *parent, const BuildConfiguration *source) { return restore(parent, source->toMap()); } } // namespace ProjectExplorer