forked from qt-creator/qt-creator
Move special CMake environment modification to CMakeBuildConfiguration. Change-Id: I27f886db48d8c036f2f77f5ae8f239b405cb5721 Reviewed-by: Christian Kandeler <christian.kandeler@qt.io>
735 lines
28 KiB
C++
735 lines
28 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 "cmakebuildconfiguration.h"
|
|
|
|
#include "builddirmanager.h"
|
|
#include "cmakebuildstep.h"
|
|
#include "cmakeconfigitem.h"
|
|
#include "cmakekitinformation.h"
|
|
#include "cmakeprojectconstants.h"
|
|
#include "cmakebuildsettingswidget.h"
|
|
#include "cmakeprojectmanager.h"
|
|
#include "cmakeprojectnodes.h"
|
|
|
|
#include <android/androidconstants.h>
|
|
|
|
#include <coreplugin/icore.h>
|
|
|
|
#include <projectexplorer/buildinfo.h>
|
|
#include <projectexplorer/buildmanager.h>
|
|
#include <projectexplorer/buildsteplist.h>
|
|
#include <projectexplorer/kit.h>
|
|
#include <projectexplorer/kitinformation.h>
|
|
#include <projectexplorer/projectexplorer.h>
|
|
#include <projectexplorer/projectexplorerconstants.h>
|
|
#include <projectexplorer/projectmacroexpander.h>
|
|
#include <projectexplorer/target.h>
|
|
|
|
#include <qtsupport/qtkitinformation.h>
|
|
|
|
#include <utils/algorithm.h>
|
|
#include <utils/mimetypes/mimedatabase.h>
|
|
#include <utils/qtcassert.h>
|
|
#include <utils/qtcprocess.h>
|
|
|
|
using namespace ProjectExplorer;
|
|
using namespace Utils;
|
|
|
|
namespace CMakeProjectManager {
|
|
|
|
class CMakeExtraBuildInfo
|
|
{
|
|
public:
|
|
QString sourceDirectory;
|
|
CMakeConfig configuration;
|
|
};
|
|
|
|
} // namespace CMakeProjectManager
|
|
|
|
Q_DECLARE_METATYPE(CMakeProjectManager::CMakeExtraBuildInfo)
|
|
|
|
|
|
namespace CMakeProjectManager {
|
|
namespace Internal {
|
|
|
|
const char CONFIGURATION_KEY[] = "CMake.Configuration";
|
|
|
|
CMakeBuildConfiguration::CMakeBuildConfiguration(Target *parent, Core::Id id)
|
|
: BuildConfiguration(parent, id)
|
|
, m_buildDirManager(qobject_cast<CMakeProject *>(parent->project()))
|
|
{
|
|
setBuildDirectory(shadowBuildDirectory(project()->projectFilePath(),
|
|
target()->kit(),
|
|
displayName(),
|
|
BuildConfiguration::Unknown));
|
|
connect(project(), &Project::parsingFinished, this, &BuildConfiguration::enabledChanged);
|
|
|
|
// BuildDirManager:
|
|
connect(&m_buildDirManager, &BuildDirManager::requestReparse, this, [this](int options) {
|
|
if (isActive())
|
|
project()->requestReparse(options);
|
|
});
|
|
connect(&m_buildDirManager,
|
|
&BuildDirManager::dataAvailable,
|
|
this,
|
|
&CMakeBuildConfiguration::handleParsingSucceeded);
|
|
connect(&m_buildDirManager, &BuildDirManager::errorOccured, this, [this](const QString &msg) {
|
|
setError(msg);
|
|
QString errorMessage;
|
|
setConfigurationFromCMake(m_buildDirManager.takeCMakeConfiguration(errorMessage));
|
|
// ignore errorMessage here, we already got one.
|
|
project()->handleParsingError(this);
|
|
});
|
|
connect(&m_buildDirManager, &BuildDirManager::parsingStarted, this, [this]() {
|
|
clearError(CMakeBuildConfiguration::ForceEnabledChanged::True);
|
|
});
|
|
|
|
// Kit changed:
|
|
connect(KitManager::instance(), &KitManager::kitUpdated, this, [this](Kit *k) {
|
|
if (k != target()->kit())
|
|
return; // not for us...
|
|
|
|
// Build configuration has not changed, but Kit settings might have:
|
|
// reparse and check the configuration, independent of whether the reader has changed
|
|
m_buildDirManager.setParametersAndRequestParse(BuildDirParameters(this),
|
|
BuildDirManager::REPARSE_CHECK_CONFIGURATION,
|
|
BuildDirManager::REPARSE_CHECK_CONFIGURATION);
|
|
});
|
|
|
|
// Became active/inactive:
|
|
connect(project(), &Project::activeBuildConfigurationChanged, this, [this]() {
|
|
if (isActive()) {
|
|
// Build configuration has switched:
|
|
// * Check configuration if reader changes due to it not existing yet:-)
|
|
// * run cmake without configuration arguments if the reader stays
|
|
m_buildDirManager
|
|
.setParametersAndRequestParse(BuildDirParameters(this),
|
|
BuildDirManager::REPARSE_CHECK_CONFIGURATION,
|
|
BuildDirManager::REPARSE_CHECK_CONFIGURATION);
|
|
} else {
|
|
m_buildDirManager.stopParsingAndClearState();
|
|
}
|
|
});
|
|
|
|
// BuildConfiguration changed:
|
|
connect(this, &CMakeBuildConfiguration::environmentChanged, this, [this]() {
|
|
if (isActive()) {
|
|
// The environment on our BC has changed:
|
|
// * Error out if the reader updates, cannot happen since all BCs share a target/kit.
|
|
// * run cmake without configuration arguments if the reader stays
|
|
m_buildDirManager.setParametersAndRequestParse(
|
|
BuildDirParameters(this),
|
|
BuildDirManager::REPARSE_CHECK_CONFIGURATION, // server-mode might need a restart...
|
|
BuildDirManager::REPARSE_CHECK_CONFIGURATION);
|
|
}
|
|
});
|
|
connect(this, &CMakeBuildConfiguration::buildDirectoryChanged, this, [this]() {
|
|
if (isActive()) {
|
|
// The build directory of our BC has changed:
|
|
// * Error out if the reader updates, cannot happen since all BCs share a target/kit.
|
|
// * run cmake without configuration arguments if the reader stays
|
|
// If no configuration exists, then the arguments will get added automatically by
|
|
// the reader.
|
|
m_buildDirManager
|
|
.setParametersAndRequestParse(BuildDirParameters(this),
|
|
BuildDirManager::REPARSE_FAIL,
|
|
BuildDirManager::REPARSE_CHECK_CONFIGURATION);
|
|
}
|
|
});
|
|
connect(this, &CMakeBuildConfiguration::configurationForCMakeChanged, this, [this]() {
|
|
if (isActive()) {
|
|
// The CMake configuration has changed on our BC:
|
|
// * Error out if the reader updates, cannot happen since all BCs share a target/kit.
|
|
// * run cmake with configuration arguments if the reader stays
|
|
m_buildDirManager
|
|
.setParametersAndRequestParse(BuildDirParameters(this),
|
|
BuildDirManager::REPARSE_FAIL,
|
|
BuildDirManager::REPARSE_FORCE_CONFIGURATION);
|
|
}
|
|
});
|
|
}
|
|
|
|
void CMakeBuildConfiguration::initialize(const BuildInfo &info)
|
|
{
|
|
BuildConfiguration::initialize(info);
|
|
|
|
BuildStepList *buildSteps = stepList(ProjectExplorer::Constants::BUILDSTEPS_BUILD);
|
|
buildSteps->appendStep(Constants::CMAKE_BUILD_STEP_ID);
|
|
|
|
if (DeviceTypeKitAspect::deviceTypeId(target()->kit())
|
|
== Android::Constants::ANDROID_DEVICE_TYPE) {
|
|
buildSteps->appendStep(Android::Constants::ANDROID_BUILD_APK_ID);
|
|
const auto &bs = buildSteps->steps().constLast();
|
|
m_initialConfiguration.prepend(CMakeProjectManager::CMakeConfigItem{"ANDROID_NATIVE_API_LEVEL",
|
|
CMakeProjectManager::CMakeConfigItem::Type::STRING,
|
|
"Android native API level",
|
|
bs->data(Android::Constants::AndroidNdkPlatform).toString().toUtf8()});
|
|
auto ndkLocation = bs->data(Android::Constants::NdkLocation).value<FilePath>();
|
|
m_initialConfiguration.prepend(CMakeProjectManager::CMakeConfigItem{"ANDROID_NDK",
|
|
CMakeProjectManager::CMakeConfigItem::Type::PATH,
|
|
"Android NDK PATH",
|
|
ndkLocation.toUserOutput().toUtf8()});
|
|
m_initialConfiguration.prepend(CMakeProjectManager::CMakeConfigItem{"CMAKE_TOOLCHAIN_FILE",
|
|
CMakeProjectManager::CMakeConfigItem::Type::PATH,
|
|
"Android CMake toolchain file",
|
|
ndkLocation.pathAppended("build/cmake/android.toolchain.cmake").toUserOutput().toUtf8()});
|
|
m_initialConfiguration.prepend(CMakeProjectManager::CMakeConfigItem{"ANDROID_ABI",
|
|
CMakeProjectManager::CMakeConfigItem::Type::STRING,
|
|
"Android ABI",
|
|
bs->data(Android::Constants::AndroidABI).toString().toUtf8()});
|
|
m_initialConfiguration.prepend(CMakeProjectManager::CMakeConfigItem{"ANDROID_STL",
|
|
CMakeProjectManager::CMakeConfigItem::Type::STRING,
|
|
"Android STL",
|
|
"c++_shared"});
|
|
m_initialConfiguration.prepend(CMakeProjectManager::CMakeConfigItem{"CMAKE_FIND_ROOT_PATH_MODE_PROGRAM", "BOTH"});
|
|
m_initialConfiguration.prepend(CMakeProjectManager::CMakeConfigItem{"CMAKE_FIND_ROOT_PATH_MODE_LIBRARY", "BOTH"});
|
|
m_initialConfiguration.prepend(CMakeProjectManager::CMakeConfigItem{"CMAKE_FIND_ROOT_PATH_MODE_INCLUDE", "BOTH"});
|
|
m_initialConfiguration.prepend(CMakeProjectManager::CMakeConfigItem{"CMAKE_FIND_ROOT_PATH_MODE_PACKAGE", "BOTH"});
|
|
}
|
|
|
|
BuildStepList *cleanSteps = stepList(ProjectExplorer::Constants::BUILDSTEPS_CLEAN);
|
|
cleanSteps->appendStep(Constants::CMAKE_BUILD_STEP_ID);
|
|
|
|
if (info.buildDirectory.isEmpty()) {
|
|
auto project = target()->project();
|
|
setBuildDirectory(CMakeBuildConfiguration::shadowBuildDirectory(project->projectFilePath(),
|
|
target()->kit(),
|
|
info.displayName, info.buildType));
|
|
}
|
|
auto extraInfo = info.extraInfo.value<CMakeExtraBuildInfo>();
|
|
setConfigurationForCMake(extraInfo.configuration);
|
|
}
|
|
|
|
bool CMakeBuildConfiguration::isEnabled() const
|
|
{
|
|
return m_error.isEmpty() && !isParsing();
|
|
}
|
|
|
|
QString CMakeBuildConfiguration::disabledReason() const
|
|
{
|
|
return error();
|
|
}
|
|
|
|
QVariantMap CMakeBuildConfiguration::toMap() const
|
|
{
|
|
QVariantMap map(ProjectExplorer::BuildConfiguration::toMap());
|
|
const QStringList config
|
|
= Utils::transform(m_configurationForCMake, [](const CMakeConfigItem &i) { return i.toString(); });
|
|
map.insert(QLatin1String(CONFIGURATION_KEY), config);
|
|
return map;
|
|
}
|
|
|
|
bool CMakeBuildConfiguration::fromMap(const QVariantMap &map)
|
|
{
|
|
if (!BuildConfiguration::fromMap(map))
|
|
return false;
|
|
|
|
const CMakeConfig conf
|
|
= Utils::filtered(Utils::transform(map.value(QLatin1String(CONFIGURATION_KEY)).toStringList(),
|
|
[](const QString &v) { return CMakeConfigItem::fromString(v); }),
|
|
[](const CMakeConfigItem &c) { return !c.isNull(); });
|
|
|
|
setConfigurationForCMake(conf);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool CMakeBuildConfiguration::isParsing() const
|
|
{
|
|
return project()->isParsing() && isActive();
|
|
}
|
|
|
|
const QList<BuildTargetInfo> CMakeBuildConfiguration::appTargets() const
|
|
{
|
|
QList<BuildTargetInfo> appTargetList;
|
|
bool forAndroid = DeviceTypeKitAspect::deviceTypeId(target()->kit()) == Android::Constants::ANDROID_DEVICE_TYPE;
|
|
for (const CMakeBuildTarget &ct : m_buildTargets) {
|
|
if (ct.targetType == UtilityType)
|
|
continue;
|
|
|
|
if (ct.targetType == ExecutableType || (forAndroid && ct.targetType == DynamicLibraryType)) {
|
|
BuildTargetInfo bti;
|
|
bti.displayName = ct.title;
|
|
bti.targetFilePath = ct.executable;
|
|
bti.projectFilePath = ct.sourceDirectory.stringAppended("/");
|
|
bti.workingDirectory = ct.workingDirectory;
|
|
bti.buildKey = ct.title;
|
|
|
|
// Workaround for QTCREATORBUG-19354:
|
|
bti.runEnvModifier = [this](Environment &env, bool) {
|
|
if (HostOsInfo::isWindowsHost()) {
|
|
const Kit *k = target()->kit();
|
|
if (const QtSupport::BaseQtVersion *qt = QtSupport::QtKitAspect::qtVersion(k)) {
|
|
const QString installBinPath = qt->qmakeProperty("QT_INSTALL_BINS");
|
|
env.prependOrSetPath(installBinPath);
|
|
}
|
|
}
|
|
};
|
|
|
|
appTargetList.append(bti);
|
|
}
|
|
}
|
|
|
|
return appTargetList;
|
|
}
|
|
|
|
DeploymentData CMakeBuildConfiguration::deploymentData() const
|
|
{
|
|
DeploymentData result;
|
|
|
|
QDir sourceDir = target()->project()->projectDirectory().toString();
|
|
QDir buildDir = buildDirectory().toString();
|
|
|
|
QString deploymentPrefix;
|
|
QString deploymentFilePath = sourceDir.filePath("QtCreatorDeployment.txt");
|
|
|
|
bool hasDeploymentFile = QFileInfo::exists(deploymentFilePath);
|
|
if (!hasDeploymentFile) {
|
|
deploymentFilePath = buildDir.filePath("QtCreatorDeployment.txt");
|
|
hasDeploymentFile = QFileInfo::exists(deploymentFilePath);
|
|
}
|
|
if (!hasDeploymentFile)
|
|
return result;
|
|
|
|
deploymentPrefix = result.addFilesFromDeploymentFile(deploymentFilePath,
|
|
sourceDir.absolutePath());
|
|
for (const CMakeBuildTarget &ct : m_buildTargets) {
|
|
if (ct.targetType == ExecutableType || ct.targetType == DynamicLibraryType) {
|
|
if (!ct.executable.isEmpty()
|
|
&& !result.deployableForLocalFile(ct.executable.toString()).isValid()) {
|
|
result.addFile(ct.executable.toString(),
|
|
deploymentPrefix + buildDir.relativeFilePath(ct.executable.toFileInfo().dir().path()),
|
|
DeployableFile::TypeExecutable);
|
|
}
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
QStringList CMakeBuildConfiguration::buildTargetTitles() const
|
|
{
|
|
return transform(m_buildTargets, &CMakeBuildTarget::title);
|
|
}
|
|
|
|
const QList<CMakeBuildTarget> &CMakeBuildConfiguration::buildTargets() const
|
|
{
|
|
return m_buildTargets;
|
|
}
|
|
|
|
FilePath CMakeBuildConfiguration::shadowBuildDirectory(const FilePath &projectFilePath,
|
|
const Kit *k,
|
|
const QString &bcName,
|
|
BuildConfiguration::BuildType buildType)
|
|
{
|
|
if (projectFilePath.isEmpty())
|
|
return FilePath();
|
|
|
|
const QString projectName = projectFilePath.parentDir().fileName();
|
|
ProjectMacroExpander expander(projectFilePath, projectName, k, bcName, buildType);
|
|
QDir projectDir = QDir(Project::projectDirectory(projectFilePath).toString());
|
|
QString buildPath = expander.expand(ProjectExplorerPlugin::buildDirectoryTemplate());
|
|
buildPath.replace(" ", "-");
|
|
return FilePath::fromUserInput(projectDir.absoluteFilePath(buildPath));
|
|
}
|
|
|
|
void CMakeBuildConfiguration::buildTarget(const QString &buildTarget)
|
|
{
|
|
const Core::Id buildStep = ProjectExplorer::Constants::BUILDSTEPS_BUILD;
|
|
auto cmBs = qobject_cast<CMakeBuildStep *>(Utils::findOrDefault(
|
|
stepList(buildStep)->steps(),
|
|
[](const ProjectExplorer::BuildStep *bs) {
|
|
return bs->id() == Constants::CMAKE_BUILD_STEP_ID;
|
|
}));
|
|
|
|
QString originalBuildTarget;
|
|
if (cmBs) {
|
|
originalBuildTarget = cmBs->buildTarget();
|
|
cmBs->setBuildTarget(buildTarget);
|
|
}
|
|
|
|
BuildManager::buildList(stepList(buildStep));
|
|
|
|
if (cmBs)
|
|
cmBs->setBuildTarget(originalBuildTarget);
|
|
}
|
|
|
|
CMakeConfig CMakeBuildConfiguration::configurationFromCMake() const
|
|
{
|
|
return m_configurationFromCMake;
|
|
}
|
|
|
|
void CMakeBuildConfiguration::setConfigurationFromCMake(const CMakeConfig &config)
|
|
{
|
|
m_configurationFromCMake = config;
|
|
}
|
|
|
|
void CMakeBuildConfiguration::setBuildTargets(const QList<CMakeBuildTarget> &targets)
|
|
{
|
|
m_buildTargets = targets;
|
|
}
|
|
|
|
void CMakeBuildConfiguration::setConfigurationForCMake(const QList<ConfigModel::DataItem> &items)
|
|
{
|
|
const CMakeConfig newConfig = Utils::transform(items, [](const ConfigModel::DataItem &i) {
|
|
CMakeConfigItem ni;
|
|
ni.key = i.key.toUtf8();
|
|
ni.value = i.value.toUtf8();
|
|
ni.documentation = i.description.toUtf8();
|
|
ni.isAdvanced = i.isAdvanced;
|
|
ni.isUnset = i.isUnset;
|
|
ni.inCMakeCache = i.inCMakeCache;
|
|
ni.values = i.values;
|
|
switch (i.type) {
|
|
case CMakeProjectManager::ConfigModel::DataItem::BOOLEAN:
|
|
ni.type = CMakeConfigItem::BOOL;
|
|
break;
|
|
case CMakeProjectManager::ConfigModel::DataItem::FILE:
|
|
ni.type = CMakeConfigItem::FILEPATH;
|
|
break;
|
|
case CMakeProjectManager::ConfigModel::DataItem::DIRECTORY:
|
|
ni.type = CMakeConfigItem::PATH;
|
|
break;
|
|
case CMakeProjectManager::ConfigModel::DataItem::STRING:
|
|
ni.type = CMakeConfigItem::STRING;
|
|
break;
|
|
case CMakeProjectManager::ConfigModel::DataItem::UNKNOWN:
|
|
default:
|
|
ni.type = CMakeConfigItem::INTERNAL;
|
|
break;
|
|
}
|
|
return ni;
|
|
});
|
|
|
|
const CMakeConfig config = configurationForCMake() + newConfig;
|
|
setConfigurationForCMake(config);
|
|
}
|
|
|
|
void CMakeBuildConfiguration::clearError(ForceEnabledChanged fec)
|
|
{
|
|
if (!m_error.isEmpty()) {
|
|
m_error.clear();
|
|
fec = ForceEnabledChanged::True;
|
|
}
|
|
if (fec == ForceEnabledChanged::True)
|
|
emit enabledChanged();
|
|
}
|
|
|
|
void CMakeBuildConfiguration::emitBuildTypeChanged()
|
|
{
|
|
emit buildTypeChanged();
|
|
}
|
|
|
|
static CMakeConfig removeDuplicates(const CMakeConfig &config)
|
|
{
|
|
CMakeConfig result;
|
|
// Remove duplicates (last value wins):
|
|
QSet<QByteArray> knownKeys;
|
|
for (int i = config.count() - 1; i >= 0; --i) {
|
|
const CMakeConfigItem &item = config.at(i);
|
|
if (knownKeys.contains(item.key))
|
|
continue;
|
|
result.append(item);
|
|
knownKeys.insert(item.key);
|
|
}
|
|
Utils::sort(result, CMakeConfigItem::sortOperator());
|
|
return result;
|
|
}
|
|
|
|
void CMakeBuildConfiguration::setConfigurationForCMake(const CMakeConfig &config)
|
|
{
|
|
auto configs = removeDuplicates(config);
|
|
if (m_configurationForCMake.isEmpty())
|
|
m_configurationForCMake = removeDuplicates(configs + m_initialConfiguration);
|
|
else
|
|
m_configurationForCMake = configs;
|
|
|
|
const Kit *k = target()->kit();
|
|
CMakeConfig kitConfig = CMakeConfigurationKitAspect::configuration(k);
|
|
bool hasKitOverride = false;
|
|
foreach (const CMakeConfigItem &i, m_configurationForCMake) {
|
|
const QString b = CMakeConfigItem::expandedValueOf(k, i.key, kitConfig);
|
|
if (!b.isNull() && (i.expandedValue(k) != b || i.isUnset)) {
|
|
hasKitOverride = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (hasKitOverride)
|
|
setWarning(tr("CMake configuration set by the kit was overridden in the project."));
|
|
else
|
|
setWarning(QString());
|
|
|
|
emit configurationForCMakeChanged();
|
|
}
|
|
|
|
CMakeConfig CMakeBuildConfiguration::configurationForCMake() const
|
|
{
|
|
return removeDuplicates(CMakeConfigurationKitAspect::configuration(target()->kit()) + m_configurationForCMake);
|
|
}
|
|
|
|
void CMakeBuildConfiguration::setError(const QString &message)
|
|
{
|
|
const QString oldMessage = m_error;
|
|
if (m_error != message)
|
|
m_error = message;
|
|
if (oldMessage.isEmpty() && !message.isEmpty())
|
|
emit enabledChanged();
|
|
emit errorOccured(m_error);
|
|
}
|
|
|
|
void CMakeBuildConfiguration::setWarning(const QString &message)
|
|
{
|
|
if (m_warning == message)
|
|
return;
|
|
m_warning = message;
|
|
emit warningOccured(m_warning);
|
|
}
|
|
|
|
void CMakeBuildConfiguration::handleParsingSucceeded()
|
|
{
|
|
if (!isActive()) {
|
|
m_buildDirManager.stopParsingAndClearState();
|
|
return;
|
|
}
|
|
|
|
clearError();
|
|
|
|
QString errorMessage;
|
|
{
|
|
const QList<CMakeBuildTarget> buildTargets = m_buildDirManager.takeBuildTargets(
|
|
errorMessage);
|
|
checkAndReportError(errorMessage);
|
|
setBuildTargets(buildTargets);
|
|
}
|
|
|
|
{
|
|
const CMakeConfig cmakeConfig = m_buildDirManager.takeCMakeConfiguration(errorMessage);
|
|
checkAndReportError(errorMessage);
|
|
setConfigurationFromCMake(cmakeConfig);
|
|
}
|
|
|
|
{
|
|
target()->setApplicationTargets(appTargets());
|
|
target()->setDeploymentData(deploymentData());
|
|
}
|
|
|
|
project()->handleParsingSuccess(this);
|
|
|
|
{
|
|
m_buildDirManager.resetData();
|
|
}
|
|
|
|
{
|
|
emitBuildTypeChanged();
|
|
}
|
|
}
|
|
|
|
std::unique_ptr<CMakeProjectNode> CMakeBuildConfiguration::generateProjectTree(
|
|
const QList<const FileNode *> &allFiles)
|
|
{
|
|
QString errorMessage;
|
|
auto root = m_buildDirManager.generateProjectTree(allFiles, errorMessage);
|
|
checkAndReportError(errorMessage);
|
|
return root;
|
|
}
|
|
|
|
void CMakeBuildConfiguration::checkAndReportError(QString &errorMessage)
|
|
{
|
|
if (!errorMessage.isEmpty()) {
|
|
setError(errorMessage);
|
|
errorMessage.clear();
|
|
}
|
|
}
|
|
|
|
QString CMakeBuildConfiguration::error() const
|
|
{
|
|
return m_error;
|
|
}
|
|
|
|
QString CMakeBuildConfiguration::warning() const
|
|
{
|
|
return m_warning;
|
|
}
|
|
|
|
ProjectExplorer::NamedWidget *CMakeBuildConfiguration::createConfigWidget()
|
|
{
|
|
return new CMakeBuildSettingsWidget(this);
|
|
}
|
|
|
|
/*!
|
|
\class CMakeBuildConfigurationFactory
|
|
*/
|
|
|
|
CMakeBuildConfigurationFactory::CMakeBuildConfigurationFactory()
|
|
{
|
|
registerBuildConfiguration<CMakeBuildConfiguration>(
|
|
"CMakeProjectManager.CMakeBuildConfiguration");
|
|
|
|
setSupportedProjectType(CMakeProjectManager::Constants::CMAKEPROJECT_ID);
|
|
setSupportedProjectMimeTypeName(Constants::CMAKEPROJECTMIMETYPE);
|
|
}
|
|
|
|
CMakeBuildConfigurationFactory::BuildType CMakeBuildConfigurationFactory::buildTypeFromByteArray(
|
|
const QByteArray &in)
|
|
{
|
|
const QByteArray bt = in.toLower();
|
|
if (bt == "debug")
|
|
return BuildTypeDebug;
|
|
if (bt == "release")
|
|
return BuildTypeRelease;
|
|
if (bt == "relwithdebinfo")
|
|
return BuildTypeRelWithDebInfo;
|
|
if (bt == "minsizerel")
|
|
return BuildTypeMinSizeRel;
|
|
return BuildTypeNone;
|
|
}
|
|
|
|
BuildConfiguration::BuildType CMakeBuildConfigurationFactory::cmakeBuildTypeToBuildType(
|
|
const CMakeBuildConfigurationFactory::BuildType &in)
|
|
{
|
|
// Cover all common CMake build types
|
|
if (in == BuildTypeRelease || in == BuildTypeMinSizeRel)
|
|
return BuildConfiguration::Release;
|
|
else if (in == BuildTypeDebug)
|
|
return BuildConfiguration::Debug;
|
|
else if (in == BuildTypeRelWithDebInfo)
|
|
return BuildConfiguration::Profile;
|
|
else
|
|
return BuildConfiguration::Unknown;
|
|
}
|
|
|
|
QList<BuildInfo> CMakeBuildConfigurationFactory::availableBuilds(const Kit *k,
|
|
const FilePath &projectPath,
|
|
bool forSetup) const
|
|
{
|
|
QList<BuildInfo> result;
|
|
|
|
FilePath path = forSetup ? Project::projectDirectory(projectPath) : projectPath;
|
|
|
|
for (int type = BuildTypeNone; type != BuildTypeLast; ++type) {
|
|
BuildInfo info = createBuildInfo(k, path.toString(), BuildType(type));
|
|
if (forSetup) {
|
|
info.displayName = info.typeName;
|
|
info.buildDirectory = CMakeBuildConfiguration::shadowBuildDirectory(projectPath,
|
|
k,
|
|
info.displayName,
|
|
info.buildType);
|
|
}
|
|
result << info;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
BuildInfo CMakeBuildConfigurationFactory::createBuildInfo(const Kit *k,
|
|
const QString &sourceDir,
|
|
BuildType buildType) const
|
|
{
|
|
BuildInfo info(this);
|
|
info.kitId = k->id();
|
|
|
|
CMakeExtraBuildInfo extra;
|
|
extra.sourceDirectory = sourceDir;
|
|
|
|
CMakeConfigItem buildTypeItem;
|
|
switch (buildType) {
|
|
case BuildTypeNone:
|
|
info.typeName = tr("Build");
|
|
break;
|
|
case BuildTypeDebug:
|
|
buildTypeItem = {CMakeConfigItem("CMAKE_BUILD_TYPE", "Debug")};
|
|
info.typeName = tr("Debug");
|
|
info.buildType = BuildConfiguration::Debug;
|
|
break;
|
|
case BuildTypeRelease:
|
|
buildTypeItem = {CMakeConfigItem("CMAKE_BUILD_TYPE", "Release")};
|
|
info.typeName = tr("Release");
|
|
info.buildType = BuildConfiguration::Release;
|
|
break;
|
|
case BuildTypeMinSizeRel:
|
|
buildTypeItem = {CMakeConfigItem("CMAKE_BUILD_TYPE", "MinSizeRel")};
|
|
info.typeName = tr("Minimum Size Release");
|
|
info.buildType = BuildConfiguration::Release;
|
|
break;
|
|
case BuildTypeRelWithDebInfo:
|
|
buildTypeItem = {CMakeConfigItem("CMAKE_BUILD_TYPE", "RelWithDebInfo")};
|
|
info.typeName = tr("Release with Debug Information");
|
|
info.buildType = BuildConfiguration::Profile;
|
|
break;
|
|
default:
|
|
QTC_CHECK(false);
|
|
break;
|
|
}
|
|
|
|
if (!buildTypeItem.isNull())
|
|
extra.configuration.append(buildTypeItem);
|
|
|
|
const QString sysRoot = SysRootKitAspect::sysRoot(k).toString();
|
|
if (!sysRoot.isEmpty()) {
|
|
extra.configuration.append(CMakeConfigItem("CMAKE_SYSROOT", sysRoot.toUtf8()));
|
|
ProjectExplorer::ToolChain *tc = ProjectExplorer::ToolChainKitAspect::toolChain(
|
|
k, ProjectExplorer::Constants::CXX_LANGUAGE_ID);
|
|
if (tc) {
|
|
const QByteArray targetTriple = tc->originalTargetTriple().toUtf8();
|
|
extra.configuration.append(CMakeConfigItem("CMAKE_C_COMPILER_TARGET", targetTriple));
|
|
extra.configuration.append(CMakeConfigItem("CMAKE_CXX_COMPILER_TARGET ", targetTriple));
|
|
}
|
|
}
|
|
info.extraInfo = QVariant::fromValue(extra);
|
|
|
|
return info;
|
|
}
|
|
|
|
ProjectExplorer::BuildConfiguration::BuildType CMakeBuildConfiguration::buildType() const
|
|
{
|
|
QByteArray cmakeBuildTypeName;
|
|
QFile cmakeCache(buildDirectory().toString() + QLatin1String("/CMakeCache.txt"));
|
|
if (cmakeCache.open(QIODevice::ReadOnly)) {
|
|
while (!cmakeCache.atEnd()) {
|
|
QByteArray line = cmakeCache.readLine();
|
|
if (line.startsWith("CMAKE_BUILD_TYPE")) {
|
|
if (int pos = line.indexOf('='))
|
|
cmakeBuildTypeName = line.mid(pos + 1).trimmed();
|
|
break;
|
|
}
|
|
}
|
|
cmakeCache.close();
|
|
}
|
|
|
|
// Cover all common CMake build types
|
|
const CMakeBuildConfigurationFactory::BuildType cmakeBuildType
|
|
= CMakeBuildConfigurationFactory::buildTypeFromByteArray(cmakeBuildTypeName);
|
|
return CMakeBuildConfigurationFactory::cmakeBuildTypeToBuildType(cmakeBuildType);
|
|
}
|
|
|
|
CMakeProject *CMakeBuildConfiguration::project() const
|
|
{
|
|
return qobject_cast<CMakeProject *>(BuildConfiguration::project());
|
|
}
|
|
|
|
} // namespace Internal
|
|
} // namespace CMakeProjectManager
|