forked from qt-creator/qt-creator
Limits conflicts with Linux tool chains and debuggers. Change-Id: Ie6408aecbc2828e8be53c46cf579441ccaba19bb Reviewed-by: Orgad Shaneh <orgads@gmail.com>
280 lines
9.2 KiB
C++
280 lines
9.2 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2016 BlackBerry Limited. All rights reserved
|
|
** Contact: 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 "qnxutils.h"
|
|
#include "qnxqtversion.h"
|
|
|
|
#include <utils/algorithm.h>
|
|
#include <utils/fileutils.h>
|
|
#include <utils/hostosinfo.h>
|
|
#include <utils/synchronousprocess.h>
|
|
#include <utils/temporaryfile.h>
|
|
|
|
#include <QDebug>
|
|
#include <QDir>
|
|
#include <QDirIterator>
|
|
#include <QDomDocument>
|
|
#include <QProcess>
|
|
#include <QStandardPaths>
|
|
#include <QApplication>
|
|
|
|
using namespace ProjectExplorer;
|
|
using namespace Qnx;
|
|
using namespace Qnx::Internal;
|
|
|
|
namespace {
|
|
const char *EVAL_ENV_VARS[] = {
|
|
"QNX_TARGET", "QNX_HOST", "QNX_CONFIGURATION", "MAKEFLAGS", "LD_LIBRARY_PATH",
|
|
"PATH", "QDE", "CPUVARDIR", "PYTHONPATH"
|
|
};
|
|
}
|
|
|
|
QString QnxUtils::addQuotes(const QString &string)
|
|
{
|
|
return QLatin1Char('"') + string + QLatin1Char('"');
|
|
}
|
|
|
|
QString QnxUtils::cpuDirShortDescription(const QString &cpuDir)
|
|
{
|
|
if (cpuDir == "armle-v7")
|
|
return QLatin1String("32-bit ARM");
|
|
|
|
if (cpuDir == "aarch64le")
|
|
return QLatin1String("64-bit ARM");
|
|
|
|
if (cpuDir == "x86")
|
|
return QLatin1String("32-bit x86");
|
|
|
|
if (cpuDir == "x86_64")
|
|
return QLatin1String("64-bit x86");
|
|
|
|
return cpuDir;
|
|
}
|
|
|
|
QStringList QnxUtils::searchPaths(QnxQtVersion *qtVersion)
|
|
{
|
|
const QDir pluginDir(qtVersion->qmakeProperty("QT_INSTALL_PLUGINS"));
|
|
const QStringList pluginSubDirs = pluginDir.entryList(QDir::Dirs | QDir::NoDotAndDotDot);
|
|
|
|
QStringList searchPaths;
|
|
|
|
Q_FOREACH (const QString &dir, pluginSubDirs) {
|
|
searchPaths << qtVersion->qmakeProperty("QT_INSTALL_PLUGINS")
|
|
+ QLatin1Char('/') + dir;
|
|
}
|
|
|
|
searchPaths << qtVersion->qmakeProperty("QT_INSTALL_LIBS");
|
|
searchPaths << qtVersion->qnxTarget() + QLatin1Char('/') + qtVersion->cpuDir()
|
|
+ QLatin1String("/lib");
|
|
searchPaths << qtVersion->qnxTarget() + QLatin1Char('/') + qtVersion->cpuDir()
|
|
+ QLatin1String("/usr/lib");
|
|
|
|
return searchPaths;
|
|
}
|
|
|
|
QList<Utils::EnvironmentItem> QnxUtils::qnxEnvironmentFromEnvFile(const QString &fileName)
|
|
{
|
|
QList <Utils::EnvironmentItem> items;
|
|
|
|
if (!QFileInfo::exists(fileName))
|
|
return items;
|
|
|
|
const bool isWindows = Utils::HostOsInfo::isWindowsHost();
|
|
|
|
// locking creating sdp-env file wrapper script
|
|
Utils::TemporaryFile tmpFile(QString::fromLatin1("sdp-env-eval-XXXXXX")
|
|
+ QString::fromLatin1(isWindows ? ".bat" : ".sh"));
|
|
if (!tmpFile.open())
|
|
return items;
|
|
tmpFile.setTextModeEnabled(true);
|
|
|
|
// writing content to wrapper script
|
|
QTextStream fileContent(&tmpFile);
|
|
if (isWindows)
|
|
fileContent << QLatin1String("@echo off\n")
|
|
<< QLatin1String("call ") << fileName << QLatin1Char('\n');
|
|
else
|
|
fileContent << QLatin1String("#!/bin/bash\n")
|
|
<< QLatin1String(". ") << fileName << QLatin1Char('\n');
|
|
QString linePattern = QString::fromLatin1(isWindows ? "echo %1=%%1%" : "echo %1=$%1");
|
|
for (int i = 0, len = sizeof(EVAL_ENV_VARS) / sizeof(const char *); i < len; ++i)
|
|
fileContent << linePattern.arg(QLatin1String(EVAL_ENV_VARS[i])) << QLatin1Char('\n');
|
|
tmpFile.close();
|
|
|
|
// running wrapper script
|
|
QProcess process;
|
|
if (isWindows)
|
|
process.start(QLatin1String("cmd.exe"),
|
|
QStringList() << QLatin1String("/C") << tmpFile.fileName());
|
|
else
|
|
process.start(QLatin1String("/bin/bash"),
|
|
QStringList() << tmpFile.fileName());
|
|
|
|
// waiting for finish
|
|
QApplication::setOverrideCursor(Qt::BusyCursor);
|
|
bool waitResult = process.waitForFinished(10000) || process.state() == QProcess::NotRunning;
|
|
QApplication::restoreOverrideCursor();
|
|
if (!waitResult) {
|
|
Utils::SynchronousProcess::stopProcess(process);
|
|
return items;
|
|
}
|
|
|
|
if (process.exitStatus() != QProcess::NormalExit || process.exitCode() != 0)
|
|
return items;
|
|
|
|
// parsing process output
|
|
QTextStream str(&process);
|
|
while (!str.atEnd()) {
|
|
QString line = str.readLine();
|
|
int equalIndex = line.indexOf(QLatin1Char('='));
|
|
if (equalIndex < 0)
|
|
continue;
|
|
QString var = line.left(equalIndex);
|
|
QString value = line.mid(equalIndex + 1);
|
|
items.append(Utils::EnvironmentItem(var, value));
|
|
}
|
|
|
|
return items;
|
|
}
|
|
|
|
QString QnxUtils::envFilePath(const QString &sdpPath)
|
|
{
|
|
QDir sdp(sdpPath);
|
|
QStringList entries;
|
|
if (Utils::HostOsInfo::isWindowsHost())
|
|
entries = sdp.entryList(QStringList(QLatin1String("*-env.bat")));
|
|
else
|
|
entries = sdp.entryList(QStringList(QLatin1String("*-env.sh")));
|
|
|
|
if (!entries.isEmpty())
|
|
return sdp.absoluteFilePath(entries.first());
|
|
|
|
return QString();
|
|
}
|
|
|
|
QString QnxUtils::defaultTargetVersion(const QString &sdpPath)
|
|
{
|
|
foreach (const ConfigInstallInformation &sdpInfo, installedConfigs()) {
|
|
if (!sdpInfo.path.compare(sdpPath, Utils::HostOsInfo::fileNameCaseSensitivity()))
|
|
return sdpInfo.version;
|
|
}
|
|
|
|
return QString();
|
|
}
|
|
|
|
QList<ConfigInstallInformation> QnxUtils::installedConfigs(const QString &configPath)
|
|
{
|
|
QList<ConfigInstallInformation> sdpList;
|
|
QString sdpConfigPath = configPath;
|
|
|
|
if (!QDir(sdpConfigPath).exists())
|
|
return sdpList;
|
|
|
|
QFileInfoList sdpfileList =
|
|
QDir(sdpConfigPath).entryInfoList(QStringList() << QLatin1String("*.xml"),
|
|
QDir::Files, QDir::Time);
|
|
foreach (const QFileInfo &sdpFile, sdpfileList) {
|
|
QFile xmlFile(sdpFile.absoluteFilePath());
|
|
if (!xmlFile.open(QIODevice::ReadOnly))
|
|
continue;
|
|
|
|
QDomDocument doc;
|
|
if (!doc.setContent(&xmlFile)) // Skip error message
|
|
continue;
|
|
|
|
QDomElement docElt = doc.documentElement();
|
|
if (docElt.tagName() != QLatin1String("qnxSystemDefinition"))
|
|
continue;
|
|
|
|
QDomElement childElt = docElt.firstChildElement(QLatin1String("installation"));
|
|
// The file contains only one installation node
|
|
if (!childElt.isNull()) {
|
|
// The file contains only one base node
|
|
ConfigInstallInformation sdpInfo;
|
|
sdpInfo.path = childElt.firstChildElement(QLatin1String("base")).text();
|
|
sdpInfo.name = childElt.firstChildElement(QLatin1String("name")).text();
|
|
sdpInfo.host = childElt.firstChildElement(QLatin1String("host")).text();
|
|
sdpInfo.target = childElt.firstChildElement(QLatin1String("target")).text();
|
|
sdpInfo.version = childElt.firstChildElement(QLatin1String("version")).text();
|
|
sdpInfo.installationXmlFilePath = sdpFile.absoluteFilePath();
|
|
|
|
sdpList.append(sdpInfo);
|
|
}
|
|
}
|
|
|
|
return sdpList;
|
|
}
|
|
|
|
QList<Utils::EnvironmentItem> QnxUtils::qnxEnvironment(const QString &sdpPath)
|
|
{
|
|
return qnxEnvironmentFromEnvFile(envFilePath(sdpPath));
|
|
}
|
|
|
|
QList<QnxTarget> QnxUtils::findTargets(const Utils::FileName &basePath)
|
|
{
|
|
using namespace Utils;
|
|
QList<QnxTarget> result;
|
|
|
|
QDirIterator iterator(basePath.toString());
|
|
while (iterator.hasNext()) {
|
|
iterator.next();
|
|
FileName libc = FileName::fromString(iterator.filePath()).appendPath("lib/libc.so");
|
|
if (libc.exists()) {
|
|
auto abis = Abi::abisOfBinary(libc);
|
|
if (abis.isEmpty()) {
|
|
qWarning() << libc << "has no ABIs ... discarded";
|
|
continue;
|
|
}
|
|
|
|
if (abis.count() > 1)
|
|
qWarning() << libc << "has more than one ABI ... processing all";
|
|
|
|
FileName path = FileName::fromString(iterator.filePath());
|
|
for (Abi abi : abis)
|
|
result.append(QnxTarget(path, QnxUtils::convertAbi(abi)));
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
Abi QnxUtils::convertAbi(const Abi &abi)
|
|
{
|
|
if (abi.os() == Abi::LinuxOS && abi.osFlavor() == Abi::GenericLinuxFlavor) {
|
|
return Abi(abi.architecture(),
|
|
Abi::QnxOS,
|
|
Abi::GenericQnxFlavor,
|
|
abi.binaryFormat(),
|
|
abi.wordWidth());
|
|
} else {
|
|
return abi;
|
|
}
|
|
}
|
|
|
|
QList<Abi> QnxUtils::convertAbis(const QList<Abi> &abis)
|
|
{
|
|
return Utils::transform(abis, &QnxUtils::convertAbi);
|
|
}
|