aboutsummaryrefslogtreecommitdiffstats
path: root/src
diff options
context:
space:
mode:
authorDenis Shienkov <denis.shienkov@gmail.com>2019-05-16 21:54:41 +0300
committerDenis Shienkov <denis.shienkov@gmail.com>2019-05-17 18:07:56 +0000
commitf08904c155adc8a724100dd6c3ccad53786ed069 (patch)
tree0bc48428272385f5f0d5cb0b583f8e978dd13bb4 /src
parent39462767f9c8c97d377cea34c83a15872901de88 (diff)
bare-metal: Improve the KEIL toolchain auto detection
Now the KEIL compiler path also extracts from the Windows registry in addition to the path feature Change-Id: I8a92233ff58119ea7c3cc40e6da95408e0ced1ba Reviewed-by: Christian Kandeler <christian.kandeler@qt.io>
Diffstat (limited to 'src')
-rw-r--r--src/app/qbs-setup-toolchains/keilprobe.cpp195
-rw-r--r--src/app/qbs-setup-toolchains/keilprobe.h65
-rw-r--r--src/app/qbs-setup-toolchains/probe.cpp63
-rw-r--r--src/app/qbs-setup-toolchains/qbs-setup-toolchains.pro2
-rw-r--r--src/app/qbs-setup-toolchains/qbs-setup-toolchains.qbs2
5 files changed, 265 insertions, 62 deletions
diff --git a/src/app/qbs-setup-toolchains/keilprobe.cpp b/src/app/qbs-setup-toolchains/keilprobe.cpp
new file mode 100644
index 000000000..9de9958cd
--- /dev/null
+++ b/src/app/qbs-setup-toolchains/keilprobe.cpp
@@ -0,0 +1,195 @@
+/****************************************************************************
+**
+** Copyright (C) 2019 Denis Shienkov <denis.shienkov@gmail.com>
+** Contact: https://www.qt.io/licensing/
+**
+** This file is part of Qbs.
+**
+** $QT_BEGIN_LICENSE:LGPL$
+** 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 Lesser General Public License Usage
+** Alternatively, this file may be used under the terms of the GNU Lesser
+** General Public License version 3 as published by the Free Software
+** Foundation and appearing in the file LICENSE.LGPL3 included in the
+** packaging of this file. Please review the following information to
+** ensure the GNU Lesser General Public License version 3 requirements
+** will be met: https://www.gnu.org/licenses/lgpl-3.0.html.
+**
+** GNU General Public License Usage
+** Alternatively, this file may be used under the terms of the GNU
+** General Public License version 2.0 or (at your option) the GNU General
+** Public license version 3 or any later version approved by the KDE Free
+** Qt Foundation. The licenses are as published by the Free Software
+** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3
+** 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-2.0.html and
+** https://www.gnu.org/licenses/gpl-3.0.html.
+**
+** $QT_END_LICENSE$
+**
+****************************************************************************/
+
+#include "probe.h"
+#include "keilprobe.h"
+
+#include "../shared/logging/consolelogger.h"
+
+#include <logging/translator.h>
+
+#include <tools/hostosinfo.h>
+#include <tools/profile.h>
+
+#include <QtCore/qfileinfo.h>
+#include <QtCore/qlist.h>
+#include <QtCore/qsettings.h>
+
+using namespace qbs;
+using Internal::Tr;
+using Internal::HostOsInfo;
+
+namespace {
+
+static QStringList knownKeilCompilerNames()
+{
+ return {QStringLiteral("c51"), QStringLiteral("armcc")};
+}
+
+static QString guessKeilArchitecture(const QFileInfo &compiler)
+{
+ const auto baseName = compiler.baseName();
+ if (baseName == QLatin1String("c51"))
+ return QStringLiteral("mcs51");
+ if (baseName == QLatin1String("armcc"))
+ return QStringLiteral("arm");
+ return {};
+}
+
+static Profile createKeilProfileHelper(const QFileInfo &compiler, Settings *settings,
+ QString profileName = QString())
+{
+ const QString architecture = guessKeilArchitecture(compiler);
+
+ // In case the profile is auto-detected.
+ if (profileName.isEmpty())
+ profileName = QLatin1String("keil-") + architecture;
+
+ Profile profile(profileName, settings);
+ profile.setValue(QStringLiteral("cpp.toolchainInstallPath"), compiler.absolutePath());
+ profile.setValue(QStringLiteral("qbs.toolchainType"), QStringLiteral("keil"));
+ if (!architecture.isEmpty())
+ profile.setValue(QStringLiteral("qbs.architecture"), architecture);
+
+ qbsInfo() << Tr::tr("Profile '%1' created for '%2'.").arg(
+ profile.name(), compiler.absoluteFilePath());
+ return profile;
+}
+
+static std::vector<KeilInstallInfo> installedKeilsFromPath()
+{
+ std::vector<KeilInstallInfo> infos;
+ const auto compilerNames = knownKeilCompilerNames();
+ for (const QString &compilerName : compilerNames) {
+ const QFileInfo keilPath(
+ findExecutable(
+ HostOsInfo::appendExecutableSuffix(compilerName)));
+ if (!keilPath.exists())
+ continue;
+ infos.push_back({keilPath.absoluteFilePath(), {}});
+ }
+ return infos;
+}
+
+static std::vector<KeilInstallInfo> installedKeilsFromRegistry()
+{
+ std::vector<KeilInstallInfo> infos;
+
+ if (HostOsInfo::isWindowsHost()) {
+
+#ifdef Q_OS_WIN64
+ static const char kRegistryNode[] = "HKEY_LOCAL_MACHINE\\SOFTWARE\\WOW6432Node\\Keil\\Products";
+#else
+ static const char kRegistryNode[] = "HKEY_LOCAL_MACHINE\\SOFTWARE\\Keil\\Products";
+#endif
+
+ // Dictionary for know toolchains.
+ static const struct Entry {
+ QString productKey;
+ QString subExePath;
+ } knowToolchains[] = {
+ {QStringLiteral("MDK"), QStringLiteral("\\ARMCC\\bin\\armcc.exe")},
+ {QStringLiteral("C51"), QStringLiteral("\\BIN\\c51.exe")},
+ };
+
+ QSettings registry(QLatin1String(kRegistryNode), QSettings::NativeFormat);
+ const auto productGroups = registry.childGroups();
+ for (const QString &productKey : productGroups) {
+ const auto entryEnd = std::end(knowToolchains);
+ const auto entryIt = std::find_if(std::begin(knowToolchains), entryEnd,
+ [productKey](const Entry &entry) {
+ return entry.productKey == productKey;
+ });
+ if (entryIt == entryEnd)
+ continue;
+
+ registry.beginGroup(productKey);
+ const QString rootPath = registry.value(QStringLiteral("Path"))
+ .toString();
+ if (!rootPath.isEmpty()) {
+ // Build full compiler path.
+ const QFileInfo keilPath(rootPath + entryIt->subExePath);
+ if (keilPath.exists()) {
+ QString version = registry.value(QStringLiteral("Version"))
+ .toString();
+ if (version.startsWith(QLatin1Char('V')))
+ version.remove(0, 1);
+ infos.push_back({keilPath.absoluteFilePath(), version});
+ }
+ }
+ registry.endGroup();
+ }
+
+ }
+
+ return infos;
+}
+
+} // end of anonymous namespace
+
+bool isKeilCompiler(const QString &compilerName)
+{
+ return Internal::any_of(knownKeilCompilerNames(), [compilerName](
+ const QString &knownName) {
+ return compilerName.contains(knownName);
+ });
+}
+
+void createKeilProfile(const QFileInfo &compiler, Settings *settings,
+ QString profileName)
+{
+ createKeilProfileHelper(compiler, settings, profileName);
+}
+
+void keilProbe(Settings *settings, QList<Profile> &profiles)
+{
+ qbsInfo() << Tr::tr("Trying to detect KEIL toolchains...");
+
+ std::vector<KeilInstallInfo> allInfos = installedKeilsFromRegistry();
+ const std::vector<KeilInstallInfo> pathInfos = installedKeilsFromPath();
+ allInfos.insert(std::end(allInfos), std::begin(pathInfos), std::end(pathInfos));
+
+ for (const KeilInstallInfo &info : allInfos) {
+ const auto profile = createKeilProfileHelper(info.compilerPath, settings);
+ profiles.push_back(profile);
+ }
+
+ if (allInfos.empty())
+ qbsInfo() << Tr::tr("No KEIL toolchains found.");
+}
diff --git a/src/app/qbs-setup-toolchains/keilprobe.h b/src/app/qbs-setup-toolchains/keilprobe.h
new file mode 100644
index 000000000..ef7b76418
--- /dev/null
+++ b/src/app/qbs-setup-toolchains/keilprobe.h
@@ -0,0 +1,65 @@
+/****************************************************************************
+**
+** Copyright (C) 2019 Denis Shienkov <denis.shienkov@gmail.com>
+** Contact: https://www.qt.io/licensing/
+**
+** This file is part of Qbs.
+**
+** $QT_BEGIN_LICENSE:LGPL$
+** 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 Lesser General Public License Usage
+** Alternatively, this file may be used under the terms of the GNU Lesser
+** General Public License version 3 as published by the Free Software
+** Foundation and appearing in the file LICENSE.LGPL3 included in the
+** packaging of this file. Please review the following information to
+** ensure the GNU Lesser General Public License version 3 requirements
+** will be met: https://www.gnu.org/licenses/lgpl-3.0.html.
+**
+** GNU General Public License Usage
+** Alternatively, this file may be used under the terms of the GNU
+** General Public License version 2.0 or (at your option) the GNU General
+** Public license version 3 or any later version approved by the KDE Free
+** Qt Foundation. The licenses are as published by the Free Software
+** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3
+** 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-2.0.html and
+** https://www.gnu.org/licenses/gpl-3.0.html.
+**
+** $QT_END_LICENSE$
+**
+****************************************************************************/
+
+#ifndef KEILPROBE_H
+#define KEILPROBE_H
+
+#include <QtCore/qlist.h>
+
+class QFileInfo;
+
+namespace qbs {
+class Profile;
+class Settings;
+}
+
+struct KeilInstallInfo
+{
+ QString compilerPath;
+ QString version;
+};
+
+bool isKeilCompiler(const QString &compilerName);
+
+void createKeilProfile(const QFileInfo &compiler, qbs::Settings *settings,
+ QString profileName);
+
+void keilProbe(qbs::Settings *settings, QList<qbs::Profile> &profiles);
+
+#endif // KEILPROBE_H
diff --git a/src/app/qbs-setup-toolchains/probe.cpp b/src/app/qbs-setup-toolchains/probe.cpp
index 625eeddd9..3fae20a6e 100644
--- a/src/app/qbs-setup-toolchains/probe.cpp
+++ b/src/app/qbs-setup-toolchains/probe.cpp
@@ -40,6 +40,7 @@
#include "clangclprobe.h"
#include "iarewprobe.h"
+#include "keilprobe.h"
#include "msvcprobe.h"
#include "sdccprobe.h"
#include "xcodeprobe.h"
@@ -110,18 +111,6 @@ static QStringList validMinGWMachines()
QStringLiteral("i586-mingw32msvc"), QStringLiteral("amd64-mingw32msvc")};
}
-static QStringList knownKeilCompilerNames()
-{
- return {QStringLiteral("c51"), QStringLiteral("armcc")};
-}
-
-static bool isKeilCompiler(const QString &compilerName)
-{
- return Internal::any_of(knownKeilCompilerNames(), [compilerName](const QString &knownName) {
- return compilerName.contains(knownName);
- });
-}
-
static QStringList toolchainTypeFromCompilerName(const QString &compilerName)
{
if (compilerName == QLatin1String("cl.exe"))
@@ -259,37 +248,6 @@ static Profile createGccProfile(const QString &compilerFilePath, Settings *setti
return profile;
}
-static QString guessKeilArchitecture(const QFileInfo &compiler)
-{
- const auto baseName = compiler.baseName();
- if (baseName == QLatin1String("c51"))
- return QStringLiteral("mcs51");
- if (baseName == QLatin1String("armcc"))
- return QStringLiteral("arm");
- return {};
-}
-
-static Profile createKeilProfile(const QFileInfo &compiler, Settings *settings,
- QString profileName = QString())
-{
- const QString architecture = guessKeilArchitecture(compiler);
-
- // In case the profile is auto-detected.
- if (profileName.isEmpty())
- profileName = QLatin1String("keil-") + architecture;
-
- Profile profile(profileName, settings);
- profile.setValue(QStringLiteral("cpp.toolchainInstallPath"), compiler.absolutePath());
- profile.setValue(QStringLiteral("qbs.toolchainType"), QStringLiteral("keil"));
- if (!architecture.isEmpty())
- profile.setValue(QStringLiteral("qbs.architecture"), architecture);
-
- qStdout << Tr::tr("Profile '%1' created for '%2'.").arg(
- profile.name(), compiler.absoluteFilePath())
- << endl;
- return profile;
-}
-
static void gccProbe(Settings *settings, QList<Profile> &profiles, const QString &compilerName)
{
qStdout << Tr::tr("Trying to detect %1...").arg(compilerName) << endl;
@@ -328,25 +286,6 @@ static void mingwProbe(Settings *settings, QList<Profile> &profiles)
}
}
-static void keilProbe(Settings *settings, QList<Profile> &profiles)
-{
- qStdout << Tr::tr("Trying to detect KEIL toolchains...") << endl;
-
- bool isFound = false;
- const auto compilerNames = knownKeilCompilerNames();
- for (const QString &compilerName : compilerNames) {
- const QString keilPath = findExecutable(HostOsInfo::appendExecutableSuffix(compilerName));
- if (!keilPath.isEmpty()) {
- const auto profile = createKeilProfile(keilPath, settings);
- profiles.push_back(profile);
- isFound = true;
- }
- }
-
- if (!isFound)
- qStdout << Tr::tr("No KEIL toolchains found.") << endl;
-}
-
void probe(Settings *settings)
{
QList<Profile> profiles;
diff --git a/src/app/qbs-setup-toolchains/qbs-setup-toolchains.pro b/src/app/qbs-setup-toolchains/qbs-setup-toolchains.pro
index a341f6469..e83a9c716 100644
--- a/src/app/qbs-setup-toolchains/qbs-setup-toolchains.pro
+++ b/src/app/qbs-setup-toolchains/qbs-setup-toolchains.pro
@@ -6,6 +6,7 @@ HEADERS += \
clangclprobe.h \
commandlineparser.h \
iarewprobe.h \
+ keilprobe.h \
msvcprobe.h \
probe.h \
sdccprobe.h \
@@ -15,6 +16,7 @@ SOURCES += \
clangclprobe.cpp \
commandlineparser.cpp \
iarewprobe.cpp \
+ keilprobe.cpp \
main.cpp \
msvcprobe.cpp \
probe.cpp \
diff --git a/src/app/qbs-setup-toolchains/qbs-setup-toolchains.qbs b/src/app/qbs-setup-toolchains/qbs-setup-toolchains.qbs
index 49e4c87e9..fd3043a28 100644
--- a/src/app/qbs-setup-toolchains/qbs-setup-toolchains.qbs
+++ b/src/app/qbs-setup-toolchains/qbs-setup-toolchains.qbs
@@ -10,6 +10,8 @@ QbsApp {
"commandlineparser.h",
"iarewprobe.cpp",
"iarewprobe.h",
+ "keilprobe.cpp",
+ "keilprobe.h",
"main.cpp",
"msvcprobe.cpp",
"msvcprobe.h",