forked from qt-creator/qt-creator
Change-Id: I43b49d325a129c180869140c4ae1774f7f89d2fd Reviewed-by: Eike Ziller <eike.ziller@qt.io>
328 lines
11 KiB
C++
328 lines
11 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2016 BlackBerry Limited. All rights reserved.
|
|
** Contact: BlackBerry (qt@blackberry.com), KDAB (info@kdab.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 "qnxtoolchain.h"
|
|
|
|
#include "qnxconfiguration.h"
|
|
#include "qnxconfigurationmanager.h"
|
|
#include "qnxconstants.h"
|
|
#include "qnxtr.h"
|
|
#include "qnxutils.h"
|
|
|
|
#include <projectexplorer/abiwidget.h>
|
|
#include <projectexplorer/projectexplorerconstants.h>
|
|
#include <utils/algorithm.h>
|
|
#include <utils/pathchooser.h>
|
|
|
|
#include <QFormLayout>
|
|
|
|
using namespace ProjectExplorer;
|
|
using namespace Utils;
|
|
|
|
namespace Qnx {
|
|
namespace Internal {
|
|
|
|
const char CompilerSdpPath[] = "Qnx.QnxToolChain.NDKPath";
|
|
const char CpuDirKey[] = "Qnx.QnxToolChain.CpuDir";
|
|
|
|
static Abis detectTargetAbis(const FilePath &sdpPath)
|
|
{
|
|
Abis result;
|
|
FilePath qnxTarget;
|
|
|
|
if (!sdpPath.fileName().isEmpty()) {
|
|
const EnvironmentItems environment = QnxUtils::qnxEnvironment(sdpPath);
|
|
for (const EnvironmentItem &item : environment) {
|
|
if (item.name == QLatin1String("QNX_TARGET"))
|
|
qnxTarget = FilePath::fromString(item.value);
|
|
}
|
|
}
|
|
|
|
if (qnxTarget.isEmpty())
|
|
return result;
|
|
|
|
QList<QnxTarget> targets = QnxUtils::findTargets(qnxTarget);
|
|
for (const auto &target : targets) {
|
|
if (!result.contains(target.m_abi))
|
|
result.append(target.m_abi);
|
|
}
|
|
|
|
Utils::sort(result,
|
|
[](const Abi &arg1, const Abi &arg2) { return arg1.toString() < arg2.toString(); });
|
|
|
|
return result;
|
|
}
|
|
|
|
static void setQnxEnvironment(Environment &env, const EnvironmentItems &qnxEnv)
|
|
{
|
|
// We only need to set QNX_HOST, QNX_TARGET, and QNX_CONFIGURATION_EXCLUSIVE
|
|
// needed when running qcc
|
|
foreach (const EnvironmentItem &item, qnxEnv) {
|
|
if (item.name == QLatin1String("QNX_HOST") ||
|
|
item.name == QLatin1String("QNX_TARGET") ||
|
|
item.name == QLatin1String("QNX_CONFIGURATION_EXCLUSIVE"))
|
|
env.set(item.name, item.value);
|
|
}
|
|
}
|
|
|
|
// Qcc is a multi-compiler driver, and most of the gcc options can be accomplished by using the -Wp, and -Wc
|
|
// options to pass the options directly down to the compiler
|
|
static QStringList reinterpretOptions(const QStringList &args)
|
|
{
|
|
QStringList arguments;
|
|
foreach (const QString &str, args) {
|
|
if (str.startsWith(QLatin1String("--sysroot=")))
|
|
continue;
|
|
QString arg = str;
|
|
if (arg == QLatin1String("-v")
|
|
|| arg == QLatin1String("-dM"))
|
|
arg.prepend(QLatin1String("-Wp,"));
|
|
arguments << arg;
|
|
}
|
|
|
|
return arguments;
|
|
}
|
|
|
|
QnxToolChain::QnxToolChain()
|
|
: GccToolChain(Constants::QNX_TOOLCHAIN_ID)
|
|
{
|
|
setOptionsReinterpreter(&reinterpretOptions);
|
|
setTypeDisplayName(Tr::tr("QCC"));
|
|
}
|
|
|
|
std::unique_ptr<ToolChainConfigWidget> QnxToolChain::createConfigurationWidget()
|
|
{
|
|
return std::make_unique<QnxToolChainConfigWidget>(this);
|
|
}
|
|
|
|
void QnxToolChain::addToEnvironment(Environment &env) const
|
|
{
|
|
if (env.expandedValueForKey("QNX_HOST").isEmpty() ||
|
|
env.expandedValueForKey("QNX_TARGET").isEmpty() ||
|
|
env.expandedValueForKey("QNX_CONFIGURATION_EXCLUSIVE").isEmpty())
|
|
setQnxEnvironment(env, QnxUtils::qnxEnvironment(m_sdpPath));
|
|
|
|
GccToolChain::addToEnvironment(env);
|
|
}
|
|
|
|
QStringList QnxToolChain::suggestedMkspecList() const
|
|
{
|
|
return {
|
|
"qnx-armle-v7-qcc",
|
|
"qnx-x86-qcc",
|
|
"qnx-aarch64le-qcc",
|
|
"qnx-x86-64-qcc"
|
|
};
|
|
}
|
|
|
|
QVariantMap QnxToolChain::toMap() const
|
|
{
|
|
QVariantMap data = GccToolChain::toMap();
|
|
data.insert(QLatin1String(CompilerSdpPath), m_sdpPath.toVariant());
|
|
data.insert(QLatin1String(CpuDirKey), m_cpuDir);
|
|
return data;
|
|
}
|
|
|
|
bool QnxToolChain::fromMap(const QVariantMap &data)
|
|
{
|
|
if (!GccToolChain::fromMap(data))
|
|
return false;
|
|
|
|
m_sdpPath = FilePath::fromVariant(data.value(CompilerSdpPath));
|
|
m_cpuDir = data.value(QLatin1String(CpuDirKey)).toString();
|
|
|
|
// Make the ABIs QNX specific (if they aren't already).
|
|
setSupportedAbis(QnxUtils::convertAbis(supportedAbis()));
|
|
setTargetAbi(QnxUtils::convertAbi(targetAbi()));
|
|
|
|
return true;
|
|
}
|
|
|
|
FilePath QnxToolChain::sdpPath() const
|
|
{
|
|
return m_sdpPath;
|
|
}
|
|
|
|
void QnxToolChain::setSdpPath(const FilePath &sdpPath)
|
|
{
|
|
if (m_sdpPath == sdpPath)
|
|
return;
|
|
m_sdpPath = sdpPath;
|
|
toolChainUpdated();
|
|
}
|
|
|
|
QString QnxToolChain::cpuDir() const
|
|
{
|
|
return m_cpuDir;
|
|
}
|
|
|
|
void QnxToolChain::setCpuDir(const QString &cpuDir)
|
|
{
|
|
if (m_cpuDir == cpuDir)
|
|
return;
|
|
m_cpuDir = cpuDir;
|
|
toolChainUpdated();
|
|
}
|
|
|
|
GccToolChain::DetectedAbisResult QnxToolChain::detectSupportedAbis() const
|
|
{
|
|
// "unknown-qnx-gnu"is needed to get the "--target=xxx" parameter sent code model,
|
|
// which gets translated as "x86_64-qnx-gnu", which gets Clang to happily parse
|
|
// the QNX code.
|
|
//
|
|
// Without it on Windows Clang defaults to a MSVC mode, which breaks with
|
|
// the QNX code, which is mostly GNU based.
|
|
return GccToolChain::DetectedAbisResult{detectTargetAbis(m_sdpPath), "unknown-qnx-gnu"};
|
|
}
|
|
|
|
bool QnxToolChain::operator ==(const ToolChain &other) const
|
|
{
|
|
if (!GccToolChain::operator ==(other))
|
|
return false;
|
|
|
|
auto qnxTc = static_cast<const QnxToolChain *>(&other);
|
|
|
|
return m_sdpPath == qnxTc->m_sdpPath && m_cpuDir == qnxTc->m_cpuDir;
|
|
}
|
|
|
|
// --------------------------------------------------------------------------
|
|
// QnxToolChainFactory
|
|
// --------------------------------------------------------------------------
|
|
|
|
QnxToolChainFactory::QnxToolChainFactory()
|
|
{
|
|
setDisplayName(Tr::tr("QCC"));
|
|
setSupportedToolChainType(Constants::QNX_TOOLCHAIN_ID);
|
|
setSupportedLanguages({ProjectExplorer::Constants::C_LANGUAGE_ID,
|
|
ProjectExplorer::Constants::CXX_LANGUAGE_ID});
|
|
setToolchainConstructor([] { return new QnxToolChain; });
|
|
setUserCreatable(true);
|
|
}
|
|
|
|
Toolchains QnxToolChainFactory::autoDetect(const ToolchainDetector &detector) const
|
|
{
|
|
// FIXME: Support detecting toolchains on remote devices
|
|
if (detector.device)
|
|
return {};
|
|
|
|
Toolchains tcs;
|
|
const auto configurations = QnxConfigurationManager::instance()->configurations();
|
|
for (QnxConfiguration *configuration : configurations)
|
|
tcs += configuration->autoDetect(detector.alreadyKnown);
|
|
return tcs;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------
|
|
// QnxToolChainConfigWidget
|
|
//---------------------------------------------------------------------------------
|
|
|
|
QnxToolChainConfigWidget::QnxToolChainConfigWidget(QnxToolChain *tc)
|
|
: ToolChainConfigWidget(tc)
|
|
, m_compilerCommand(new PathChooser)
|
|
, m_sdpPath(new PathChooser)
|
|
, m_abiWidget(new AbiWidget)
|
|
{
|
|
m_compilerCommand->setExpectedKind(PathChooser::ExistingCommand);
|
|
m_compilerCommand->setHistoryCompleter(QLatin1String("Qnx.ToolChain.History"));
|
|
m_compilerCommand->setFilePath(tc->compilerCommand());
|
|
m_compilerCommand->setEnabled(!tc->isAutoDetected());
|
|
|
|
m_sdpPath->setExpectedKind(PathChooser::ExistingDirectory);
|
|
m_sdpPath->setHistoryCompleter(QLatin1String("Qnx.Sdp.History"));
|
|
m_sdpPath->setFilePath(tc->sdpPath());
|
|
m_sdpPath->setEnabled(!tc->isAutoDetected());
|
|
|
|
const Abis abiList = detectTargetAbis(m_sdpPath->filePath());
|
|
m_abiWidget->setAbis(abiList, tc->targetAbi());
|
|
m_abiWidget->setEnabled(!tc->isAutoDetected() && !abiList.isEmpty());
|
|
|
|
m_mainLayout->addRow(Tr::tr("&Compiler path:"), m_compilerCommand);
|
|
//: SDP refers to 'Software Development Platform'.
|
|
m_mainLayout->addRow(Tr::tr("SDP path:"), m_sdpPath);
|
|
m_mainLayout->addRow(Tr::tr("&ABI:"), m_abiWidget);
|
|
|
|
connect(m_compilerCommand, &PathChooser::rawPathChanged, this, &ToolChainConfigWidget::dirty);
|
|
connect(m_sdpPath, &PathChooser::rawPathChanged,
|
|
this, &QnxToolChainConfigWidget::handleSdpPathChange);
|
|
connect(m_abiWidget, &AbiWidget::abiChanged, this, &ToolChainConfigWidget::dirty);
|
|
}
|
|
|
|
void QnxToolChainConfigWidget::applyImpl()
|
|
{
|
|
if (toolChain()->isAutoDetected())
|
|
return;
|
|
|
|
auto tc = static_cast<QnxToolChain *>(toolChain());
|
|
Q_ASSERT(tc);
|
|
QString displayName = tc->displayName();
|
|
tc->setDisplayName(displayName); // reset display name
|
|
tc->setSdpPath(m_sdpPath->filePath());
|
|
tc->setTargetAbi(m_abiWidget->currentAbi());
|
|
tc->resetToolChain(m_compilerCommand->filePath());
|
|
}
|
|
|
|
void QnxToolChainConfigWidget::discardImpl()
|
|
{
|
|
// subwidgets are not yet connected!
|
|
QSignalBlocker blocker(this);
|
|
auto tc = static_cast<const QnxToolChain *>(toolChain());
|
|
m_compilerCommand->setFilePath(tc->compilerCommand());
|
|
m_sdpPath->setFilePath(tc->sdpPath());
|
|
m_abiWidget->setAbis(tc->supportedAbis(), tc->targetAbi());
|
|
if (!m_compilerCommand->filePath().toString().isEmpty())
|
|
m_abiWidget->setEnabled(true);
|
|
}
|
|
|
|
bool QnxToolChainConfigWidget::isDirtyImpl() const
|
|
{
|
|
auto tc = static_cast<const QnxToolChain *>(toolChain());
|
|
Q_ASSERT(tc);
|
|
return m_compilerCommand->filePath() != tc->compilerCommand()
|
|
|| m_sdpPath->filePath() != tc->sdpPath()
|
|
|| m_abiWidget->currentAbi() != tc->targetAbi();
|
|
}
|
|
|
|
void QnxToolChainConfigWidget::handleSdpPathChange()
|
|
{
|
|
const Abi currentAbi = m_abiWidget->currentAbi();
|
|
const bool customAbi = m_abiWidget->isCustomAbi();
|
|
const Abis abiList = detectTargetAbis(m_sdpPath->filePath());
|
|
|
|
m_abiWidget->setEnabled(!abiList.isEmpty());
|
|
|
|
// Find a good ABI for the new compiler:
|
|
Abi newAbi;
|
|
if (customAbi)
|
|
newAbi = currentAbi;
|
|
else if (abiList.contains(currentAbi))
|
|
newAbi = currentAbi;
|
|
|
|
m_abiWidget->setAbis(abiList, newAbi);
|
|
emit dirty();
|
|
}
|
|
|
|
} // namespace Internal
|
|
} // namespace Qnx
|