forked from qt-creator/qt-creator
modernize-use-auto modernize-use-nullptr modernize-use-override modernize-use-using modernize-use-default-member-init modernize-use-equals-default modernize-use-transparent-functors Change-Id: Iebed22caa2e733d292f334e956e3d16b844e14e3 Reviewed-by: Orgad Shaneh <orgads@gmail.com> Reviewed-by: Tobias Hunger <tobias.hunger@qt.io>
471 lines
15 KiB
C++
471 lines
15 KiB
C++
/****************************************************************************
|
|
**
|
|
** 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 "buildenvironmentwidget.h"
|
|
#include "buildinfo.h"
|
|
#include "buildsteplist.h"
|
|
#include "kit.h"
|
|
#include "kitinformation.h"
|
|
#include "kitmanager.h"
|
|
#include "project.h"
|
|
#include "projectexplorer.h"
|
|
#include "projectexplorerconstants.h"
|
|
#include "projectmacroexpander.h"
|
|
#include "projecttree.h"
|
|
#include "target.h"
|
|
|
|
#include <coreplugin/idocument.h>
|
|
|
|
#include <utils/qtcassert.h>
|
|
#include <utils/macroexpander.h>
|
|
#include <utils/algorithm.h>
|
|
#include <utils/mimetypes/mimetype.h>
|
|
#include <utils/mimetypes/mimedatabase.h>
|
|
|
|
#include <QDebug>
|
|
|
|
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::updateCacheAndEmitEnvironmentChanged);
|
|
connect(this, &BuildConfiguration::environmentChanged,
|
|
this, &BuildConfiguration::emitBuildDirectoryChanged);
|
|
// Many macroexpanders are based on the current project, so they may change the environment:
|
|
connect(ProjectTree::instance(), &ProjectTree::currentProjectChanged,
|
|
this, &BuildConfiguration::updateCacheAndEmitEnvironmentChanged);
|
|
}
|
|
|
|
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<NamedWidget *> BuildConfiguration::createSubConfigWidgets()
|
|
{
|
|
return QList<NamedWidget *>() << new BuildEnvironmentWidget(this);
|
|
}
|
|
|
|
QList<Core::Id> 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::emitBuildDirectoryChanged()
|
|
{
|
|
if (buildDirectory() != m_lastEmmitedBuildDirectory) {
|
|
m_lastEmmitedBuildDirectory = buildDirectory();
|
|
emit buildDirectoryChanged();
|
|
}
|
|
}
|
|
|
|
Target *BuildConfiguration::target() const
|
|
{
|
|
return static_cast<Target *>(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<Utils::EnvironmentItem> BuildConfiguration::userEnvironmentChanges() const
|
|
{
|
|
return m_userEnvironmentChanges;
|
|
}
|
|
|
|
void BuildConfiguration::setUserEnvironmentChanges(const QList<Utils::EnvironmentItem> &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<IBuildConfigurationFactory *> 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 = nullptr;
|
|
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 = nullptr;
|
|
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<Core::Id> &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
|