2019-10-22 14:33:31 +02:00
|
|
|
/****************************************************************************
|
|
|
|
|
**
|
|
|
|
|
** Copyright (C) 2016 BlackBerry Limited. All rights reserved.
|
|
|
|
|
** Contact: BlackBerry (qt@blackberry.com)
|
|
|
|
|
**
|
|
|
|
|
** 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 "mcusupportconstants.h"
|
|
|
|
|
#include "mcusupportoptions.h"
|
|
|
|
|
|
|
|
|
|
#include <coreplugin/icore.h>
|
|
|
|
|
#include <cmakeprojectmanager/cmakekitinformation.h>
|
|
|
|
|
#include <debugger/debuggeritem.h>
|
|
|
|
|
#include <debugger/debuggeritemmanager.h>
|
|
|
|
|
#include <debugger/debuggerkitinformation.h>
|
|
|
|
|
#include <projectexplorer/projectexplorerconstants.h>
|
|
|
|
|
#include <projectexplorer/toolchain.h>
|
|
|
|
|
#include <projectexplorer/toolchainmanager.h>
|
|
|
|
|
#include <projectexplorer/kitmanager.h>
|
|
|
|
|
#include <projectexplorer/kitinformation.h>
|
|
|
|
|
#include <projectexplorer/devicesupport/devicemanager.h>
|
|
|
|
|
#include <utils/algorithm.h>
|
|
|
|
|
#include <utils/fileutils.h>
|
|
|
|
|
#include <utils/pathchooser.h>
|
|
|
|
|
#include <utils/qtcassert.h>
|
|
|
|
|
#include <utils/utilsicons.h>
|
|
|
|
|
|
|
|
|
|
#include <QDesktopServices>
|
|
|
|
|
#include <QDir>
|
|
|
|
|
#include <QFileInfo>
|
|
|
|
|
#include <QLabel>
|
|
|
|
|
#include <QToolButton>
|
|
|
|
|
#include <QVBoxLayout>
|
|
|
|
|
#include <QVariant>
|
|
|
|
|
|
|
|
|
|
namespace McuSupport {
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
|
|
|
|
PackageOptions::PackageOptions(const QString &label, const QString &defaultPath,
|
|
|
|
|
const QString &detectionPath, const QString &settingsKey)
|
|
|
|
|
: m_label(label)
|
|
|
|
|
, m_defaultPath(defaultPath)
|
|
|
|
|
, m_detectionPath(detectionPath)
|
|
|
|
|
, m_settingsKey(settingsKey)
|
|
|
|
|
{
|
|
|
|
|
QSettings *s = Core::ICore::settings();
|
|
|
|
|
s->beginGroup(Constants::SETTINGS_GROUP);
|
|
|
|
|
m_path = s->value(QLatin1String(Constants::SETTINGS_KEY_PACKAGE_PREFIX) + m_settingsKey,
|
|
|
|
|
m_defaultPath).toString();
|
|
|
|
|
s->endGroup();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString PackageOptions::path() const
|
|
|
|
|
{
|
|
|
|
|
return QFileInfo(m_fileChooser->path() + m_relativePathModifier).absoluteFilePath();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString PackageOptions::label() const
|
|
|
|
|
{
|
|
|
|
|
return m_label;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString PackageOptions::detectionPath() const
|
|
|
|
|
{
|
|
|
|
|
return m_detectionPath;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QWidget *PackageOptions::widget()
|
|
|
|
|
{
|
|
|
|
|
if (m_widget)
|
|
|
|
|
return m_widget;
|
|
|
|
|
|
|
|
|
|
m_widget = new QWidget;
|
|
|
|
|
m_fileChooser = new Utils::PathChooser;
|
|
|
|
|
|
|
|
|
|
auto layout = new QGridLayout(m_widget);
|
|
|
|
|
layout->setContentsMargins(0, 0, 0, 0);
|
|
|
|
|
m_statusIcon = new QLabel;
|
|
|
|
|
m_statusIcon->setSizePolicy(QSizePolicy::Maximum, QSizePolicy::MinimumExpanding);
|
|
|
|
|
m_statusIcon->setAlignment(Qt::AlignTop);
|
|
|
|
|
m_statusLabel = new QLabel;
|
|
|
|
|
m_statusLabel->setWordWrap(true);
|
|
|
|
|
m_statusLabel->setAlignment(Qt::AlignTop | Qt::AlignLeft);
|
|
|
|
|
|
|
|
|
|
if (!m_downloadUrl.isEmpty()) {
|
|
|
|
|
auto downLoadButton = new QToolButton;
|
|
|
|
|
downLoadButton->setIcon(Utils::Icons::DOWNLOAD.icon());
|
|
|
|
|
downLoadButton->setToolTip(tr("Download from \"%1\"").arg(m_downloadUrl));
|
|
|
|
|
QObject::connect(downLoadButton, &QToolButton::pressed, [this]{
|
|
|
|
|
QDesktopServices::openUrl(m_downloadUrl);
|
|
|
|
|
});
|
|
|
|
|
layout->addWidget(downLoadButton, 0, 2);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
layout->addWidget(m_fileChooser, 0, 0, 1, 2);
|
|
|
|
|
layout->addWidget(m_statusIcon, 1, 0);
|
|
|
|
|
layout->addWidget(m_statusLabel, 1, 1, 1, -1);
|
|
|
|
|
|
2019-11-13 18:37:27 +01:00
|
|
|
m_fileChooser->setPath(m_path);
|
|
|
|
|
|
|
|
|
|
QObject::connect(m_fileChooser, &Utils::PathChooser::pathChanged,
|
|
|
|
|
[this](){
|
|
|
|
|
updateStatus();
|
|
|
|
|
emit changed();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
updateStatus();
|
2019-10-22 14:33:31 +02:00
|
|
|
return m_widget;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
PackageOptions::Status PackageOptions::status() const
|
|
|
|
|
{
|
|
|
|
|
return m_status;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void PackageOptions::setDownloadUrl(const QString &url)
|
|
|
|
|
{
|
|
|
|
|
m_downloadUrl = url;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void PackageOptions::setEnvironmentVariableName(const QString &name)
|
|
|
|
|
{
|
|
|
|
|
m_environmentVariableName = name;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString PackageOptions::environmentVariableName() const
|
|
|
|
|
{
|
|
|
|
|
return m_environmentVariableName;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void PackageOptions::setAddToPath(bool addToPath)
|
|
|
|
|
{
|
|
|
|
|
m_addToPath = addToPath;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool PackageOptions::addToPath() const
|
|
|
|
|
{
|
|
|
|
|
return m_addToPath;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void PackageOptions::writeToSettings() const
|
|
|
|
|
{
|
|
|
|
|
if (m_path.compare(m_defaultPath) == 0)
|
|
|
|
|
return;
|
|
|
|
|
QSettings *s = Core::ICore::settings();
|
|
|
|
|
s->beginGroup(Constants::SETTINGS_GROUP);
|
|
|
|
|
s->setValue(QLatin1String(Constants::SETTINGS_KEY_PACKAGE_PREFIX) + m_settingsKey, m_path);
|
|
|
|
|
s->endGroup();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void PackageOptions::setRelativePathModifier(const QString &path)
|
|
|
|
|
{
|
|
|
|
|
m_relativePathModifier = path;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void PackageOptions::updateStatus()
|
|
|
|
|
{
|
|
|
|
|
m_path = m_fileChooser->rawPath();
|
|
|
|
|
const bool validPath = m_fileChooser->isValid();
|
|
|
|
|
const Utils::FilePath detectionPath = Utils::FilePath::fromString(
|
|
|
|
|
m_fileChooser->path() + "/" + m_detectionPath);
|
|
|
|
|
const QString displayDetectionPath = Utils::FilePath::fromString(m_detectionPath).toUserOutput();
|
|
|
|
|
const bool validPackage = detectionPath.exists();
|
|
|
|
|
|
|
|
|
|
m_status = validPath ? (validPackage ? ValidPackage : ValidPathInvalidPackage) : InvalidPath;
|
|
|
|
|
|
|
|
|
|
static const QPixmap okIcon = Utils::Icons::OK.pixmap();
|
|
|
|
|
static const QPixmap notOkIcon = Utils::Icons::BROKEN.pixmap();
|
|
|
|
|
m_statusIcon->setPixmap(m_status == ValidPackage ? okIcon : notOkIcon);
|
|
|
|
|
|
|
|
|
|
QString statusText;
|
|
|
|
|
switch (m_status) {
|
|
|
|
|
case ValidPackage:
|
|
|
|
|
statusText = tr("Path is valid, \"%1\" was found.").arg(displayDetectionPath);
|
|
|
|
|
break;
|
|
|
|
|
case ValidPathInvalidPackage:
|
|
|
|
|
statusText = tr("Path exists, but does not contain \"%1\".").arg(displayDetectionPath);
|
|
|
|
|
break;
|
|
|
|
|
case InvalidPath:
|
|
|
|
|
statusText = tr("Path does not exist.");
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
m_statusLabel->setText(statusText);
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-04 18:01:54 +01:00
|
|
|
BoardOptions::BoardOptions(const QString &vendor, const QString &model,
|
|
|
|
|
const QString &toolChainFileName, const QString &qulPlatform,
|
|
|
|
|
const QVector<PackageOptions*> &packages)
|
|
|
|
|
: m_vendor(vendor)
|
|
|
|
|
, m_model(model)
|
2019-10-22 14:33:31 +02:00
|
|
|
, m_toolChainFile(toolChainFileName)
|
2019-11-02 06:19:51 +01:00
|
|
|
, m_qulPlatform(qulPlatform)
|
2019-10-22 14:33:31 +02:00
|
|
|
, m_packages(packages)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString BoardOptions::model() const
|
|
|
|
|
{
|
|
|
|
|
return m_model;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString BoardOptions::toolChainFile() const
|
|
|
|
|
{
|
|
|
|
|
return m_toolChainFile;
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-02 06:19:51 +01:00
|
|
|
QString BoardOptions::qulPlatform() const
|
|
|
|
|
{
|
|
|
|
|
return m_qulPlatform;
|
|
|
|
|
}
|
|
|
|
|
|
2019-10-22 14:33:31 +02:00
|
|
|
QVector<PackageOptions *> BoardOptions::packages() const
|
|
|
|
|
{
|
|
|
|
|
return m_packages;
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-13 18:37:27 +01:00
|
|
|
bool BoardOptions::isValid() const
|
|
|
|
|
{
|
|
|
|
|
return !Utils::anyOf(packages(), [](PackageOptions *package) {
|
|
|
|
|
return package->status() != PackageOptions::ValidPackage;
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-04 18:01:54 +01:00
|
|
|
QString BoardOptions::vendor() const
|
|
|
|
|
{
|
|
|
|
|
return m_vendor;
|
|
|
|
|
}
|
|
|
|
|
|
2019-10-22 14:33:31 +02:00
|
|
|
static PackageOptions *createQulPackage()
|
|
|
|
|
{
|
|
|
|
|
auto result = new PackageOptions(
|
2019-11-14 11:26:07 +01:00
|
|
|
PackageOptions::tr("Qt for MCUs SDK"),
|
2019-10-22 14:33:31 +02:00
|
|
|
QDir::homePath(),
|
|
|
|
|
Utils::HostOsInfo::withExecutableSuffix("bin/qmltocpp"),
|
2019-11-14 10:38:12 +01:00
|
|
|
"QtMCUSdk");
|
2019-10-22 14:33:31 +02:00
|
|
|
result->setEnvironmentVariableName("Qul_DIR");
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static PackageOptions *createArmGccPackage()
|
|
|
|
|
{
|
2019-11-01 07:44:31 +01:00
|
|
|
const char envVar[] = "ARMGCC_DIR";
|
|
|
|
|
|
|
|
|
|
QString defaultPath;
|
|
|
|
|
if (qEnvironmentVariableIsSet(envVar))
|
|
|
|
|
defaultPath = qEnvironmentVariable(envVar);
|
|
|
|
|
if (defaultPath.isEmpty() && Utils::HostOsInfo::isWindowsHost()) {
|
|
|
|
|
const QDir installDir(
|
|
|
|
|
qEnvironmentVariable("ProgramFiles(x86)") + "/GNU Tools ARM Embedded/");
|
|
|
|
|
if (installDir.exists()) {
|
|
|
|
|
// If GNU Tools installation dir has only one sub dir,
|
|
|
|
|
// select the sub dir, otherwise the installation dir.
|
|
|
|
|
const QFileInfoList subDirs =
|
|
|
|
|
installDir.entryInfoList(QDir::Dirs | QDir::NoDotAndDotDot);
|
|
|
|
|
if (subDirs.count() == 1)
|
|
|
|
|
defaultPath = subDirs.first().filePath() + '/';
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (defaultPath.isEmpty())
|
|
|
|
|
defaultPath = QDir::homePath();
|
|
|
|
|
|
2019-10-22 14:33:31 +02:00
|
|
|
auto result = new PackageOptions(
|
|
|
|
|
PackageOptions::tr("GNU Arm Embedded Toolchain"),
|
|
|
|
|
defaultPath,
|
|
|
|
|
Utils::HostOsInfo::withExecutableSuffix("bin/arm-none-eabi-g++"),
|
2019-11-14 10:38:12 +01:00
|
|
|
"GNUArmEmbeddedToolchain");
|
2019-10-22 14:33:31 +02:00
|
|
|
result->setDownloadUrl(
|
|
|
|
|
"https://developer.arm.com/open-source/gnu-toolchain/gnu-rm/downloads");
|
2019-11-01 07:44:31 +01:00
|
|
|
result->setEnvironmentVariableName(envVar);
|
2019-10-22 14:33:31 +02:00
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static PackageOptions *createStm32CubeFwF7SdkPackage()
|
|
|
|
|
{
|
|
|
|
|
auto result = new PackageOptions(
|
|
|
|
|
PackageOptions::tr("STM32Cube SDK"),
|
|
|
|
|
"%{Env:STM32Cube_FW_F7_SDK_PATH}",
|
|
|
|
|
"Drivers/STM32F7xx_HAL_Driver",
|
2019-11-14 10:38:12 +01:00
|
|
|
"Stm32CubeFwF7Sdk");
|
2019-10-22 14:33:31 +02:00
|
|
|
result->setDownloadUrl(
|
|
|
|
|
"https://www.st.com/content/st_com/en/products/embedded-software/mcus-embedded-software/stm32-embedded-software/stm32cube-mcu-packages/stm32cubef7.html");
|
|
|
|
|
result->setEnvironmentVariableName("STM32Cube_FW_F7_SDK_PATH");
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static PackageOptions *createStm32CubeProgrammerPackage()
|
|
|
|
|
{
|
|
|
|
|
const QString defaultPath =
|
|
|
|
|
Utils::HostOsInfo::isWindowsHost() ?
|
|
|
|
|
QDir::fromNativeSeparators(qEnvironmentVariable("ProgramFiles"))
|
|
|
|
|
+ "/STMicroelectronics/STM32Cube/STM32CubeProgrammer/"
|
|
|
|
|
: QDir::homePath();
|
|
|
|
|
auto result = new PackageOptions(
|
|
|
|
|
PackageOptions::tr("STM32CubeProgrammer"),
|
|
|
|
|
defaultPath,
|
|
|
|
|
QLatin1String(Utils::HostOsInfo::isWindowsHost() ? "/bin/STM32_Programmer_CLI.exe"
|
|
|
|
|
: "/bin/STM32_Programmer.sh"),
|
2019-11-14 10:38:12 +01:00
|
|
|
"Stm32CubeProgrammer");
|
2019-10-22 14:33:31 +02:00
|
|
|
result->setRelativePathModifier("/bin");
|
|
|
|
|
result->setDownloadUrl(
|
|
|
|
|
"https://www.st.com/en/development-tools/stm32cubeprog.html");
|
|
|
|
|
result->setAddToPath(true);
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static PackageOptions *createEvkbImxrt1050SdkPackage()
|
|
|
|
|
{
|
|
|
|
|
auto result = new PackageOptions(
|
2019-10-30 18:55:11 +01:00
|
|
|
PackageOptions::tr("NXP i.MXRT SDK"),
|
|
|
|
|
"%{Env:EVKB_IMXRT1050_SDK_PATH}", // TODO: Try to not use 1050 specifics
|
2019-10-22 14:33:31 +02:00
|
|
|
"EVKB-IMXRT1050_manifest_v3_5.xml",
|
2019-11-14 10:38:12 +01:00
|
|
|
"EvkbImxrt1050Sdk");
|
2019-10-22 14:33:31 +02:00
|
|
|
result->setDownloadUrl("https://mcuxpresso.nxp.com/en/welcome");
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static PackageOptions *createSeggerJLinkPackage()
|
|
|
|
|
{
|
|
|
|
|
const QString defaultPath =
|
|
|
|
|
Utils::HostOsInfo::isWindowsHost() ?
|
2019-11-04 16:24:19 +01:00
|
|
|
QDir::fromNativeSeparators(qEnvironmentVariable("ProgramFiles(x86)"))
|
|
|
|
|
+ "/SEGGER/JLink"
|
2019-10-22 14:33:31 +02:00
|
|
|
: QString("%{Env:SEGGER_JLINK_SOFTWARE_AND_DOCUMENTATION_PATH}");
|
|
|
|
|
auto result = new PackageOptions(
|
|
|
|
|
PackageOptions::tr("SEGGER JLink"),
|
|
|
|
|
defaultPath,
|
|
|
|
|
Utils::HostOsInfo::withExecutableSuffix("JLink"),
|
2019-11-14 10:38:12 +01:00
|
|
|
"SeggerJLink");
|
2019-10-22 14:33:31 +02:00
|
|
|
result->setDownloadUrl("https://www.segger.com/downloads/jlink");
|
2019-10-30 18:55:57 +01:00
|
|
|
result->setEnvironmentVariableName("SEGGER_JLINK_SOFTWARE_AND_DOCUMENTATION_PATH");
|
2019-10-22 14:33:31 +02:00
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
McuSupportOptions::McuSupportOptions(QObject *parent)
|
|
|
|
|
: QObject(parent)
|
|
|
|
|
{
|
|
|
|
|
PackageOptions* qulPackage = createQulPackage();
|
|
|
|
|
PackageOptions* armGccPackage = createArmGccPackage();
|
|
|
|
|
PackageOptions* stm32CubeFwF7SdkPackage = createStm32CubeFwF7SdkPackage();
|
|
|
|
|
PackageOptions* stm32CubeProgrammerPackage = createStm32CubeProgrammerPackage();
|
|
|
|
|
PackageOptions* evkbImxrt1050SdkPackage = createEvkbImxrt1050SdkPackage();
|
|
|
|
|
PackageOptions* seggerJLinkPackage = createSeggerJLinkPackage();
|
|
|
|
|
|
|
|
|
|
toolchainPackage = armGccPackage;
|
2019-11-12 10:39:18 +01:00
|
|
|
qulSdkPackage = qulPackage;
|
2019-10-22 14:33:31 +02:00
|
|
|
|
2019-11-12 13:58:23 +01:00
|
|
|
auto stmEvalPackages = {
|
|
|
|
|
armGccPackage, stm32CubeProgrammerPackage, qulPackage};
|
|
|
|
|
auto stmEngPackages = {
|
|
|
|
|
armGccPackage, stm32CubeFwF7SdkPackage, stm32CubeProgrammerPackage, qulPackage};
|
|
|
|
|
auto nxpEvalPackages = {
|
|
|
|
|
armGccPackage, seggerJLinkPackage, qulPackage};
|
|
|
|
|
auto nxpEngPackages = {
|
|
|
|
|
armGccPackage, evkbImxrt1050SdkPackage, seggerJLinkPackage, qulPackage};
|
|
|
|
|
auto desktopPackages = {
|
|
|
|
|
qulPackage};
|
|
|
|
|
packages = {
|
|
|
|
|
armGccPackage, stm32CubeFwF7SdkPackage, stm32CubeProgrammerPackage, evkbImxrt1050SdkPackage,
|
|
|
|
|
seggerJLinkPackage, qulPackage};
|
2019-10-22 14:33:31 +02:00
|
|
|
|
2019-11-04 18:01:54 +01:00
|
|
|
const QString vendorStm = "STM";
|
|
|
|
|
const QString vendorNxp = "NXP";
|
|
|
|
|
const QString vendorQt = "Qt";
|
2019-11-12 13:58:23 +01:00
|
|
|
|
|
|
|
|
boards.append(new BoardOptions(vendorStm,
|
|
|
|
|
"stm32f7508",
|
|
|
|
|
"CMake/stm32f7508-discovery.cmake", "", stmEvalPackages));
|
2019-11-04 18:01:54 +01:00
|
|
|
boards.append(new BoardOptions(vendorStm,
|
2019-11-12 13:58:23 +01:00
|
|
|
"stm32f769i",
|
|
|
|
|
"CMake/stm32f769i-discovery.cmake", "", stmEvalPackages));
|
2019-11-04 18:01:54 +01:00
|
|
|
boards.append(new BoardOptions(vendorStm,
|
2019-11-12 13:58:23 +01:00
|
|
|
"Engineering",
|
|
|
|
|
"CMake/<toolchain file>", "", stmEngPackages));
|
|
|
|
|
|
|
|
|
|
boards.append(new BoardOptions(vendorNxp,
|
|
|
|
|
"evkbimxrt1050",
|
|
|
|
|
"CMake/evkbimxrt1050-toolchain.cmake", "", nxpEvalPackages));
|
2019-11-04 18:01:54 +01:00
|
|
|
boards.append(new BoardOptions(vendorNxp,
|
2019-11-12 13:58:23 +01:00
|
|
|
"Engineering",
|
|
|
|
|
"CMake/<toolchain file>", "", nxpEngPackages));
|
|
|
|
|
|
2019-11-04 18:01:54 +01:00
|
|
|
boards.append(new BoardOptions(vendorQt,
|
2019-11-12 13:58:23 +01:00
|
|
|
"Desktop", "", "Qt", desktopPackages));
|
2019-10-22 14:33:31 +02:00
|
|
|
|
|
|
|
|
for (auto package : packages)
|
|
|
|
|
connect(package, &PackageOptions::changed, [this](){
|
|
|
|
|
emit changed();
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
McuSupportOptions::~McuSupportOptions()
|
|
|
|
|
{
|
|
|
|
|
qDeleteAll(packages);
|
|
|
|
|
packages.clear();
|
|
|
|
|
qDeleteAll(boards);
|
|
|
|
|
boards.clear();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static ProjectExplorer::ToolChain* armGccToolchain(const Utils::FilePath &path, Core::Id language)
|
|
|
|
|
{
|
|
|
|
|
using namespace ProjectExplorer;
|
|
|
|
|
|
|
|
|
|
ToolChain *toolChain = ToolChainManager::toolChain([&path, language](const ToolChain *t){
|
|
|
|
|
return t->compilerCommand() == path && t->language() == language;
|
|
|
|
|
});
|
|
|
|
|
if (!toolChain) {
|
|
|
|
|
ToolChainFactory *gccFactory =
|
|
|
|
|
Utils::findOrDefault(ToolChainFactory::allToolChainFactories(), [](ToolChainFactory *f){
|
|
|
|
|
return f->supportedToolChainType() == ProjectExplorer::Constants::GCC_TOOLCHAIN_TYPEID;
|
|
|
|
|
});
|
|
|
|
|
if (gccFactory) {
|
|
|
|
|
const QList<ToolChain*> detected = gccFactory->detectForImport({path, language});
|
|
|
|
|
if (!detected.isEmpty()) {
|
|
|
|
|
toolChain = detected.first();
|
|
|
|
|
toolChain->setDetection(ToolChain::ManualDetection);
|
|
|
|
|
toolChain->setDisplayName("Arm GCC");
|
|
|
|
|
ToolChainManager::registerToolChain(toolChain);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return toolChain;
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-02 06:19:51 +01:00
|
|
|
static bool isDesktop(const BoardOptions* board)
|
|
|
|
|
{
|
|
|
|
|
return board->qulPlatform() == "Qt";
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-12 10:39:18 +01:00
|
|
|
static void setKitProperties(const QString &kitName, ProjectExplorer::Kit *k,
|
2019-11-12 15:14:20 +01:00
|
|
|
const BoardOptions* board)
|
2019-10-22 14:33:31 +02:00
|
|
|
{
|
|
|
|
|
using namespace ProjectExplorer;
|
|
|
|
|
|
2019-11-12 10:39:18 +01:00
|
|
|
k->setUnexpandedDisplayName(kitName);
|
2019-11-04 18:01:54 +01:00
|
|
|
k->setValue(Constants::KIT_BOARD_VENDOR_KEY, board->vendor());
|
2019-10-22 14:33:31 +02:00
|
|
|
k->setValue(Constants::KIT_BOARD_MODEL_KEY, board->model());
|
2019-11-13 18:37:27 +01:00
|
|
|
k->setAutoDetected(true);
|
|
|
|
|
k->makeSticky();
|
2019-11-13 23:39:08 +01:00
|
|
|
if (isDesktop(board)) {
|
|
|
|
|
k->setDeviceTypeForIcon(Constants::DEVICE_TYPE);
|
|
|
|
|
} else {
|
2019-11-02 06:19:51 +01:00
|
|
|
k->setIrrelevantAspects({SysRootKitAspect::id(),
|
|
|
|
|
"QtSupport.QtInformation" // QtKitAspect::id()
|
|
|
|
|
});
|
|
|
|
|
}
|
2019-10-22 14:33:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void setKitToolchains(ProjectExplorer::Kit *k, const QString &armGccPath)
|
|
|
|
|
{
|
|
|
|
|
using namespace ProjectExplorer;
|
|
|
|
|
|
|
|
|
|
const QString compileNameScheme = Utils::HostOsInfo::withExecutableSuffix(
|
|
|
|
|
armGccPath + "/bin/arm-none-eabi-%1");
|
|
|
|
|
ToolChain *cTc = armGccToolchain(
|
|
|
|
|
Utils::FilePath::fromUserInput(compileNameScheme.arg("gcc")),
|
|
|
|
|
ProjectExplorer::Constants::C_LANGUAGE_ID);
|
|
|
|
|
ToolChain *cxxTc = armGccToolchain(
|
|
|
|
|
Utils::FilePath::fromUserInput(compileNameScheme.arg("g++")),
|
|
|
|
|
ProjectExplorer::Constants::CXX_LANGUAGE_ID);
|
|
|
|
|
ToolChainKitAspect::setToolChain(k, cTc);
|
|
|
|
|
ToolChainKitAspect::setToolChain(k, cxxTc);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void setKitDebugger(ProjectExplorer::Kit *k, const QString &armGccPath)
|
|
|
|
|
{
|
|
|
|
|
using namespace Debugger;
|
|
|
|
|
|
|
|
|
|
const Utils::FilePath command = Utils::FilePath::fromUserInput(
|
|
|
|
|
Utils::HostOsInfo::withExecutableSuffix(armGccPath + "/bin/arm-none-eabi-gdb-py"));
|
|
|
|
|
const DebuggerItem *debugger = DebuggerItemManager::findByCommand(command);
|
|
|
|
|
QVariant debuggerId;
|
|
|
|
|
if (!debugger) {
|
|
|
|
|
DebuggerItem newDebugger;
|
|
|
|
|
newDebugger.setCommand(command);
|
|
|
|
|
newDebugger.setUnexpandedDisplayName(
|
|
|
|
|
PackageOptions::tr("Arm GDB at %1").arg(command.toUserOutput()));
|
|
|
|
|
debuggerId = DebuggerItemManager::registerDebugger(newDebugger);
|
|
|
|
|
} else {
|
|
|
|
|
debuggerId = debugger->id();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DebuggerKitAspect::setDebugger(k, debuggerId);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void setKitDevice(ProjectExplorer::Kit *k)
|
|
|
|
|
{
|
|
|
|
|
using namespace ProjectExplorer;
|
|
|
|
|
|
|
|
|
|
DeviceTypeKitAspect::setDeviceTypeId(k, Constants::DEVICE_TYPE);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void setKitEnvironment(ProjectExplorer::Kit *k, const BoardOptions* board)
|
|
|
|
|
{
|
|
|
|
|
using namespace ProjectExplorer;
|
|
|
|
|
|
|
|
|
|
Utils::EnvironmentItems changes;
|
|
|
|
|
QStringList pathAdditions;
|
|
|
|
|
for (auto package : board->packages()) {
|
|
|
|
|
if (package->addToPath())
|
|
|
|
|
pathAdditions.append(QDir::toNativeSeparators(package->path()));
|
|
|
|
|
if (!package->environmentVariableName().isEmpty())
|
|
|
|
|
changes.append({package->environmentVariableName(),
|
|
|
|
|
QDir::toNativeSeparators(package->path())});
|
|
|
|
|
}
|
2019-10-31 16:01:35 +01:00
|
|
|
pathAdditions.append("${Path}");
|
|
|
|
|
if (Utils::HostOsInfo::isWindowsHost())
|
|
|
|
|
pathAdditions.append(QDir::toNativeSeparators(Core::ICore::libexecPath())); // for jom
|
2019-10-31 16:04:59 +01:00
|
|
|
pathAdditions.append(QDir::toNativeSeparators(Core::ICore::libexecPath() + "/clang/bin"));
|
2019-10-31 16:01:35 +01:00
|
|
|
changes.append({"Path", pathAdditions.join(Utils::HostOsInfo::pathListSeparator())});
|
2019-10-22 14:33:31 +02:00
|
|
|
EnvironmentKitAspect::setEnvironmentChanges(k, changes);
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-12 10:39:18 +01:00
|
|
|
static void setKitCMakeOptions(ProjectExplorer::Kit *k, const BoardOptions* board,
|
|
|
|
|
const QString &qulDir)
|
2019-10-22 14:33:31 +02:00
|
|
|
{
|
|
|
|
|
using namespace CMakeProjectManager;
|
|
|
|
|
|
|
|
|
|
CMakeConfig config = CMakeConfigurationKitAspect::configuration(k);
|
2019-11-13 23:32:51 +01:00
|
|
|
config.append(CMakeConfigItem("CMAKE_CXX_COMPILER", "%{Compiler:Executable:Cxx}"));
|
|
|
|
|
config.append(CMakeConfigItem("CMAKE_C_COMPILER", "%{Compiler:Executable:C}"));
|
2019-11-02 06:19:51 +01:00
|
|
|
if (!board->toolChainFile().isEmpty())
|
|
|
|
|
config.append(CMakeConfigItem("CMAKE_TOOLCHAIN_FILE",
|
2019-11-12 10:39:18 +01:00
|
|
|
(qulDir + "/" + board->toolChainFile()).toUtf8()));
|
2019-11-02 06:19:51 +01:00
|
|
|
if (!board->qulPlatform().isEmpty())
|
|
|
|
|
config.append(CMakeConfigItem("QUL_PLATFORM",
|
|
|
|
|
board->qulPlatform().toUtf8()));
|
2019-11-07 13:27:18 +01:00
|
|
|
if (isDesktop(board)) {
|
|
|
|
|
config.append(CMakeConfigItem("CMAKE_PREFIX_PATH", "%{Qt:QT_INSTALL_PREFIX}"));
|
|
|
|
|
// TODO: Hack! Implement color depth variants of all targets
|
2019-11-02 06:19:51 +01:00
|
|
|
config.append(CMakeConfigItem("QUL_COLOR_DEPTH", "32"));
|
2019-11-07 13:27:18 +01:00
|
|
|
}
|
2019-10-22 14:33:31 +02:00
|
|
|
CMakeConfigurationKitAspect::setConfiguration(k, config);
|
2019-10-31 16:01:35 +01:00
|
|
|
if (Utils::HostOsInfo::isWindowsHost())
|
|
|
|
|
CMakeGeneratorKitAspect::setGenerator(k, "NMake Makefiles JOM");
|
2019-10-22 14:33:31 +02:00
|
|
|
}
|
|
|
|
|
|
2019-11-12 15:14:20 +01:00
|
|
|
QString McuSupportOptions::kitName(const BoardOptions *board) const
|
|
|
|
|
{
|
|
|
|
|
// TODO: get version from qulSdkPackage and insert into name
|
2019-11-14 11:26:07 +01:00
|
|
|
return QString::fromLatin1("Qt for MCUs - %1 %2").arg(board->vendor(), board->model());
|
2019-11-12 15:14:20 +01:00
|
|
|
}
|
|
|
|
|
|
2019-11-13 18:37:27 +01:00
|
|
|
QList<ProjectExplorer::Kit *> McuSupportOptions::existingKits(const BoardOptions *board)
|
2019-10-22 14:33:31 +02:00
|
|
|
{
|
|
|
|
|
using namespace ProjectExplorer;
|
2019-11-13 18:37:27 +01:00
|
|
|
const QString boardKitName = kitName(board);
|
|
|
|
|
return Utils::filtered(KitManager::kits(), [&boardKitName](Kit *kit) {
|
|
|
|
|
return kit->isAutoDetected() && kit->unexpandedDisplayName() == boardKitName;
|
2019-10-22 14:33:31 +02:00
|
|
|
});
|
2019-11-13 18:37:27 +01:00
|
|
|
}
|
2019-10-22 14:33:31 +02:00
|
|
|
|
2019-11-13 18:37:27 +01:00
|
|
|
ProjectExplorer::Kit *McuSupportOptions::newKit(const BoardOptions* board)
|
|
|
|
|
{
|
|
|
|
|
using namespace ProjectExplorer;
|
|
|
|
|
|
|
|
|
|
const QString armGccPath = toolchainPackage->path();
|
|
|
|
|
const QString qulDir = qulSdkPackage->path();
|
|
|
|
|
const auto init = [this, board](Kit *k) {
|
|
|
|
|
KitGuard kitGuard(k);
|
|
|
|
|
|
|
|
|
|
setKitProperties(kitName(board), k, board);
|
|
|
|
|
if (!isDesktop(board)) {
|
|
|
|
|
setKitToolchains(k, toolchainPackage->path());
|
|
|
|
|
setKitDebugger(k, toolchainPackage->path());
|
|
|
|
|
setKitDevice(k);
|
|
|
|
|
}
|
|
|
|
|
setKitEnvironment(k, board);
|
|
|
|
|
setKitCMakeOptions(k, board, qulSdkPackage->path());
|
|
|
|
|
|
|
|
|
|
k->setup();
|
|
|
|
|
k->fix();
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return KitManager::registerKit(init);
|
2019-10-22 14:33:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // Internal
|
|
|
|
|
} // McuSupport
|