2012-10-02 09:12:39 +02:00
|
|
|
/****************************************************************************
|
2012-04-24 15:49:09 +02:00
|
|
|
**
|
2016-01-15 14:57:40 +01:00
|
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
|
|
|
** Contact: https://www.qt.io/licensing/
|
2012-04-24 15:49:09 +02:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** This file is part of Qt Creator.
|
2012-04-24 15:49:09 +02:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** 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
|
2016-01-15 14:57:40 +01:00
|
|
|
** 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.
|
2012-04-24 15:49:09 +02:00
|
|
|
**
|
2016-01-15 14:57:40 +01:00
|
|
|
** 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.
|
2012-04-24 15:49:09 +02:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
****************************************************************************/
|
2012-04-24 15:49:09 +02:00
|
|
|
|
2012-09-03 18:31:44 +02:00
|
|
|
#include "kitinformation.h"
|
2012-04-24 15:49:09 +02:00
|
|
|
|
2017-12-08 17:20:48 +01:00
|
|
|
#include "abi.h"
|
2012-04-24 15:49:09 +02:00
|
|
|
#include "devicesupport/desktopdevice.h"
|
|
|
|
|
#include "devicesupport/devicemanager.h"
|
2019-02-06 15:04:17 +01:00
|
|
|
#include "devicesupport/devicemanagermodel.h"
|
|
|
|
|
#include "devicesupport/idevicefactory.h"
|
2012-04-24 15:49:09 +02:00
|
|
|
#include "projectexplorerconstants.h"
|
2012-09-03 18:31:44 +02:00
|
|
|
#include "kit.h"
|
2012-04-24 15:49:09 +02:00
|
|
|
#include "toolchain.h"
|
|
|
|
|
#include "toolchainmanager.h"
|
|
|
|
|
|
2019-02-06 15:04:17 +01:00
|
|
|
#include <coreplugin/icore.h>
|
|
|
|
|
#include <coreplugin/variablechooser.h>
|
2014-10-22 10:00:07 +02:00
|
|
|
#include <ssh/sshconnection.h>
|
2014-07-07 19:02:26 +02:00
|
|
|
#include <utils/algorithm.h>
|
2019-11-21 15:35:24 +01:00
|
|
|
#include <utils/elidinglabel.h>
|
2019-02-06 15:04:17 +01:00
|
|
|
#include <utils/environment.h>
|
|
|
|
|
#include <utils/environmentdialog.h>
|
2014-10-22 10:00:07 +02:00
|
|
|
#include <utils/macroexpander.h>
|
2019-02-06 15:04:17 +01:00
|
|
|
#include <utils/pathchooser.h>
|
2013-02-28 14:06:17 +01:00
|
|
|
#include <utils/qtcassert.h>
|
2012-04-24 15:49:09 +02:00
|
|
|
|
2019-02-06 15:04:17 +01:00
|
|
|
#include <QCheckBox>
|
|
|
|
|
#include <QComboBox>
|
2016-04-15 11:03:33 +02:00
|
|
|
#include <QDir>
|
2013-03-18 14:47:33 +01:00
|
|
|
#include <QFileInfo>
|
2019-02-06 15:04:17 +01:00
|
|
|
#include <QFontMetrics>
|
|
|
|
|
#include <QGridLayout>
|
|
|
|
|
#include <QLabel>
|
|
|
|
|
#include <QPushButton>
|
|
|
|
|
#include <QVBoxLayout>
|
2012-04-24 15:49:09 +02:00
|
|
|
|
|
|
|
|
namespace ProjectExplorer {
|
|
|
|
|
|
2017-03-17 15:33:07 +02:00
|
|
|
const char KITINFORMATION_ID_V1[] = "PE.Profile.ToolChain";
|
|
|
|
|
const char KITINFORMATION_ID_V2[] = "PE.Profile.ToolChains";
|
|
|
|
|
const char KITINFORMATION_ID_V3[] = "PE.Profile.ToolChainsV3";
|
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
// --------------------------------------------------------------------------
|
2019-02-06 12:50:51 +01:00
|
|
|
// SysRootKitAspect:
|
2012-04-24 15:49:09 +02:00
|
|
|
// --------------------------------------------------------------------------
|
|
|
|
|
|
2019-02-06 15:04:17 +01:00
|
|
|
namespace Internal {
|
|
|
|
|
class SysRootKitAspectWidget : public KitAspectWidget
|
|
|
|
|
{
|
|
|
|
|
Q_DECLARE_TR_FUNCTIONS(ProjectExplorer::SysRootKitAspect)
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
SysRootKitAspectWidget(Kit *k, const KitAspect *ki) : KitAspectWidget(k, ki)
|
|
|
|
|
{
|
|
|
|
|
m_chooser = new Utils::PathChooser;
|
|
|
|
|
m_chooser->setExpectedKind(Utils::PathChooser::ExistingDirectory);
|
|
|
|
|
m_chooser->setHistoryCompleter(QLatin1String("PE.SysRoot.History"));
|
|
|
|
|
m_chooser->setFileName(SysRootKitAspect::sysRoot(k));
|
|
|
|
|
connect(m_chooser, &Utils::PathChooser::pathChanged,
|
|
|
|
|
this, &SysRootKitAspectWidget::pathWasChanged);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
~SysRootKitAspectWidget() override { delete m_chooser; }
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
void makeReadOnly() override { m_chooser->setReadOnly(true); }
|
|
|
|
|
QWidget *buttonWidget() const override { return m_chooser->buttonAtIndex(0); }
|
|
|
|
|
QWidget *mainWidget() const override { return m_chooser->lineEdit(); }
|
|
|
|
|
|
|
|
|
|
void refresh() override
|
|
|
|
|
{
|
|
|
|
|
if (!m_ignoreChange)
|
|
|
|
|
m_chooser->setFileName(SysRootKitAspect::sysRoot(m_kit));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void pathWasChanged()
|
|
|
|
|
{
|
|
|
|
|
m_ignoreChange = true;
|
|
|
|
|
SysRootKitAspect::setSysRoot(m_kit, m_chooser->fileName());
|
|
|
|
|
m_ignoreChange = false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Utils::PathChooser *m_chooser;
|
|
|
|
|
bool m_ignoreChange = false;
|
|
|
|
|
};
|
|
|
|
|
} // namespace Internal
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
SysRootKitAspect::SysRootKitAspect()
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
|
|
|
|
setObjectName(QLatin1String("SysRootInformation"));
|
2019-02-06 12:50:51 +01:00
|
|
|
setId(SysRootKitAspect::id());
|
2019-02-06 16:16:07 +01:00
|
|
|
setDisplayName(tr("Sysroot"));
|
|
|
|
|
setDescription(tr("The root directory of the system image to use.<br>"
|
|
|
|
|
"Leave empty when building for the desktop."));
|
2013-08-28 18:57:25 +02:00
|
|
|
setPriority(31000);
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
|
|
|
|
|
2019-05-27 16:09:44 +02:00
|
|
|
Tasks SysRootKitAspect::validate(const Kit *k) const
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2019-05-27 16:09:44 +02:00
|
|
|
Tasks result;
|
2019-05-28 13:49:26 +02:00
|
|
|
const Utils::FilePath dir = SysRootKitAspect::sysRoot(k);
|
2016-04-15 11:03:33 +02:00
|
|
|
if (dir.isEmpty())
|
|
|
|
|
return result;
|
|
|
|
|
|
2017-10-04 18:18:50 +02:00
|
|
|
if (dir.toString().startsWith("target:") || dir.toString().startsWith("remote:"))
|
|
|
|
|
return result;
|
|
|
|
|
|
2016-04-15 11:03:33 +02:00
|
|
|
const QFileInfo fi = dir.toFileInfo();
|
|
|
|
|
|
|
|
|
|
if (!fi.exists()) {
|
2020-01-15 08:56:11 +01:00
|
|
|
result << BuildSystemTask(Task::Warning,
|
|
|
|
|
tr("Sys Root \"%1\" does not exist in the file system.").arg(dir.toUserOutput()));
|
2016-04-15 11:03:33 +02:00
|
|
|
} else if (!fi.isDir()) {
|
2020-01-15 08:56:11 +01:00
|
|
|
result << BuildSystemTask(Task::Warning,
|
|
|
|
|
tr("Sys Root \"%1\" is not a directory.").arg(dir.toUserOutput()));
|
2016-04-15 11:03:33 +02:00
|
|
|
} else if (QDir(dir.toString()).entryList(QDir::AllEntries | QDir::NoDotAndDotDot).isEmpty()) {
|
2020-01-15 08:56:11 +01:00
|
|
|
result << BuildSystemTask(Task::Warning,
|
|
|
|
|
tr("Sys Root \"%1\" is empty.").arg(dir.toUserOutput()));
|
2012-07-27 15:43:00 +02:00
|
|
|
}
|
2012-04-24 15:49:09 +02:00
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
KitAspectWidget *SysRootKitAspect::createConfigWidget(Kit *k) const
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2018-02-12 12:49:22 +01:00
|
|
|
QTC_ASSERT(k, return nullptr);
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
return new Internal::SysRootKitAspectWidget(k, this);
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
KitAspect::ItemList SysRootKitAspect::toUserOutput(const Kit *k) const
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2019-05-28 08:28:51 +02:00
|
|
|
return {{tr("Sys Root"), sysRoot(k).toUserOutput()}};
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
void SysRootKitAspect::addToMacroExpander(Kit *kit, Utils::MacroExpander *expander) const
|
2016-03-22 12:21:01 +01:00
|
|
|
{
|
2018-02-12 12:49:22 +01:00
|
|
|
QTC_ASSERT(kit, return);
|
|
|
|
|
|
2017-09-07 17:05:47 +02:00
|
|
|
expander->registerFileVariables("SysRoot", tr("Sys Root"), [kit]() -> QString {
|
2019-02-06 12:50:51 +01:00
|
|
|
return SysRootKitAspect::sysRoot(kit).toString();
|
2016-03-22 12:21:01 +01:00
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
Core::Id SysRootKitAspect::id()
|
2013-09-09 17:10:41 +02:00
|
|
|
{
|
|
|
|
|
return "PE.Profile.SysRoot";
|
|
|
|
|
}
|
|
|
|
|
|
2019-05-28 13:49:26 +02:00
|
|
|
Utils::FilePath SysRootKitAspect::sysRoot(const Kit *k)
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2012-09-03 18:31:44 +02:00
|
|
|
if (!k)
|
2019-05-28 13:49:26 +02:00
|
|
|
return Utils::FilePath();
|
2019-01-07 12:15:40 +01:00
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
if (!k->value(SysRootKitAspect::id()).toString().isEmpty())
|
2019-05-28 13:49:26 +02:00
|
|
|
return Utils::FilePath::fromString(k->value(SysRootKitAspect::id()).toString());
|
2019-01-07 12:15:40 +01:00
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
for (ToolChain *tc : ToolChainKitAspect::toolChains(k)) {
|
2019-01-07 12:15:40 +01:00
|
|
|
if (!tc->sysRoot().isEmpty())
|
2019-05-28 13:49:26 +02:00
|
|
|
return Utils::FilePath::fromString(tc->sysRoot());
|
2019-01-07 12:15:40 +01:00
|
|
|
}
|
|
|
|
|
|
2019-05-28 13:49:26 +02:00
|
|
|
return Utils::FilePath();
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
|
|
|
|
|
2019-05-28 13:49:26 +02:00
|
|
|
void SysRootKitAspect::setSysRoot(Kit *k, const Utils::FilePath &v)
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2019-01-07 12:15:40 +01:00
|
|
|
if (!k)
|
|
|
|
|
return;
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
for (ToolChain *tc : ToolChainKitAspect::toolChains(k)) {
|
2019-01-07 12:15:40 +01:00
|
|
|
if (!tc->sysRoot().isEmpty()) {
|
|
|
|
|
// It's the sysroot from toolchain, don't set it.
|
|
|
|
|
if (tc->sysRoot() == v.toString())
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
// We've changed the default toolchain sysroot, set it.
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-02-06 12:50:51 +01:00
|
|
|
k->setValue(SysRootKitAspect::id(), v.toString());
|
2013-07-24 16:22:37 +02:00
|
|
|
}
|
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
// --------------------------------------------------------------------------
|
2019-02-06 12:50:51 +01:00
|
|
|
// ToolChainKitAspect:
|
2012-04-24 15:49:09 +02:00
|
|
|
// --------------------------------------------------------------------------
|
|
|
|
|
|
2019-02-06 15:04:17 +01:00
|
|
|
namespace Internal {
|
|
|
|
|
class ToolChainKitAspectWidget : public KitAspectWidget
|
|
|
|
|
{
|
|
|
|
|
Q_DECLARE_TR_FUNCTIONS(ProjectExplorer::ToolChainKitAspect)
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
ToolChainKitAspectWidget(Kit *k, const KitAspect *ki) : KitAspectWidget(k, ki)
|
|
|
|
|
{
|
|
|
|
|
m_mainWidget = new QWidget;
|
|
|
|
|
m_mainWidget->setContentsMargins(0, 0, 0, 0);
|
|
|
|
|
|
|
|
|
|
auto layout = new QGridLayout(m_mainWidget);
|
|
|
|
|
layout->setContentsMargins(0, 0, 0, 0);
|
|
|
|
|
layout->setColumnStretch(1, 2);
|
|
|
|
|
|
2019-07-03 18:34:30 +02:00
|
|
|
QList<Core::Id> languageList = Utils::toList(ToolChainManager::allLanguages());
|
2019-02-06 15:04:17 +01:00
|
|
|
Utils::sort(languageList, [](Core::Id l1, Core::Id l2) {
|
|
|
|
|
return ToolChainManager::displayNameOfLanguageId(l1)
|
|
|
|
|
< ToolChainManager::displayNameOfLanguageId(l2);
|
|
|
|
|
});
|
|
|
|
|
QTC_ASSERT(!languageList.isEmpty(), return);
|
|
|
|
|
int row = 0;
|
2019-07-03 18:34:30 +02:00
|
|
|
for (Core::Id l : qAsConst(languageList)) {
|
2019-02-06 15:04:17 +01:00
|
|
|
layout->addWidget(new QLabel(ToolChainManager::displayNameOfLanguageId(l) + ':'), row, 0);
|
|
|
|
|
auto cb = new QComboBox;
|
|
|
|
|
cb->setSizePolicy(QSizePolicy::Ignored, cb->sizePolicy().verticalPolicy());
|
2019-02-11 17:10:03 +01:00
|
|
|
cb->setToolTip(ki->description());
|
2019-02-06 15:04:17 +01:00
|
|
|
|
|
|
|
|
m_languageComboboxMap.insert(l, cb);
|
|
|
|
|
layout->addWidget(cb, row, 1);
|
|
|
|
|
++row;
|
|
|
|
|
|
2019-02-26 09:40:49 +01:00
|
|
|
connect(cb, QOverload<int>::of(&QComboBox::currentIndexChanged),
|
2019-02-06 15:04:17 +01:00
|
|
|
this, [this, l](int idx) { currentToolChainChanged(l, idx); });
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
refresh();
|
|
|
|
|
|
|
|
|
|
m_manageButton = new QPushButton(KitAspectWidget::msgManage());
|
|
|
|
|
m_manageButton->setContentsMargins(0, 0, 0, 0);
|
|
|
|
|
connect(m_manageButton, &QAbstractButton::clicked,
|
|
|
|
|
this, &ToolChainKitAspectWidget::manageToolChains);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
~ToolChainKitAspectWidget() override
|
|
|
|
|
{
|
|
|
|
|
delete m_mainWidget;
|
|
|
|
|
delete m_manageButton;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
QWidget *mainWidget() const override { return m_mainWidget; }
|
|
|
|
|
QWidget *buttonWidget() const override { return m_manageButton; }
|
|
|
|
|
|
|
|
|
|
void refresh() override
|
|
|
|
|
{
|
|
|
|
|
m_ignoreChanges = true;
|
|
|
|
|
foreach (Core::Id l, m_languageComboboxMap.keys()) {
|
|
|
|
|
const QList<ToolChain *> ltcList
|
|
|
|
|
= ToolChainManager::toolChains(Utils::equal(&ToolChain::language, l));
|
|
|
|
|
|
|
|
|
|
QComboBox *cb = m_languageComboboxMap.value(l);
|
|
|
|
|
cb->clear();
|
|
|
|
|
cb->addItem(tr("<No compiler>"), QByteArray());
|
|
|
|
|
|
|
|
|
|
foreach (ToolChain *tc, ltcList)
|
|
|
|
|
cb->addItem(tc->displayName(), tc->id());
|
|
|
|
|
|
|
|
|
|
cb->setEnabled(cb->count() > 1 && !m_isReadOnly);
|
|
|
|
|
const int index = indexOf(cb, ToolChainKitAspect::toolChain(m_kit, l));
|
|
|
|
|
cb->setCurrentIndex(index);
|
|
|
|
|
}
|
|
|
|
|
m_ignoreChanges = false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void makeReadOnly() override
|
|
|
|
|
{
|
|
|
|
|
m_isReadOnly = true;
|
|
|
|
|
foreach (Core::Id l, m_languageComboboxMap.keys()) {
|
|
|
|
|
m_languageComboboxMap.value(l)->setEnabled(false);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void manageToolChains()
|
|
|
|
|
{
|
|
|
|
|
Core::ICore::showOptionsDialog(Constants::TOOLCHAIN_SETTINGS_PAGE_ID, buttonWidget());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void currentToolChainChanged(Core::Id language, int idx)
|
|
|
|
|
{
|
|
|
|
|
if (m_ignoreChanges || idx < 0)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
const QByteArray id = m_languageComboboxMap.value(language)->itemData(idx).toByteArray();
|
|
|
|
|
ToolChain *tc = ToolChainManager::findToolChain(id);
|
|
|
|
|
QTC_ASSERT(!tc || tc->language() == language, return);
|
|
|
|
|
if (tc)
|
|
|
|
|
ToolChainKitAspect::setToolChain(m_kit, tc);
|
|
|
|
|
else
|
|
|
|
|
ToolChainKitAspect::clearToolChain(m_kit, language);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int indexOf(QComboBox *cb, const ToolChain *tc)
|
|
|
|
|
{
|
|
|
|
|
const QByteArray id = tc ? tc->id() : QByteArray();
|
|
|
|
|
for (int i = 0; i < cb->count(); ++i) {
|
|
|
|
|
if (id == cb->itemData(i).toByteArray())
|
|
|
|
|
return i;
|
|
|
|
|
}
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QWidget *m_mainWidget = nullptr;
|
|
|
|
|
QPushButton *m_manageButton = nullptr;
|
|
|
|
|
QHash<Core::Id, QComboBox *> m_languageComboboxMap;
|
|
|
|
|
bool m_ignoreChanges = false;
|
|
|
|
|
bool m_isReadOnly = false;
|
|
|
|
|
};
|
|
|
|
|
} // namespace Internal
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
ToolChainKitAspect::ToolChainKitAspect()
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
|
|
|
|
setObjectName(QLatin1String("ToolChainInformation"));
|
2019-02-06 12:50:51 +01:00
|
|
|
setId(ToolChainKitAspect::id());
|
2019-02-06 16:16:07 +01:00
|
|
|
setDisplayName(tr("Compiler"));
|
|
|
|
|
setDescription(tr("The compiler to use for building.<br>"
|
|
|
|
|
"Make sure the compiler will produce binaries compatible "
|
|
|
|
|
"with the target device, Qt version and other libraries used."));
|
2013-08-28 18:57:25 +02:00
|
|
|
setPriority(30000);
|
|
|
|
|
|
2016-01-29 16:38:37 +02:00
|
|
|
connect(KitManager::instance(), &KitManager::kitsLoaded,
|
2019-02-06 12:50:51 +01:00
|
|
|
this, &ToolChainKitAspect::kitsWereLoaded);
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
|
|
|
|
|
2016-07-20 16:24:32 +02:00
|
|
|
// language id -> tool chain id
|
2016-12-16 00:43:14 +01:00
|
|
|
static QMap<Core::Id, QByteArray> defaultToolChainIds()
|
2016-07-20 16:24:32 +02:00
|
|
|
{
|
2016-12-16 00:43:14 +01:00
|
|
|
QMap<Core::Id, QByteArray> toolChains;
|
2016-09-14 11:48:46 +02:00
|
|
|
const Abi abi = Abi::hostAbi();
|
2017-01-11 16:12:32 +01:00
|
|
|
QList<ToolChain *> tcList = ToolChainManager::toolChains(Utils::equal(&ToolChain::targetAbi, abi));
|
2016-12-16 00:43:14 +01:00
|
|
|
foreach (Core::Id l, ToolChainManager::allLanguages()) {
|
2016-07-20 16:24:32 +02:00
|
|
|
ToolChain *tc = Utils::findOrDefault(tcList, Utils::equal(&ToolChain::language, l));
|
|
|
|
|
toolChains.insert(l, tc ? tc->id() : QByteArray());
|
|
|
|
|
}
|
|
|
|
|
return toolChains;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static QVariant defaultToolChainValue()
|
|
|
|
|
{
|
2016-12-16 00:43:14 +01:00
|
|
|
const QMap<Core::Id, QByteArray> toolChains = defaultToolChainIds();
|
2016-07-20 16:24:32 +02:00
|
|
|
QVariantMap result;
|
|
|
|
|
auto end = toolChains.end();
|
|
|
|
|
for (auto it = toolChains.begin(); it != end; ++it) {
|
2016-12-16 00:43:14 +01:00
|
|
|
result.insert(it.key().toString(), it.value());
|
2016-07-20 16:24:32 +02:00
|
|
|
}
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2019-05-27 16:09:44 +02:00
|
|
|
Tasks ToolChainKitAspect::validate(const Kit *k) const
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2019-05-27 16:09:44 +02:00
|
|
|
Tasks result;
|
2013-03-21 12:27:18 +01:00
|
|
|
|
2016-07-12 16:27:45 +02:00
|
|
|
const QList<ToolChain*> tcList = toolChains(k);
|
|
|
|
|
if (tcList.isEmpty()) {
|
2020-01-15 08:56:11 +01:00
|
|
|
result << BuildSystemTask(Task::Warning, ToolChainKitAspect::msgNoToolChainInTarget());
|
2013-03-21 12:27:18 +01:00
|
|
|
} else {
|
2016-07-12 16:33:19 +02:00
|
|
|
QSet<Abi> targetAbis;
|
|
|
|
|
foreach (ToolChain *tc, tcList) {
|
|
|
|
|
targetAbis.insert(tc->targetAbi());
|
2016-07-12 16:27:45 +02:00
|
|
|
result << tc->validateKit(k);
|
2016-07-12 16:33:19 +02:00
|
|
|
}
|
|
|
|
|
if (targetAbis.count() != 1) {
|
2020-01-15 08:56:11 +01:00
|
|
|
result << BuildSystemTask(Task::Error,
|
|
|
|
|
tr("Compilers produce code for different ABIs: %1")
|
|
|
|
|
.arg(Utils::transform<QList>(targetAbis, &Abi::toString).join(", ")));
|
2016-07-12 16:33:19 +02:00
|
|
|
}
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
void ToolChainKitAspect::upgrade(Kit *k)
|
2016-07-20 16:24:32 +02:00
|
|
|
{
|
2018-02-12 12:49:22 +01:00
|
|
|
QTC_ASSERT(k, return);
|
|
|
|
|
|
2017-03-17 15:33:07 +02:00
|
|
|
const Core::Id oldIdV1 = KITINFORMATION_ID_V1;
|
|
|
|
|
const Core::Id oldIdV2 = KITINFORMATION_ID_V2;
|
2016-12-16 00:43:14 +01:00
|
|
|
|
2016-07-20 16:24:32 +02:00
|
|
|
// upgrade <=4.1 to 4.2 (keep old settings around for now)
|
2016-12-16 00:43:14 +01:00
|
|
|
{
|
|
|
|
|
const QVariant oldValue = k->value(oldIdV1);
|
|
|
|
|
const QVariant value = k->value(oldIdV2);
|
|
|
|
|
if (value.isNull() && !oldValue.isNull()) {
|
|
|
|
|
QVariantMap newValue;
|
|
|
|
|
if (oldValue.type() == QVariant::Map) {
|
|
|
|
|
// Used between 4.1 and 4.2:
|
|
|
|
|
newValue = oldValue.toMap();
|
|
|
|
|
} else {
|
|
|
|
|
// Used up to 4.1:
|
|
|
|
|
newValue.insert(Deprecated::Toolchain::languageId(Deprecated::Toolchain::Cxx), oldValue.toString());
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
const Core::Id typeId = DeviceTypeKitAspect::deviceTypeId(k);
|
2016-12-16 00:43:14 +01:00
|
|
|
if (typeId == Constants::DESKTOP_DEVICE_TYPE) {
|
|
|
|
|
// insert default C compiler which did not exist before
|
|
|
|
|
newValue.insert(Deprecated::Toolchain::languageId(Deprecated::Toolchain::C),
|
|
|
|
|
defaultToolChainIds().value(Core::Id(Constants::C_LANGUAGE_ID)));
|
|
|
|
|
}
|
2016-09-14 11:48:46 +02:00
|
|
|
}
|
2016-12-16 00:43:14 +01:00
|
|
|
k->setValue(oldIdV2, newValue);
|
|
|
|
|
k->setSticky(oldIdV2, k->isSticky(oldIdV1));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// upgrade 4.2 to 4.3 (keep old settings around for now)
|
|
|
|
|
{
|
|
|
|
|
const QVariant oldValue = k->value(oldIdV2);
|
2019-02-06 12:50:51 +01:00
|
|
|
const QVariant value = k->value(ToolChainKitAspect::id());
|
2016-12-16 00:43:14 +01:00
|
|
|
if (value.isNull() && !oldValue.isNull()) {
|
|
|
|
|
QVariantMap newValue = oldValue.toMap();
|
|
|
|
|
QVariantMap::iterator it = newValue.find(Deprecated::Toolchain::languageId(Deprecated::Toolchain::C));
|
|
|
|
|
if (it != newValue.end())
|
|
|
|
|
newValue.insert(Core::Id(Constants::C_LANGUAGE_ID).toString(), it.value());
|
|
|
|
|
it = newValue.find(Deprecated::Toolchain::languageId(Deprecated::Toolchain::Cxx));
|
|
|
|
|
if (it != newValue.end())
|
|
|
|
|
newValue.insert(Core::Id(Constants::CXX_LANGUAGE_ID).toString(), it.value());
|
2019-02-06 12:50:51 +01:00
|
|
|
k->setValue(ToolChainKitAspect::id(), newValue);
|
|
|
|
|
k->setSticky(ToolChainKitAspect::id(), k->isSticky(oldIdV2));
|
2016-07-20 16:24:32 +02:00
|
|
|
}
|
|
|
|
|
}
|
2017-01-23 16:30:06 +01:00
|
|
|
|
|
|
|
|
// upgrade 4.3-temporary-master-state to 4.3:
|
|
|
|
|
{
|
2019-02-06 12:50:51 +01:00
|
|
|
const QVariantMap valueMap = k->value(ToolChainKitAspect::id()).toMap();
|
2017-01-23 16:30:06 +01:00
|
|
|
QVariantMap result;
|
|
|
|
|
for (const QString &key : valueMap.keys()) {
|
|
|
|
|
const int pos = key.lastIndexOf('.');
|
|
|
|
|
if (pos >= 0)
|
|
|
|
|
result.insert(key.mid(pos + 1), valueMap.value(key));
|
|
|
|
|
else
|
|
|
|
|
result.insert(key, valueMap.value(key));
|
|
|
|
|
}
|
2019-02-06 12:50:51 +01:00
|
|
|
k->setValue(ToolChainKitAspect::id(), result);
|
2017-01-23 16:30:06 +01:00
|
|
|
}
|
2016-07-20 16:24:32 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
void ToolChainKitAspect::fix(Kit *k)
|
2012-10-11 13:47:23 +02:00
|
|
|
{
|
2013-08-29 13:14:19 +02:00
|
|
|
QTC_ASSERT(ToolChainManager::isLoaded(), return);
|
2016-12-16 00:43:14 +01:00
|
|
|
foreach (const Core::Id& l, ToolChainManager::allLanguages()) {
|
2018-04-19 15:59:05 +02:00
|
|
|
const QByteArray tcId = toolChainId(k, l);
|
|
|
|
|
if (!tcId.isEmpty() && !ToolChainManager::findToolChain(tcId)) {
|
|
|
|
|
qWarning("Tool chain set up in kit \"%s\" for \"%s\" not found.",
|
2016-12-09 13:53:09 +01:00
|
|
|
qPrintable(k->displayName()),
|
2016-12-16 00:43:14 +01:00
|
|
|
qPrintable(ToolChainManager::displayNameOfLanguageId(l)));
|
2016-12-12 17:13:29 +01:00
|
|
|
clearToolChain(k, l); // make sure to clear out no longer known tool chains
|
2016-07-12 16:27:45 +02:00
|
|
|
}
|
|
|
|
|
}
|
2012-10-11 13:47:23 +02:00
|
|
|
}
|
|
|
|
|
|
2016-12-16 00:43:14 +01:00
|
|
|
static Core::Id findLanguage(const QString &ls)
|
|
|
|
|
{
|
|
|
|
|
QString lsUpper = ls.toUpper();
|
|
|
|
|
return Utils::findOrDefault(ToolChainManager::allLanguages(),
|
|
|
|
|
[lsUpper](Core::Id l) { return lsUpper == l.toString().toUpper(); });
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
void ToolChainKitAspect::setup(Kit *k)
|
2013-01-22 12:47:07 +01:00
|
|
|
{
|
2013-08-29 13:14:19 +02:00
|
|
|
QTC_ASSERT(ToolChainManager::isLoaded(), return);
|
2018-02-12 12:49:22 +01:00
|
|
|
QTC_ASSERT(k, return);
|
|
|
|
|
|
2019-03-12 16:50:39 +01:00
|
|
|
QVariantMap value = k->value(id()).toMap();
|
|
|
|
|
if (value.empty())
|
|
|
|
|
value = defaultToolChainValue().toMap();
|
2016-07-15 13:15:27 +02:00
|
|
|
|
|
|
|
|
for (auto i = value.constBegin(); i != value.constEnd(); ++i) {
|
2016-12-16 00:43:14 +01:00
|
|
|
Core::Id l = findLanguage(i.key());
|
|
|
|
|
|
|
|
|
|
if (!l.isValid())
|
2016-07-15 13:15:27 +02:00
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
const QByteArray id = i.value().toByteArray();
|
|
|
|
|
ToolChain *tc = ToolChainManager::findToolChain(id);
|
|
|
|
|
if (tc)
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
// ID is not found: Might be an ABI string...
|
|
|
|
|
const QString abi = QString::fromUtf8(id);
|
2017-01-11 16:12:32 +01:00
|
|
|
tc = ToolChainManager::toolChain([abi, l](const ToolChain *t) {
|
2016-07-15 13:15:27 +02:00
|
|
|
return t->targetAbi().toString() == abi && t->language() == l;
|
|
|
|
|
});
|
2016-12-12 17:13:29 +01:00
|
|
|
if (tc)
|
|
|
|
|
setToolChain(k, tc);
|
|
|
|
|
else
|
|
|
|
|
clearToolChain(k, l);
|
2013-01-22 12:47:07 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
KitAspectWidget *ToolChainKitAspect::createConfigWidget(Kit *k) const
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2018-02-12 12:49:22 +01:00
|
|
|
QTC_ASSERT(k, return nullptr);
|
2019-02-06 12:50:51 +01:00
|
|
|
return new Internal::ToolChainKitAspectWidget(k, this);
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
QString ToolChainKitAspect::displayNamePostfix(const Kit *k) const
|
2012-08-07 13:09:56 +02:00
|
|
|
{
|
2020-02-18 18:25:26 +01:00
|
|
|
ToolChain *tc = cxxToolChain(k);
|
2012-08-07 13:09:56 +02:00
|
|
|
return tc ? tc->displayName() : QString();
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
KitAspect::ItemList ToolChainKitAspect::toUserOutput(const Kit *k) const
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2020-02-18 18:25:26 +01:00
|
|
|
ToolChain *tc = cxxToolChain(k);
|
2019-05-28 08:28:51 +02:00
|
|
|
return {{tr("Compiler"), tc ? tc->displayName() : tr("None")}};
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
void ToolChainKitAspect::addToEnvironment(const Kit *k, Utils::Environment &env) const
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2020-02-18 18:25:26 +01:00
|
|
|
ToolChain *tc = cxxToolChain(k);
|
2012-04-24 15:49:09 +02:00
|
|
|
if (tc)
|
|
|
|
|
tc->addToEnvironment(env);
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
void ToolChainKitAspect::addToMacroExpander(Kit *kit, Utils::MacroExpander *expander) const
|
2014-11-17 17:46:19 +01:00
|
|
|
{
|
2018-02-12 12:49:22 +01:00
|
|
|
QTC_ASSERT(kit, return);
|
|
|
|
|
|
2016-07-15 11:15:27 +02:00
|
|
|
// Compatibility with Qt Creator < 4.2:
|
2014-11-17 17:46:19 +01:00
|
|
|
expander->registerVariable("Compiler:Name", tr("Compiler"),
|
2020-02-18 18:25:26 +01:00
|
|
|
[kit] {
|
|
|
|
|
const ToolChain *tc = cxxToolChain(kit);
|
2014-11-17 17:46:19 +01:00
|
|
|
return tc ? tc->displayName() : tr("None");
|
|
|
|
|
});
|
2016-03-16 11:56:28 +01:00
|
|
|
|
|
|
|
|
expander->registerVariable("Compiler:Executable", tr("Path to the compiler executable"),
|
2020-02-18 18:25:26 +01:00
|
|
|
[kit] {
|
|
|
|
|
const ToolChain *tc = cxxToolChain(kit);
|
2016-03-16 11:56:28 +01:00
|
|
|
return tc ? tc->compilerCommand().toString() : QString();
|
|
|
|
|
});
|
2016-07-15 11:15:27 +02:00
|
|
|
|
|
|
|
|
expander->registerPrefix("Compiler:Name", tr("Compiler for different languages"),
|
2020-02-18 18:25:26 +01:00
|
|
|
[kit](const QString &ls) {
|
2016-12-16 00:43:14 +01:00
|
|
|
const ToolChain *tc = toolChain(kit, findLanguage(ls));
|
2016-07-15 11:15:27 +02:00
|
|
|
return tc ? tc->displayName() : tr("None");
|
|
|
|
|
});
|
|
|
|
|
expander->registerPrefix("Compiler:Executable", tr("Compiler executable for different languages"),
|
2020-02-18 18:25:26 +01:00
|
|
|
[kit](const QString &ls) {
|
2016-12-16 00:43:14 +01:00
|
|
|
const ToolChain *tc = toolChain(kit, findLanguage(ls));
|
2016-07-15 11:15:27 +02:00
|
|
|
return tc ? tc->compilerCommand().toString() : QString();
|
|
|
|
|
});
|
2014-11-17 17:46:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
IOutputParser *ToolChainKitAspect::createOutputParser(const Kit *k) const
|
2012-11-02 12:45:09 +01:00
|
|
|
{
|
2019-02-07 15:20:58 +01:00
|
|
|
for (const Core::Id langId : {Constants::CXX_LANGUAGE_ID, Constants::C_LANGUAGE_ID}) {
|
|
|
|
|
if (const ToolChain * const tc = toolChain(k, langId))
|
|
|
|
|
return tc->outputParser();
|
|
|
|
|
}
|
|
|
|
|
return nullptr;
|
2012-11-02 12:45:09 +01:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
QSet<Core::Id> ToolChainKitAspect::availableFeatures(const Kit *k) const
|
2017-02-07 00:26:51 +01:00
|
|
|
{
|
|
|
|
|
QSet<Core::Id> result;
|
|
|
|
|
for (ToolChain *tc : toolChains(k))
|
|
|
|
|
result.insert(tc->typeId().withPrefix("ToolChain."));
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
Core::Id ToolChainKitAspect::id()
|
2013-09-09 17:10:41 +02:00
|
|
|
{
|
2017-03-17 15:33:07 +02:00
|
|
|
return KITINFORMATION_ID_V3;
|
2013-09-09 17:10:41 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
QByteArray ToolChainKitAspect::toolChainId(const Kit *k, Core::Id language)
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2018-02-12 12:49:22 +01:00
|
|
|
QTC_ASSERT(ToolChainManager::isLoaded(), return nullptr);
|
2012-09-03 18:31:44 +02:00
|
|
|
if (!k)
|
2018-04-19 15:59:05 +02:00
|
|
|
return QByteArray();
|
2019-02-06 12:50:51 +01:00
|
|
|
QVariantMap value = k->value(ToolChainKitAspect::id()).toMap();
|
2018-04-19 15:59:05 +02:00
|
|
|
return value.value(language.toString(), QByteArray()).toByteArray();
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
ToolChain *ToolChainKitAspect::toolChain(const Kit *k, Core::Id language)
|
2018-04-19 15:59:05 +02:00
|
|
|
{
|
|
|
|
|
return ToolChainManager::findToolChain(toolChainId(k, language));
|
2016-07-12 16:27:45 +02:00
|
|
|
}
|
|
|
|
|
|
2020-02-18 18:25:26 +01:00
|
|
|
ToolChain *ToolChainKitAspect::cToolChain(const Kit *k)
|
|
|
|
|
{
|
|
|
|
|
return ToolChainManager::findToolChain(toolChainId(k, ProjectExplorer::Constants::C_LANGUAGE_ID));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ToolChain *ToolChainKitAspect::cxxToolChain(const Kit *k)
|
|
|
|
|
{
|
|
|
|
|
return ToolChainManager::findToolChain(toolChainId(k, ProjectExplorer::Constants::CXX_LANGUAGE_ID));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
QList<ToolChain *> ToolChainKitAspect::toolChains(const Kit *k)
|
2016-07-12 16:27:45 +02:00
|
|
|
{
|
2018-02-12 12:49:22 +01:00
|
|
|
QTC_ASSERT(k, return QList<ToolChain *>());
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
const QVariantMap value = k->value(ToolChainKitAspect::id()).toMap();
|
2016-07-12 16:27:45 +02:00
|
|
|
const QList<ToolChain *> tcList
|
2019-07-03 18:34:30 +02:00
|
|
|
= Utils::transform<QList>(ToolChainManager::allLanguages(),
|
2016-12-16 00:43:14 +01:00
|
|
|
[&value](Core::Id l) -> ToolChain * {
|
|
|
|
|
return ToolChainManager::findToolChain(value.value(l.toString()).toByteArray());
|
2016-07-12 16:27:45 +02:00
|
|
|
});
|
|
|
|
|
return Utils::filtered(tcList, [](ToolChain *tc) { return tc; });
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
void ToolChainKitAspect::setToolChain(Kit *k, ToolChain *tc)
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2016-07-12 16:27:45 +02:00
|
|
|
QTC_ASSERT(tc, return);
|
2018-02-12 12:49:22 +01:00
|
|
|
QTC_ASSERT(k, return);
|
2019-02-06 12:50:51 +01:00
|
|
|
QVariantMap result = k->value(ToolChainKitAspect::id()).toMap();
|
2016-12-16 00:43:14 +01:00
|
|
|
result.insert(tc->language().toString(), tc->id());
|
|
|
|
|
|
2016-12-12 17:13:29 +01:00
|
|
|
k->setValue(id(), result);
|
2016-07-12 16:27:45 +02:00
|
|
|
}
|
|
|
|
|
|
2017-08-04 13:56:57 +02:00
|
|
|
/**
|
2019-02-06 12:50:51 +01:00
|
|
|
* @brief ToolChainKitAspect::setAllToolChainsToMatch
|
2017-08-04 13:56:57 +02:00
|
|
|
*
|
|
|
|
|
* Set up all toolchains to be similar to the one toolchain provided. Similar ideally means
|
|
|
|
|
* that all toolchains use the "same" compiler from the same installation, but we will
|
|
|
|
|
* settle for a toolchain with a matching API instead.
|
|
|
|
|
*
|
|
|
|
|
* @param k The kit to set up
|
|
|
|
|
* @param tc The toolchain to match other languages for.
|
|
|
|
|
*/
|
2019-02-06 12:50:51 +01:00
|
|
|
void ToolChainKitAspect::setAllToolChainsToMatch(Kit *k, ToolChain *tc)
|
2017-08-04 13:56:57 +02:00
|
|
|
{
|
|
|
|
|
QTC_ASSERT(tc, return);
|
2018-02-12 12:49:22 +01:00
|
|
|
QTC_ASSERT(k, return);
|
2017-08-04 13:56:57 +02:00
|
|
|
|
|
|
|
|
const QList<ToolChain *> allTcList = ToolChainManager::toolChains();
|
|
|
|
|
QTC_ASSERT(allTcList.contains(tc), return);
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
QVariantMap result = k->value(ToolChainKitAspect::id()).toMap();
|
2017-08-04 13:56:57 +02:00
|
|
|
result.insert(tc->language().toString(), tc->id());
|
|
|
|
|
|
|
|
|
|
for (Core::Id l : ToolChainManager::allLanguages()) {
|
|
|
|
|
if (l == tc->language())
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
ToolChain *match = nullptr;
|
|
|
|
|
ToolChain *bestMatch = nullptr;
|
|
|
|
|
for (ToolChain *other : allTcList) {
|
|
|
|
|
if (!other->isValid() || other->language() != l)
|
|
|
|
|
continue;
|
|
|
|
|
if (other->targetAbi() == tc->targetAbi())
|
|
|
|
|
match = other;
|
|
|
|
|
if (match == other
|
|
|
|
|
&& other->compilerCommand().parentDir() == tc->compilerCommand().parentDir()) {
|
|
|
|
|
bestMatch = other;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (bestMatch)
|
|
|
|
|
result.insert(l.toString(), bestMatch->id());
|
|
|
|
|
else if (match)
|
|
|
|
|
result.insert(l.toString(), match->id());
|
|
|
|
|
else
|
|
|
|
|
result.insert(l.toString(), QByteArray());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
k->setValue(id(), result);
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
void ToolChainKitAspect::clearToolChain(Kit *k, Core::Id language)
|
2016-07-12 16:27:45 +02:00
|
|
|
{
|
2016-12-16 00:43:14 +01:00
|
|
|
QTC_ASSERT(language.isValid(), return);
|
2018-02-12 12:49:22 +01:00
|
|
|
QTC_ASSERT(k, return);
|
2016-07-12 16:27:45 +02:00
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
QVariantMap result = k->value(ToolChainKitAspect::id()).toMap();
|
2016-12-16 00:43:14 +01:00
|
|
|
result.insert(language.toString(), QByteArray());
|
2016-07-12 16:27:45 +02:00
|
|
|
k->setValue(id(), result);
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
Abi ToolChainKitAspect::targetAbi(const Kit *k)
|
2016-07-13 10:40:05 +02:00
|
|
|
{
|
|
|
|
|
QList<ToolChain *> tcList = toolChains(k);
|
|
|
|
|
// Find the best possible ABI for all the tool chains...
|
|
|
|
|
Abi cxxAbi;
|
|
|
|
|
QHash<Abi, int> abiCount;
|
|
|
|
|
foreach (ToolChain *tc, tcList) {
|
|
|
|
|
Abi ta = tc->targetAbi();
|
2016-12-16 00:43:14 +01:00
|
|
|
if (tc->language() == Core::Id(Constants::CXX_LANGUAGE_ID))
|
2016-07-13 10:40:05 +02:00
|
|
|
cxxAbi = tc->targetAbi();
|
|
|
|
|
abiCount[ta] = (abiCount.contains(ta) ? abiCount[ta] + 1 : 1);
|
|
|
|
|
}
|
|
|
|
|
QVector<Abi> candidates;
|
|
|
|
|
int count = -1;
|
|
|
|
|
candidates.reserve(tcList.count());
|
|
|
|
|
for (auto i = abiCount.begin(); i != abiCount.end(); ++i) {
|
|
|
|
|
if (i.value() > count) {
|
|
|
|
|
candidates.clear();
|
|
|
|
|
candidates.append(i.key());
|
|
|
|
|
count = i.value();
|
|
|
|
|
} else if (i.value() == count) {
|
|
|
|
|
candidates.append(i.key());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Found a good candidate:
|
|
|
|
|
if (candidates.isEmpty())
|
|
|
|
|
return Abi::hostAbi();
|
|
|
|
|
if (candidates.contains(cxxAbi)) // Use Cxx compiler as a tie breaker
|
|
|
|
|
return cxxAbi;
|
|
|
|
|
return candidates.at(0); // Use basically a random Abi...
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
QString ToolChainKitAspect::msgNoToolChainInTarget()
|
2012-07-27 15:43:00 +02:00
|
|
|
{
|
2012-09-03 18:31:44 +02:00
|
|
|
return tr("No compiler set in kit.");
|
2012-07-27 15:43:00 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
void ToolChainKitAspect::kitsWereLoaded()
|
2012-10-17 17:27:17 +02:00
|
|
|
{
|
2013-08-21 12:48:46 +02:00
|
|
|
foreach (Kit *k, KitManager::kits())
|
2013-02-28 15:07:01 +01:00
|
|
|
fix(k);
|
|
|
|
|
|
2016-01-29 16:38:37 +02:00
|
|
|
connect(ToolChainManager::instance(), &ToolChainManager::toolChainRemoved,
|
2019-02-06 12:50:51 +01:00
|
|
|
this, &ToolChainKitAspect::toolChainRemoved);
|
2016-01-29 16:38:37 +02:00
|
|
|
connect(ToolChainManager::instance(), &ToolChainManager::toolChainUpdated,
|
2019-02-06 12:50:51 +01:00
|
|
|
this, &ToolChainKitAspect::toolChainUpdated);
|
2013-02-28 15:07:01 +01:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
void ToolChainKitAspect::toolChainUpdated(ToolChain *tc)
|
2013-02-28 15:07:01 +01:00
|
|
|
{
|
2019-08-13 14:03:12 +02:00
|
|
|
for (Kit *k : KitManager::kits()) {
|
|
|
|
|
if (toolChain(k, tc->language()) == tc)
|
|
|
|
|
notifyAboutUpdate(k);
|
|
|
|
|
}
|
2013-02-28 15:07:01 +01:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
void ToolChainKitAspect::toolChainRemoved(ToolChain *tc)
|
2012-10-17 17:27:17 +02:00
|
|
|
{
|
2019-07-23 10:58:00 +02:00
|
|
|
Q_UNUSED(tc)
|
2013-08-21 12:48:46 +02:00
|
|
|
foreach (Kit *k, KitManager::kits())
|
2013-02-28 15:07:01 +01:00
|
|
|
fix(k);
|
2012-10-17 17:27:17 +02:00
|
|
|
}
|
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
// --------------------------------------------------------------------------
|
2019-02-06 12:50:51 +01:00
|
|
|
// DeviceTypeKitAspect:
|
2012-04-24 15:49:09 +02:00
|
|
|
// --------------------------------------------------------------------------
|
2019-02-06 15:04:17 +01:00
|
|
|
namespace Internal {
|
|
|
|
|
class DeviceTypeKitAspectWidget : public KitAspectWidget
|
|
|
|
|
{
|
|
|
|
|
Q_DECLARE_TR_FUNCTIONS(ProjectExplorer::DeviceTypeKitAspect)
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
DeviceTypeKitAspectWidget(Kit *workingCopy, const KitAspect *ki)
|
|
|
|
|
: KitAspectWidget(workingCopy, ki), m_comboBox(new QComboBox)
|
|
|
|
|
{
|
|
|
|
|
for (IDeviceFactory *factory : IDeviceFactory::allDeviceFactories())
|
|
|
|
|
m_comboBox->addItem(factory->displayName(), factory->deviceType().toSetting());
|
2019-02-11 17:10:03 +01:00
|
|
|
m_comboBox->setToolTip(ki->description());
|
2019-02-06 15:04:17 +01:00
|
|
|
refresh();
|
2019-02-26 09:40:49 +01:00
|
|
|
connect(m_comboBox, QOverload<int>::of(&QComboBox::currentIndexChanged),
|
2019-02-06 15:04:17 +01:00
|
|
|
this, &DeviceTypeKitAspectWidget::currentTypeChanged);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
~DeviceTypeKitAspectWidget() override { delete m_comboBox; }
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
QWidget *mainWidget() const override { return m_comboBox; }
|
|
|
|
|
void makeReadOnly() override { m_comboBox->setEnabled(false); }
|
|
|
|
|
|
|
|
|
|
void refresh() override
|
|
|
|
|
{
|
|
|
|
|
Core::Id devType = DeviceTypeKitAspect::deviceTypeId(m_kit);
|
|
|
|
|
if (!devType.isValid())
|
|
|
|
|
m_comboBox->setCurrentIndex(-1);
|
|
|
|
|
for (int i = 0; i < m_comboBox->count(); ++i) {
|
|
|
|
|
if (m_comboBox->itemData(i) == devType.toSetting()) {
|
|
|
|
|
m_comboBox->setCurrentIndex(i);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void currentTypeChanged(int idx)
|
|
|
|
|
{
|
|
|
|
|
Core::Id type = idx < 0 ? Core::Id() : Core::Id::fromSetting(m_comboBox->itemData(idx));
|
|
|
|
|
DeviceTypeKitAspect::setDeviceTypeId(m_kit, type);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QComboBox *m_comboBox;
|
|
|
|
|
};
|
|
|
|
|
} // namespace Internal
|
2012-04-24 15:49:09 +02:00
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
DeviceTypeKitAspect::DeviceTypeKitAspect()
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
|
|
|
|
setObjectName(QLatin1String("DeviceTypeInformation"));
|
2019-02-06 12:50:51 +01:00
|
|
|
setId(DeviceTypeKitAspect::id());
|
2019-02-06 16:16:07 +01:00
|
|
|
setDisplayName(tr("Device type"));
|
|
|
|
|
setDescription(tr("The type of device to run applications on."));
|
2013-08-28 18:57:25 +02:00
|
|
|
setPriority(33000);
|
2019-02-05 14:42:30 +01:00
|
|
|
makeEssential();
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
|
|
|
|
|
2019-03-12 16:50:39 +01:00
|
|
|
void DeviceTypeKitAspect::setup(Kit *k)
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2019-05-06 14:55:34 +02:00
|
|
|
if (k && !k->hasValue(id()))
|
2019-03-12 16:50:39 +01:00
|
|
|
k->setValue(id(), QByteArray(Constants::DESKTOP_DEVICE_TYPE));
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
|
|
|
|
|
2019-05-27 16:09:44 +02:00
|
|
|
Tasks DeviceTypeKitAspect::validate(const Kit *k) const
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2019-07-23 10:58:00 +02:00
|
|
|
Q_UNUSED(k)
|
2019-05-27 16:09:44 +02:00
|
|
|
return {};
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
KitAspectWidget *DeviceTypeKitAspect::createConfigWidget(Kit *k) const
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2018-02-12 12:49:22 +01:00
|
|
|
QTC_ASSERT(k, return nullptr);
|
2019-02-06 12:50:51 +01:00
|
|
|
return new Internal::DeviceTypeKitAspectWidget(k, this);
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
KitAspect::ItemList DeviceTypeKitAspect::toUserOutput(const Kit *k) const
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2018-02-12 12:49:22 +01:00
|
|
|
QTC_ASSERT(k, return {});
|
2012-09-03 18:31:44 +02:00
|
|
|
Core::Id type = deviceTypeId(k);
|
2012-04-24 15:49:09 +02:00
|
|
|
QString typeDisplayName = tr("Unknown device type");
|
|
|
|
|
if (type.isValid()) {
|
2018-10-04 19:08:16 +02:00
|
|
|
if (IDeviceFactory *factory = IDeviceFactory::find(type))
|
|
|
|
|
typeDisplayName = factory->displayName();
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
2019-05-28 08:28:51 +02:00
|
|
|
return {{tr("Device type"), typeDisplayName}};
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
const Core::Id DeviceTypeKitAspect::id()
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2013-09-09 17:10:41 +02:00
|
|
|
return "PE.Profile.DeviceType";
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
const Core::Id DeviceTypeKitAspect::deviceTypeId(const Kit *k)
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2019-02-06 12:50:51 +01:00
|
|
|
return k ? Core::Id::fromSetting(k->value(DeviceTypeKitAspect::id())) : Core::Id();
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
void DeviceTypeKitAspect::setDeviceTypeId(Kit *k, Core::Id type)
|
2013-07-24 16:22:37 +02:00
|
|
|
{
|
2018-02-12 12:49:22 +01:00
|
|
|
QTC_ASSERT(k, return);
|
2019-02-06 12:50:51 +01:00
|
|
|
k->setValue(DeviceTypeKitAspect::id(), type.toSetting());
|
2013-07-24 16:22:37 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
QSet<Core::Id> DeviceTypeKitAspect::supportedPlatforms(const Kit *k) const
|
2015-11-27 16:14:14 +01:00
|
|
|
{
|
2017-02-22 15:09:35 +01:00
|
|
|
return {deviceTypeId(k)};
|
2015-11-27 16:14:14 +01:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
QSet<Core::Id> DeviceTypeKitAspect::availableFeatures(const Kit *k) const
|
2014-07-29 15:15:27 +02:00
|
|
|
{
|
2019-02-06 12:50:51 +01:00
|
|
|
Core::Id id = DeviceTypeKitAspect::deviceTypeId(k);
|
2014-07-29 15:15:27 +02:00
|
|
|
if (id.isValid())
|
2017-02-22 15:09:35 +01:00
|
|
|
return {id.withPrefix("DeviceType.")};
|
2015-11-27 16:14:14 +01:00
|
|
|
return QSet<Core::Id>();
|
2014-07-29 15:15:27 +02:00
|
|
|
}
|
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
// --------------------------------------------------------------------------
|
2019-02-06 12:50:51 +01:00
|
|
|
// DeviceKitAspect:
|
2012-04-24 15:49:09 +02:00
|
|
|
// --------------------------------------------------------------------------
|
2019-02-06 15:04:17 +01:00
|
|
|
namespace Internal {
|
|
|
|
|
class DeviceKitAspectWidget : public KitAspectWidget
|
|
|
|
|
{
|
|
|
|
|
Q_DECLARE_TR_FUNCTIONS(ProjectExplorer::DeviceKitAspect)
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
DeviceKitAspectWidget(Kit *workingCopy, const KitAspect *ki)
|
|
|
|
|
: KitAspectWidget(workingCopy, ki), m_comboBox(new QComboBox),
|
|
|
|
|
m_model(new DeviceManagerModel(DeviceManager::instance()))
|
|
|
|
|
{
|
|
|
|
|
m_comboBox->setSizePolicy(QSizePolicy::Ignored, m_comboBox->sizePolicy().verticalPolicy());
|
|
|
|
|
m_comboBox->setModel(m_model);
|
|
|
|
|
m_manageButton = new QPushButton(KitAspectWidget::msgManage());
|
|
|
|
|
refresh();
|
2019-02-11 17:10:03 +01:00
|
|
|
m_comboBox->setToolTip(ki->description());
|
2019-02-06 15:04:17 +01:00
|
|
|
|
|
|
|
|
connect(m_model, &QAbstractItemModel::modelAboutToBeReset,
|
|
|
|
|
this, &DeviceKitAspectWidget::modelAboutToReset);
|
|
|
|
|
connect(m_model, &QAbstractItemModel::modelReset,
|
|
|
|
|
this, &DeviceKitAspectWidget::modelReset);
|
2019-02-26 09:40:49 +01:00
|
|
|
connect(m_comboBox, QOverload<int>::of(&QComboBox::currentIndexChanged),
|
2019-02-06 15:04:17 +01:00
|
|
|
this, &DeviceKitAspectWidget::currentDeviceChanged);
|
|
|
|
|
connect(m_manageButton, &QAbstractButton::clicked,
|
|
|
|
|
this, &DeviceKitAspectWidget::manageDevices);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
~DeviceKitAspectWidget() override
|
|
|
|
|
{
|
|
|
|
|
delete m_comboBox;
|
|
|
|
|
delete m_model;
|
|
|
|
|
delete m_manageButton;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
QWidget *mainWidget() const override { return m_comboBox; }
|
|
|
|
|
QWidget *buttonWidget() const override { return m_manageButton; }
|
|
|
|
|
void makeReadOnly() override { m_comboBox->setEnabled(false); }
|
|
|
|
|
|
|
|
|
|
void refresh() override
|
|
|
|
|
{
|
|
|
|
|
m_model->setTypeFilter(DeviceTypeKitAspect::deviceTypeId(m_kit));
|
|
|
|
|
m_comboBox->setCurrentIndex(m_model->indexOf(DeviceKitAspect::device(m_kit)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void manageDevices()
|
|
|
|
|
{
|
|
|
|
|
Core::ICore::showOptionsDialog(Constants::DEVICE_SETTINGS_PAGE_ID, buttonWidget());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void modelAboutToReset()
|
|
|
|
|
{
|
|
|
|
|
m_selectedId = m_model->deviceId(m_comboBox->currentIndex());
|
|
|
|
|
m_ignoreChange = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void modelReset()
|
|
|
|
|
{
|
|
|
|
|
m_comboBox->setCurrentIndex(m_model->indexForId(m_selectedId));
|
|
|
|
|
m_ignoreChange = false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void currentDeviceChanged()
|
|
|
|
|
{
|
|
|
|
|
if (m_ignoreChange)
|
|
|
|
|
return;
|
|
|
|
|
DeviceKitAspect::setDeviceId(m_kit, m_model->deviceId(m_comboBox->currentIndex()));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool m_isReadOnly = false;
|
|
|
|
|
bool m_ignoreChange = false;
|
|
|
|
|
QComboBox *m_comboBox;
|
|
|
|
|
QPushButton *m_manageButton;
|
|
|
|
|
DeviceManagerModel *m_model;
|
|
|
|
|
Core::Id m_selectedId;
|
|
|
|
|
};
|
|
|
|
|
} // namespace Internal
|
2012-04-24 15:49:09 +02:00
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
DeviceKitAspect::DeviceKitAspect()
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
|
|
|
|
setObjectName(QLatin1String("DeviceInformation"));
|
2019-02-06 12:50:51 +01:00
|
|
|
setId(DeviceKitAspect::id());
|
2019-02-06 16:16:07 +01:00
|
|
|
setDisplayName(tr("Device"));
|
|
|
|
|
setDescription(tr("The device to run the applications on."));
|
2013-08-28 18:57:25 +02:00
|
|
|
setPriority(32000);
|
|
|
|
|
|
2016-01-29 16:38:37 +02:00
|
|
|
connect(KitManager::instance(), &KitManager::kitsLoaded,
|
2019-02-06 12:50:51 +01:00
|
|
|
this, &DeviceKitAspect::kitsWereLoaded);
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
QVariant DeviceKitAspect::defaultValue(const Kit *k) const
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2019-02-06 12:50:51 +01:00
|
|
|
Core::Id type = DeviceTypeKitAspect::deviceTypeId(k);
|
2014-08-01 13:06:21 +02:00
|
|
|
// Use default device if that is compatible:
|
2012-10-11 18:17:15 +02:00
|
|
|
IDevice::ConstPtr dev = DeviceManager::instance()->defaultDevice(type);
|
2014-08-01 13:06:21 +02:00
|
|
|
if (dev && dev->isCompatibleWith(k))
|
|
|
|
|
return dev->id().toString();
|
|
|
|
|
// Use any other device that is compatible:
|
|
|
|
|
for (int i = 0; i < DeviceManager::instance()->deviceCount(); ++i) {
|
|
|
|
|
dev = DeviceManager::instance()->deviceAt(i);
|
|
|
|
|
if (dev && dev->isCompatibleWith(k))
|
|
|
|
|
return dev->id().toString();
|
|
|
|
|
}
|
|
|
|
|
// Fail: No device set up.
|
|
|
|
|
return QString();
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
|
|
|
|
|
2019-05-27 16:09:44 +02:00
|
|
|
Tasks DeviceKitAspect::validate(const Kit *k) const
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2019-02-06 12:50:51 +01:00
|
|
|
IDevice::ConstPtr dev = DeviceKitAspect::device(k);
|
2019-05-27 16:09:44 +02:00
|
|
|
Tasks result;
|
2012-10-11 18:17:15 +02:00
|
|
|
if (dev.isNull())
|
2020-01-15 08:56:11 +01:00
|
|
|
result.append(BuildSystemTask(Task::Warning, tr("No device set.")));
|
2014-08-01 13:06:21 +02:00
|
|
|
else if (!dev->isCompatibleWith(k))
|
2020-01-15 08:56:11 +01:00
|
|
|
result.append(BuildSystemTask(Task::Error, tr("Device is incompatible with this kit.")));
|
2014-08-01 13:06:21 +02:00
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
void DeviceKitAspect::fix(Kit *k)
|
2012-10-11 13:47:23 +02:00
|
|
|
{
|
2019-02-06 12:50:51 +01:00
|
|
|
IDevice::ConstPtr dev = DeviceKitAspect::device(k);
|
2014-08-01 13:06:21 +02:00
|
|
|
if (!dev.isNull() && !dev->isCompatibleWith(k)) {
|
|
|
|
|
qWarning("Device is no longer compatible with kit \"%s\", removing it.",
|
|
|
|
|
qPrintable(k->displayName()));
|
2013-11-29 12:17:53 +01:00
|
|
|
setDeviceId(k, Core::Id());
|
|
|
|
|
}
|
2013-02-28 14:06:17 +01:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
void DeviceKitAspect::setup(Kit *k)
|
2013-02-28 14:06:17 +01:00
|
|
|
{
|
|
|
|
|
QTC_ASSERT(DeviceManager::instance()->isLoaded(), return);
|
2019-02-06 12:50:51 +01:00
|
|
|
IDevice::ConstPtr dev = DeviceKitAspect::device(k);
|
2014-08-01 13:06:21 +02:00
|
|
|
if (!dev.isNull() && dev->isCompatibleWith(k))
|
2013-02-28 14:06:17 +01:00
|
|
|
return;
|
|
|
|
|
|
2013-01-18 12:10:22 +01:00
|
|
|
setDeviceId(k, Core::Id::fromSetting(defaultValue(k)));
|
2012-10-11 13:47:23 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
KitAspectWidget *DeviceKitAspect::createConfigWidget(Kit *k) const
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2018-02-12 12:49:22 +01:00
|
|
|
QTC_ASSERT(k, return nullptr);
|
2019-02-06 12:50:51 +01:00
|
|
|
return new Internal::DeviceKitAspectWidget(k, this);
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
QString DeviceKitAspect::displayNamePostfix(const Kit *k) const
|
2012-08-07 13:09:56 +02:00
|
|
|
{
|
2012-09-03 18:31:44 +02:00
|
|
|
IDevice::ConstPtr dev = device(k);
|
2012-08-07 13:09:56 +02:00
|
|
|
return dev.isNull() ? QString() : dev->displayName();
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
KitAspect::ItemList DeviceKitAspect::toUserOutput(const Kit *k) const
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2012-09-03 18:31:44 +02:00
|
|
|
IDevice::ConstPtr dev = device(k);
|
2019-05-28 08:28:51 +02:00
|
|
|
return {{tr("Device"), dev.isNull() ? tr("Unconfigured") : dev->displayName()}};
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
void DeviceKitAspect::addToMacroExpander(Kit *kit, Utils::MacroExpander *expander) const
|
2014-10-22 10:00:07 +02:00
|
|
|
{
|
2018-02-12 12:49:22 +01:00
|
|
|
QTC_ASSERT(kit, return);
|
2019-08-28 07:51:56 +00:00
|
|
|
expander->registerVariable("Device:HostAddress", tr("Host address"),
|
|
|
|
|
[kit]() -> QString {
|
|
|
|
|
const IDevice::ConstPtr device = DeviceKitAspect::device(kit);
|
|
|
|
|
return device ? device->sshParameters().host() : QString();
|
|
|
|
|
});
|
|
|
|
|
expander->registerVariable("Device:SshPort", tr("SSH port"),
|
|
|
|
|
[kit]() -> QString {
|
|
|
|
|
const IDevice::ConstPtr device = DeviceKitAspect::device(kit);
|
|
|
|
|
return device ? QString::number(device->sshParameters().port()) : QString();
|
|
|
|
|
});
|
|
|
|
|
expander->registerVariable("Device:UserName", tr("User name"),
|
|
|
|
|
[kit]() -> QString {
|
|
|
|
|
const IDevice::ConstPtr device = DeviceKitAspect::device(kit);
|
|
|
|
|
return device ? device->sshParameters().userName() : QString();
|
|
|
|
|
});
|
|
|
|
|
expander->registerVariable("Device:KeyFile", tr("Private key file"),
|
|
|
|
|
[kit]() -> QString {
|
|
|
|
|
const IDevice::ConstPtr device = DeviceKitAspect::device(kit);
|
|
|
|
|
return device ? device->sshParameters().privateKeyFile : QString();
|
|
|
|
|
});
|
|
|
|
|
expander->registerVariable("Device:Name", tr("Device name"),
|
|
|
|
|
[kit]() -> QString {
|
|
|
|
|
const IDevice::ConstPtr device = DeviceKitAspect::device(kit);
|
|
|
|
|
return device ? device->displayName() : QString();
|
|
|
|
|
});
|
2014-10-22 10:00:07 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
Core::Id DeviceKitAspect::id()
|
2013-09-09 17:10:41 +02:00
|
|
|
{
|
|
|
|
|
return "PE.Profile.Device";
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
IDevice::ConstPtr DeviceKitAspect::device(const Kit *k)
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2013-02-28 14:06:17 +01:00
|
|
|
QTC_ASSERT(DeviceManager::instance()->isLoaded(), return IDevice::ConstPtr());
|
2013-02-28 17:07:05 +01:00
|
|
|
return DeviceManager::instance()->find(deviceId(k));
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
Core::Id DeviceKitAspect::deviceId(const Kit *k)
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2019-02-06 12:50:51 +01:00
|
|
|
return k ? Core::Id::fromSetting(k->value(DeviceKitAspect::id())) : Core::Id();
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
void DeviceKitAspect::setDevice(Kit *k, IDevice::ConstPtr dev)
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2013-01-18 12:29:40 +01:00
|
|
|
setDeviceId(k, dev ? dev->id() : Core::Id());
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
void DeviceKitAspect::setDeviceId(Kit *k, Core::Id id)
|
2012-04-24 15:49:09 +02:00
|
|
|
{
|
2018-02-12 12:49:22 +01:00
|
|
|
QTC_ASSERT(k, return);
|
2019-02-06 12:50:51 +01:00
|
|
|
k->setValue(DeviceKitAspect::id(), id.toSetting());
|
2013-07-24 16:22:37 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
void DeviceKitAspect::kitsWereLoaded()
|
2012-10-17 17:27:17 +02:00
|
|
|
{
|
2013-08-21 12:48:46 +02:00
|
|
|
foreach (Kit *k, KitManager::kits())
|
2013-02-28 14:06:17 +01:00
|
|
|
fix(k);
|
|
|
|
|
|
2013-07-08 16:39:03 +02:00
|
|
|
DeviceManager *dm = DeviceManager::instance();
|
2019-02-06 12:50:51 +01:00
|
|
|
connect(dm, &DeviceManager::deviceListReplaced, this, &DeviceKitAspect::devicesChanged);
|
|
|
|
|
connect(dm, &DeviceManager::deviceAdded, this, &DeviceKitAspect::devicesChanged);
|
|
|
|
|
connect(dm, &DeviceManager::deviceRemoved, this, &DeviceKitAspect::devicesChanged);
|
|
|
|
|
connect(dm, &DeviceManager::deviceUpdated, this, &DeviceKitAspect::deviceUpdated);
|
2016-01-29 16:38:37 +02:00
|
|
|
|
|
|
|
|
connect(KitManager::instance(), &KitManager::kitUpdated,
|
2019-02-06 12:50:51 +01:00
|
|
|
this, &DeviceKitAspect::kitUpdated);
|
2016-01-29 16:38:37 +02:00
|
|
|
connect(KitManager::instance(), &KitManager::unmanagedKitUpdated,
|
2019-02-06 12:50:51 +01:00
|
|
|
this, &DeviceKitAspect::kitUpdated);
|
2013-02-28 14:06:17 +01:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
void DeviceKitAspect::deviceUpdated(Core::Id id)
|
2013-02-28 14:06:17 +01:00
|
|
|
{
|
2013-08-21 12:48:46 +02:00
|
|
|
foreach (Kit *k, KitManager::kits()) {
|
2012-10-17 17:27:17 +02:00
|
|
|
if (deviceId(k) == id)
|
|
|
|
|
notifyAboutUpdate(k);
|
2013-02-28 14:06:17 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
void DeviceKitAspect::kitUpdated(Kit *k)
|
2013-02-28 18:29:16 +01:00
|
|
|
{
|
|
|
|
|
setup(k); // Set default device if necessary
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
void DeviceKitAspect::devicesChanged()
|
2013-02-28 14:06:17 +01:00
|
|
|
{
|
2013-08-21 12:48:46 +02:00
|
|
|
foreach (Kit *k, KitManager::kits())
|
2013-02-28 18:29:16 +01:00
|
|
|
setup(k); // Set default device if necessary
|
2012-10-17 17:27:17 +02:00
|
|
|
}
|
|
|
|
|
|
2014-07-22 21:12:34 +02:00
|
|
|
// --------------------------------------------------------------------------
|
2019-02-06 12:50:51 +01:00
|
|
|
// EnvironmentKitAspect:
|
2014-07-22 21:12:34 +02:00
|
|
|
// --------------------------------------------------------------------------
|
2019-02-06 15:04:17 +01:00
|
|
|
namespace Internal {
|
|
|
|
|
class EnvironmentKitAspectWidget : public KitAspectWidget
|
|
|
|
|
{
|
|
|
|
|
Q_DECLARE_TR_FUNCTIONS(ProjectExplorer::EnvironmentKitAspect)
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
EnvironmentKitAspectWidget(Kit *workingCopy, const KitAspect *ki)
|
|
|
|
|
: KitAspectWidget(workingCopy, ki),
|
2019-11-21 15:35:24 +01:00
|
|
|
m_summaryLabel(new Utils::ElidingLabel),
|
2019-02-06 15:04:17 +01:00
|
|
|
m_manageButton(new QPushButton),
|
|
|
|
|
m_mainWidget(new QWidget)
|
|
|
|
|
{
|
|
|
|
|
auto *layout = new QVBoxLayout;
|
|
|
|
|
layout->setContentsMargins(0, 0, 0, 0);
|
|
|
|
|
layout->addWidget(m_summaryLabel);
|
|
|
|
|
if (Utils::HostOsInfo::isWindowsHost())
|
|
|
|
|
initMSVCOutputSwitch(layout);
|
|
|
|
|
m_mainWidget->setLayout(layout);
|
|
|
|
|
refresh();
|
|
|
|
|
m_manageButton->setText(tr("Change..."));
|
|
|
|
|
connect(m_manageButton, &QAbstractButton::clicked,
|
|
|
|
|
this, &EnvironmentKitAspectWidget::editEnvironmentChanges);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
QWidget *mainWidget() const override { return m_mainWidget; }
|
|
|
|
|
QWidget *buttonWidget() const override { return m_manageButton; }
|
|
|
|
|
void makeReadOnly() override { m_manageButton->setEnabled(false); }
|
|
|
|
|
|
|
|
|
|
void refresh() override
|
|
|
|
|
{
|
2019-05-07 16:51:22 +02:00
|
|
|
const Utils::EnvironmentItems changes = currentEnvironment();
|
2019-11-21 15:35:24 +01:00
|
|
|
const QString shortSummary = Utils::EnvironmentItem::toStringList(changes).join("; ");
|
2019-02-06 15:04:17 +01:00
|
|
|
m_summaryLabel->setText(shortSummary.isEmpty() ? tr("No changes to apply.") : shortSummary);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void editEnvironmentChanges()
|
|
|
|
|
{
|
|
|
|
|
Utils::MacroExpander *expander = m_kit->macroExpander();
|
|
|
|
|
Utils::EnvironmentDialog::Polisher polisher = [expander](QWidget *w) {
|
|
|
|
|
Core::VariableChooser::addSupportForChildWidgets(w, expander);
|
|
|
|
|
};
|
2019-05-07 16:51:22 +02:00
|
|
|
auto changes = Utils::EnvironmentDialog::getEnvironmentItems(m_summaryLabel,
|
|
|
|
|
currentEnvironment(),
|
|
|
|
|
QString(),
|
|
|
|
|
polisher);
|
|
|
|
|
if (!changes)
|
2019-02-06 15:04:17 +01:00
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (Utils::HostOsInfo::isWindowsHost()) {
|
|
|
|
|
const Utils::EnvironmentItem forceMSVCEnglishItem("VSLANG", "1033");
|
2019-05-07 16:51:22 +02:00
|
|
|
if (m_vslangCheckbox->isChecked() && changes->indexOf(forceMSVCEnglishItem) < 0)
|
|
|
|
|
changes->append(forceMSVCEnglishItem);
|
2019-02-06 15:04:17 +01:00
|
|
|
}
|
|
|
|
|
|
2019-05-07 16:51:22 +02:00
|
|
|
EnvironmentKitAspect::setEnvironmentChanges(m_kit, *changes);
|
2019-02-06 15:04:17 +01:00
|
|
|
}
|
|
|
|
|
|
2019-05-07 16:51:22 +02:00
|
|
|
Utils::EnvironmentItems currentEnvironment() const
|
2019-02-06 15:04:17 +01:00
|
|
|
{
|
2019-05-07 16:51:22 +02:00
|
|
|
Utils::EnvironmentItems changes = EnvironmentKitAspect::environmentChanges(m_kit);
|
2019-02-06 15:04:17 +01:00
|
|
|
|
|
|
|
|
if (Utils::HostOsInfo::isWindowsHost()) {
|
|
|
|
|
const Utils::EnvironmentItem forceMSVCEnglishItem("VSLANG", "1033");
|
|
|
|
|
if (changes.indexOf(forceMSVCEnglishItem) >= 0) {
|
|
|
|
|
m_vslangCheckbox->setCheckState(Qt::Checked);
|
|
|
|
|
changes.removeAll(forceMSVCEnglishItem);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Utils::sort(changes, [](const Utils::EnvironmentItem &lhs, const Utils::EnvironmentItem &rhs)
|
|
|
|
|
{ return QString::localeAwareCompare(lhs.name, rhs.name) < 0; });
|
|
|
|
|
return changes;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void initMSVCOutputSwitch(QVBoxLayout *layout)
|
|
|
|
|
{
|
|
|
|
|
m_vslangCheckbox = new QCheckBox(tr("Force UTF-8 MSVC compiler output"));
|
|
|
|
|
layout->addWidget(m_vslangCheckbox);
|
|
|
|
|
m_vslangCheckbox->setToolTip(tr("Either switches MSVC to English or keeps the language and "
|
|
|
|
|
"just forces UTF-8 output (may vary depending on the used MSVC "
|
|
|
|
|
"compiler)."));
|
|
|
|
|
connect(m_vslangCheckbox, &QCheckBox::toggled, this, [this](bool checked) {
|
2019-05-07 16:51:22 +02:00
|
|
|
Utils::EnvironmentItems changes = EnvironmentKitAspect::environmentChanges(m_kit);
|
2019-02-06 15:04:17 +01:00
|
|
|
const Utils::EnvironmentItem forceMSVCEnglishItem("VSLANG", "1033");
|
|
|
|
|
if (!checked && changes.indexOf(forceMSVCEnglishItem) >= 0)
|
|
|
|
|
changes.removeAll(forceMSVCEnglishItem);
|
|
|
|
|
if (checked && changes.indexOf(forceMSVCEnglishItem) < 0)
|
|
|
|
|
changes.append(forceMSVCEnglishItem);
|
|
|
|
|
EnvironmentKitAspect::setEnvironmentChanges(m_kit, changes);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-21 15:35:24 +01:00
|
|
|
Utils::ElidingLabel *m_summaryLabel;
|
2019-02-06 15:04:17 +01:00
|
|
|
QPushButton *m_manageButton;
|
|
|
|
|
QCheckBox *m_vslangCheckbox;
|
|
|
|
|
QWidget *m_mainWidget;
|
|
|
|
|
};
|
|
|
|
|
} // namespace Internal
|
2014-07-22 21:12:34 +02:00
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
EnvironmentKitAspect::EnvironmentKitAspect()
|
2014-07-22 21:12:34 +02:00
|
|
|
{
|
2019-02-06 12:50:51 +01:00
|
|
|
setObjectName(QLatin1String("EnvironmentKitAspect"));
|
|
|
|
|
setId(EnvironmentKitAspect::id());
|
2019-02-06 16:16:07 +01:00
|
|
|
setDisplayName(tr("Environment"));
|
|
|
|
|
setDescription(tr("Additional build environment settings when using this kit."));
|
2014-07-22 21:12:34 +02:00
|
|
|
setPriority(29000);
|
|
|
|
|
}
|
|
|
|
|
|
2019-05-27 16:09:44 +02:00
|
|
|
Tasks EnvironmentKitAspect::validate(const Kit *k) const
|
2014-07-22 21:12:34 +02:00
|
|
|
{
|
2019-05-27 16:09:44 +02:00
|
|
|
Tasks result;
|
2018-02-12 12:49:22 +01:00
|
|
|
QTC_ASSERT(k, return result);
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
const QVariant variant = k->value(EnvironmentKitAspect::id());
|
2020-01-15 08:56:11 +01:00
|
|
|
if (!variant.isNull() && !variant.canConvert(QVariant::List))
|
|
|
|
|
result << BuildSystemTask(Task::Error, tr("The environment setting value is invalid."));
|
|
|
|
|
|
2014-07-22 21:12:34 +02:00
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
void EnvironmentKitAspect::fix(Kit *k)
|
2014-07-22 21:12:34 +02:00
|
|
|
{
|
2018-02-12 12:49:22 +01:00
|
|
|
QTC_ASSERT(k, return);
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
const QVariant variant = k->value(EnvironmentKitAspect::id());
|
2014-07-22 21:12:34 +02:00
|
|
|
if (!variant.isNull() && !variant.canConvert(QVariant::List)) {
|
|
|
|
|
qWarning("Kit \"%s\" has a wrong environment value set.", qPrintable(k->displayName()));
|
2019-05-07 16:51:22 +02:00
|
|
|
setEnvironmentChanges(k, Utils::EnvironmentItems());
|
2014-07-22 21:12:34 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
void EnvironmentKitAspect::addToEnvironment(const Kit *k, Utils::Environment &env) const
|
2014-07-22 21:12:34 +02:00
|
|
|
{
|
2018-02-12 12:49:22 +01:00
|
|
|
const QStringList values
|
|
|
|
|
= Utils::transform(Utils::EnvironmentItem::toStringList(environmentChanges(k)),
|
|
|
|
|
[k](const QString &v) { return k->macroExpander()->expand(v); });
|
|
|
|
|
env.modify(Utils::EnvironmentItem::fromStringList(values));
|
2014-07-22 21:12:34 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
KitAspectWidget *EnvironmentKitAspect::createConfigWidget(Kit *k) const
|
2014-07-22 21:12:34 +02:00
|
|
|
{
|
2018-02-12 12:49:22 +01:00
|
|
|
QTC_ASSERT(k, return nullptr);
|
2019-02-06 12:50:51 +01:00
|
|
|
return new Internal::EnvironmentKitAspectWidget(k, this);
|
2014-07-22 21:12:34 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
KitAspect::ItemList EnvironmentKitAspect::toUserOutput(const Kit *k) const
|
2014-07-22 21:12:34 +02:00
|
|
|
{
|
2018-02-12 12:49:22 +01:00
|
|
|
return { qMakePair(tr("Environment"),
|
|
|
|
|
Utils::EnvironmentItem::toStringList(environmentChanges(k)).join("<br>")) };
|
2014-07-22 21:12:34 +02:00
|
|
|
}
|
|
|
|
|
|
2019-02-06 12:50:51 +01:00
|
|
|
Core::Id EnvironmentKitAspect::id()
|
2014-07-22 21:12:34 +02:00
|
|
|
{
|
|
|
|
|
return "PE.Profile.Environment";
|
|
|
|
|
}
|
|
|
|
|
|
2019-05-07 16:51:22 +02:00
|
|
|
Utils::EnvironmentItems EnvironmentKitAspect::environmentChanges(const Kit *k)
|
2014-07-22 21:12:34 +02:00
|
|
|
{
|
|
|
|
|
if (k)
|
2019-02-06 12:50:51 +01:00
|
|
|
return Utils::EnvironmentItem::fromStringList(k->value(EnvironmentKitAspect::id()).toStringList());
|
2019-05-07 16:51:22 +02:00
|
|
|
return Utils::EnvironmentItems();
|
2014-07-22 21:12:34 +02:00
|
|
|
}
|
|
|
|
|
|
2019-05-07 16:51:22 +02:00
|
|
|
void EnvironmentKitAspect::setEnvironmentChanges(Kit *k, const Utils::EnvironmentItems &changes)
|
2014-07-22 21:12:34 +02:00
|
|
|
{
|
|
|
|
|
if (k)
|
2019-02-06 12:50:51 +01:00
|
|
|
k->setValue(EnvironmentKitAspect::id(), Utils::EnvironmentItem::toStringList(changes));
|
2014-07-22 21:12:34 +02:00
|
|
|
}
|
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
} // namespace ProjectExplorer
|