forked from qt-creator/qt-creator
		
	Don't enforce the presence of a "platforms" subdirectory if the NDK is of version 22 or higher. The last NDK version with that directory is 21. Change-Id: Ib431e7db4521533206304d252dcf93b7ea6169e5 Reviewed-by: Assam Boudjelthia <assam.boudjelthia@qt.io>
		
			
				
	
	
		
			1570 lines
		
	
	
		
			56 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1570 lines
		
	
	
		
			56 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /****************************************************************************
 | |
| **
 | |
| ** Copyright (C) 2016 BogDan Vatra <bog_dan_ro@yahoo.com>
 | |
| ** Contact: https://www.qt.io/licensing/
 | |
| **
 | |
| ** 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 "androidconfigurations.h"
 | |
| #include "androidconstants.h"
 | |
| #include "androidtoolchain.h"
 | |
| #include "androiddevice.h"
 | |
| #include "androidmanager.h"
 | |
| #include "androidqtversion.h"
 | |
| #include "avddialog.h"
 | |
| 
 | |
| #include <coreplugin/icore.h>
 | |
| #include <coreplugin/messagemanager.h>
 | |
| 
 | |
| #include <projectexplorer/devicesupport/devicemanager.h>
 | |
| #include <projectexplorer/kitinformation.h>
 | |
| #include <projectexplorer/kitmanager.h>
 | |
| #include <projectexplorer/project.h>
 | |
| #include <projectexplorer/projectexplorerconstants.h>
 | |
| #include <projectexplorer/session.h>
 | |
| #include <projectexplorer/toolchainmanager.h>
 | |
| 
 | |
| #include <debugger/debuggeritemmanager.h>
 | |
| #include <debugger/debuggeritem.h>
 | |
| #include <debugger/debuggerkitinformation.h>
 | |
| 
 | |
| #include <qtsupport/baseqtversion.h>
 | |
| #include <qtsupport/qtkitinformation.h>
 | |
| #include <qtsupport/qtversionmanager.h>
 | |
| 
 | |
| #include <utils/algorithm.h>
 | |
| #include <utils/environment.h>
 | |
| #include <utils/environment.h>
 | |
| #include <utils/hostosinfo.h>
 | |
| #include <utils/persistentsettings.h>
 | |
| #include <utils/qtcassert.h>
 | |
| #include <utils/qtcprocess.h>
 | |
| #include <utils/runextensions.h>
 | |
| #include <utils/stringutils.h>
 | |
| 
 | |
| #include <QApplication>
 | |
| #include <QDirIterator>
 | |
| #include <QFileInfo>
 | |
| #include <QHostAddress>
 | |
| #include <QJsonArray>
 | |
| #include <QJsonDocument>
 | |
| #include <QJsonObject>
 | |
| #include <QLoggingCategory>
 | |
| #include <QProcess>
 | |
| #include <QRegularExpression>
 | |
| #include <QSettings>
 | |
| #include <QStandardPaths>
 | |
| #include <QStringList>
 | |
| #include <QTcpSocket>
 | |
| 
 | |
| #include <functional>
 | |
| #include <memory>
 | |
| 
 | |
| using namespace QtSupport;
 | |
| using namespace ProjectExplorer;
 | |
| using namespace Utils;
 | |
| 
 | |
| namespace {
 | |
| static Q_LOGGING_CATEGORY(avdConfigLog, "qtc.android.androidconfig", QtWarningMsg)
 | |
| }
 | |
| 
 | |
| namespace Android {
 | |
| using namespace Internal;
 | |
| 
 | |
| const char JsonFilePath[] = "android/sdk_definitions.json";
 | |
| const char SdkToolsUrlKey[] = "sdk_tools_url";
 | |
| const char CommonKey[] = "common";
 | |
| const char SdkEssentialPkgsKey[] = "sdk_essential_packages";
 | |
| const char VersionsKey[] = "versions";
 | |
| const char NdkPathKey[] = "ndk_path";
 | |
| const char SpecificQtVersionsKey[] = "specific_qt_versions";
 | |
| const char DefaultVersionKey[] = "default";
 | |
| const char LinuxOsKey[] = "linux";
 | |
| const char WindowsOsKey[] = "windows";
 | |
| const char macOsKey[] = "mac";
 | |
| 
 | |
| 
 | |
| namespace {
 | |
|     const char jdk8SettingsPath[] = "HKEY_LOCAL_MACHINE\\SOFTWARE\\JavaSoft\\Java Development Kit";
 | |
|     const char jdkLatestSettingsPath[] = "HKEY_LOCAL_MACHINE\\SOFTWARE\\JavaSoft\\JDK\\";
 | |
| 
 | |
|     const QLatin1String SettingsGroup("AndroidConfigurations");
 | |
|     const QLatin1String SDKLocationKey("SDKLocation");
 | |
|     const QLatin1String CustomNdkLocationsKey("CustomNdkLocations");
 | |
|     const QLatin1String SdkFullyConfiguredKey("AllEssentialsInstalled");
 | |
|     const QLatin1String SDKManagerToolArgsKey("SDKManagerToolArgs");
 | |
|     const QLatin1String OpenJDKLocationKey("OpenJDKLocation");
 | |
|     const QLatin1String OpenSslPriLocationKey("OpenSSLPriLocation");
 | |
|     const QLatin1String AutomaticKitCreationKey("AutomatiKitCreation");
 | |
|     const QLatin1String EmulatorArgsKey("EmulatorArgs");
 | |
| 
 | |
|     const QLatin1String ArmToolchainPrefix("arm-linux-androideabi");
 | |
|     const QLatin1String X86ToolchainPrefix("x86");
 | |
|     const QLatin1String AArch64ToolchainPrefix("aarch64-linux-android");
 | |
|     const QLatin1String X86_64ToolchainPrefix("x86_64");
 | |
| 
 | |
|     const QLatin1String ArmToolsPrefix("arm-linux-androideabi");
 | |
|     const QLatin1String X86ToolsPrefix("i686-linux-android");
 | |
|     const QLatin1String AArch64ToolsPrefix("aarch64-linux-android");
 | |
|     const QLatin1String X86_64ToolsPrefix("x86_64-linux-android");
 | |
| 
 | |
|     const QLatin1String ArmToolsDisplayName("arm");
 | |
|     const QLatin1String X86ToolsDisplayName("i686");
 | |
|     const QLatin1String AArch64ToolsDisplayName("aarch64");
 | |
|     const QLatin1String X86_64ToolsDisplayName("x86_64");
 | |
| 
 | |
|     const QLatin1String Unknown("unknown");
 | |
|     const QLatin1String keytoolName("keytool");
 | |
|     const QLatin1String changeTimeStamp("ChangeTimeStamp");
 | |
| 
 | |
|     const QLatin1String sdkToolsVersionKey("Pkg.Revision");
 | |
|     const QLatin1String ndkRevisionKey("Pkg.Revision");
 | |
| 
 | |
|     static QString sdkSettingsFileName()
 | |
|     {
 | |
|         return Core::ICore::installerResourcePath("android.xml").toString();
 | |
|     }
 | |
| 
 | |
|     static bool is32BitUserSpace()
 | |
|     {
 | |
|         // Do the exact same check as android's emulator is doing:
 | |
|         if (HostOsInfo::isLinuxHost()) {
 | |
|             if (QSysInfo::WordSize == 32 ) {
 | |
|                 Environment env = Environment::systemEnvironment();
 | |
|                 FilePath executable = env.searchInPath("file");
 | |
|                 QString shell = env.value(QLatin1String("SHELL"));
 | |
|                 if (executable.isEmpty() || shell.isEmpty())
 | |
|                     return true; // we can't detect, but creator is 32bit so assume 32bit
 | |
| 
 | |
|                 QtcProcess proc;
 | |
|                 proc.setProcessChannelMode(QProcess::MergedChannels);
 | |
|                 proc.setTimeoutS(30);
 | |
|                 proc.setCommand({executable, {shell}});
 | |
|                 proc.runBlocking();
 | |
|                 if (proc.result() != QtcProcess::FinishedWithSuccess)
 | |
|                     return true;
 | |
|                 return !proc.allOutput().contains("x86-64");
 | |
|             }
 | |
|         }
 | |
|         return false;
 | |
|     }
 | |
| }
 | |
| 
 | |
| //////////////////////////////////
 | |
| // AndroidConfig
 | |
| //////////////////////////////////
 | |
| 
 | |
| QLatin1String AndroidConfig::toolchainPrefix(const Abi &abi)
 | |
| {
 | |
|     switch (abi.architecture()) {
 | |
|     case Abi::ArmArchitecture:
 | |
|         if (abi.wordWidth() == 64)
 | |
|             return AArch64ToolchainPrefix;
 | |
|         return ArmToolchainPrefix;
 | |
|     case Abi::X86Architecture:
 | |
|         if (abi.wordWidth() == 64)
 | |
|             return X86_64ToolchainPrefix;
 | |
|         return X86ToolchainPrefix;
 | |
|     default:
 | |
|         return Unknown;
 | |
|     }
 | |
| }
 | |
| 
 | |
| QLatin1String AndroidConfig::toolsPrefix(const Abi &abi)
 | |
| {
 | |
|     switch (abi.architecture()) {
 | |
|     case Abi::ArmArchitecture:
 | |
|         if (abi.wordWidth() == 64)
 | |
|             return AArch64ToolsPrefix;
 | |
|         return ArmToolsPrefix;
 | |
|     case Abi::X86Architecture:
 | |
|         if (abi.wordWidth() == 64)
 | |
|             return X86_64ToolsPrefix;
 | |
|         return X86ToolsPrefix;
 | |
|     default:
 | |
|         return Unknown;
 | |
|     }
 | |
| }
 | |
| 
 | |
| QLatin1String AndroidConfig::displayName(const Abi &abi)
 | |
| {
 | |
|     switch (abi.architecture()) {
 | |
|     case Abi::ArmArchitecture:
 | |
|         if (abi.wordWidth() == 64)
 | |
|             return AArch64ToolsDisplayName;
 | |
|         return ArmToolsDisplayName;
 | |
|     case Abi::X86Architecture:
 | |
|         if (abi.wordWidth() == 64)
 | |
|             return X86_64ToolsDisplayName;
 | |
|         return X86ToolsDisplayName;
 | |
|     default:
 | |
|         return Unknown;
 | |
|     }
 | |
| }
 | |
| 
 | |
| void AndroidConfig::load(const QSettings &settings)
 | |
| {
 | |
|     // user settings
 | |
|     m_emulatorArgs = settings.value(EmulatorArgsKey,
 | |
|                          QStringList({"-netdelay", "none", "-netspeed", "full"})).toStringList();
 | |
|     m_sdkLocation = FilePath::fromUserInput(settings.value(SDKLocationKey).toString()).cleanPath();
 | |
|     m_customNdkList = settings.value(CustomNdkLocationsKey).toStringList();
 | |
|     m_sdkManagerToolArgs = settings.value(SDKManagerToolArgsKey).toStringList();
 | |
|     m_openJDKLocation = FilePath::fromString(settings.value(OpenJDKLocationKey).toString());
 | |
|     m_openSslLocation = FilePath::fromString(settings.value(OpenSslPriLocationKey).toString());
 | |
|     m_automaticKitCreation = settings.value(AutomaticKitCreationKey, true).toBool();
 | |
|     m_sdkFullyConfigured = settings.value(SdkFullyConfiguredKey, false).toBool();
 | |
| 
 | |
|     PersistentSettingsReader reader;
 | |
|     if (reader.load(FilePath::fromString(sdkSettingsFileName()))
 | |
|             && settings.value(changeTimeStamp).toInt() != QFileInfo(sdkSettingsFileName()).lastModified().toMSecsSinceEpoch() / 1000) {
 | |
|         // persisten settings
 | |
|         m_sdkLocation = FilePath::fromUserInput(reader.restoreValue(SDKLocationKey, m_sdkLocation.toString()).toString()).cleanPath();
 | |
|         m_customNdkList = reader.restoreValue(CustomNdkLocationsKey).toStringList();
 | |
|         m_sdkManagerToolArgs = reader.restoreValue(SDKManagerToolArgsKey, m_sdkManagerToolArgs).toStringList();
 | |
|         m_openJDKLocation = FilePath::fromString(reader.restoreValue(OpenJDKLocationKey, m_openJDKLocation.toString()).toString());
 | |
|         m_openSslLocation = FilePath::fromString(reader.restoreValue(OpenSslPriLocationKey, m_openSslLocation.toString()).toString());
 | |
|         m_automaticKitCreation = reader.restoreValue(AutomaticKitCreationKey, m_automaticKitCreation).toBool();
 | |
|         m_sdkFullyConfigured = reader.restoreValue(SdkFullyConfiguredKey, m_sdkFullyConfigured).toBool();
 | |
|         // persistent settings
 | |
|     }
 | |
|     m_customNdkList.removeAll("");
 | |
|     parseDependenciesJson();
 | |
| }
 | |
| 
 | |
| void AndroidConfig::save(QSettings &settings) const
 | |
| {
 | |
|     QFileInfo fileInfo(sdkSettingsFileName());
 | |
|     if (fileInfo.exists())
 | |
|         settings.setValue(changeTimeStamp, fileInfo.lastModified().toMSecsSinceEpoch() / 1000);
 | |
| 
 | |
|     // user settings
 | |
|     settings.setValue(SDKLocationKey, m_sdkLocation.toString());
 | |
|     settings.setValue(CustomNdkLocationsKey, m_customNdkList);
 | |
|     settings.setValue(SDKManagerToolArgsKey, m_sdkManagerToolArgs);
 | |
|     settings.setValue(OpenJDKLocationKey, m_openJDKLocation.toString());
 | |
|     settings.setValue(OpenSslPriLocationKey, m_openSslLocation.toString());
 | |
|     settings.setValue(EmulatorArgsKey, m_emulatorArgs);
 | |
|     settings.setValue(AutomaticKitCreationKey, m_automaticKitCreation);
 | |
|     settings.setValue(SdkFullyConfiguredKey, m_sdkFullyConfigured);
 | |
| }
 | |
| 
 | |
| void AndroidConfig::parseDependenciesJson()
 | |
| {
 | |
|     const FilePath sdkConfigUserFile = Core::ICore::userResourcePath(JsonFilePath);
 | |
|     const FilePath sdkConfigFile = Core::ICore::resourcePath(JsonFilePath);
 | |
| 
 | |
|     if (!sdkConfigUserFile.exists()) {
 | |
|         sdkConfigUserFile.absolutePath().ensureWritableDir();
 | |
|         sdkConfigFile.copyFile(sdkConfigUserFile);
 | |
|     }
 | |
| 
 | |
|     if (sdkConfigFile.lastModified() > sdkConfigUserFile.lastModified()) {
 | |
|         const FilePath oldUserFile = sdkConfigUserFile + ".old";
 | |
|         oldUserFile.removeFile();
 | |
|         sdkConfigUserFile.renameFile(oldUserFile);
 | |
|         sdkConfigFile.copyFile(sdkConfigUserFile);
 | |
|     }
 | |
| 
 | |
|     QFile jsonFile(sdkConfigUserFile.toString());
 | |
|     if (!jsonFile.open(QIODevice::ReadOnly)) {
 | |
|         qCDebug(avdConfigLog, "Couldn't open JSON config file %s.", qPrintable(jsonFile.fileName()));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     QJsonObject jsonObject = QJsonDocument::fromJson(jsonFile.readAll()).object();
 | |
| 
 | |
|     if (jsonObject.contains(CommonKey) && jsonObject[CommonKey].isObject()) {
 | |
|         QJsonObject commonObject = jsonObject[CommonKey].toObject();
 | |
|         // Parse SDK Tools URL
 | |
|         if (commonObject.contains(SdkToolsUrlKey) && commonObject[SdkToolsUrlKey].isObject()) {
 | |
|             QJsonObject sdkToolsObj(commonObject[SdkToolsUrlKey].toObject());
 | |
|             if (HostOsInfo::isMacHost()) {
 | |
|                 m_sdkToolsUrl = sdkToolsObj[macOsKey].toString();
 | |
|                 m_sdkToolsSha256 = QByteArray::fromHex(sdkToolsObj["mac_sha256"].toString().toUtf8());
 | |
|             } else if (HostOsInfo::isWindowsHost()) {
 | |
|                 m_sdkToolsUrl = sdkToolsObj[WindowsOsKey].toString();
 | |
|                 m_sdkToolsSha256 = QByteArray::fromHex(sdkToolsObj["windows_sha256"].toString().toUtf8());
 | |
|             } else {
 | |
|                 m_sdkToolsUrl = sdkToolsObj[LinuxOsKey].toString();
 | |
|                 m_sdkToolsSha256 = QByteArray::fromHex(sdkToolsObj["linux_sha256"].toString().toUtf8());
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // Parse common essential packages
 | |
|         auto appendEssentialsFromArray = [this](QJsonArray array) {
 | |
|             for (const QJsonValueRef &pkg : array)
 | |
|                 m_commonEssentialPkgs.append(pkg.toString());
 | |
|         };
 | |
| 
 | |
|         QJsonObject commonEssentials = commonObject[SdkEssentialPkgsKey].toObject();
 | |
|         appendEssentialsFromArray(commonEssentials[DefaultVersionKey].toArray());
 | |
| 
 | |
|         if (HostOsInfo::isWindowsHost())
 | |
|             appendEssentialsFromArray(commonEssentials[WindowsOsKey].toArray());
 | |
|         if (HostOsInfo::isMacHost())
 | |
|             appendEssentialsFromArray(commonEssentials[macOsKey].toArray());
 | |
|         else
 | |
|             appendEssentialsFromArray(commonEssentials[LinuxOsKey].toArray());
 | |
|     }
 | |
| 
 | |
|     auto fillQtVersionsRange = [](const QString &shortVersion) {
 | |
|         QList<QtVersionNumber> versions;
 | |
|         QRegularExpression re("([0-9]\\.[0-9]*\\.)\\[([0-9])\\-([0-9])\\]");
 | |
|         QRegularExpressionMatch match = re.match(shortVersion);
 | |
|         if (match.hasMatch() && match.lastCapturedIndex() == 3)
 | |
|             for (int i = match.captured(2).toInt(); i <= match.captured(3).toInt(); ++i)
 | |
|                 versions.append(QtVersionNumber(match.captured(1) + QString::number(i)));
 | |
|         else
 | |
|             versions.append(QtVersionNumber(shortVersion));
 | |
| 
 | |
|         return versions;
 | |
|     };
 | |
| 
 | |
|     if (jsonObject.contains(SpecificQtVersionsKey) && jsonObject[SpecificQtVersionsKey].isArray()) {
 | |
|         const QJsonArray versionsArray = jsonObject[SpecificQtVersionsKey].toArray();
 | |
|         for (const QJsonValue &item : versionsArray) {
 | |
|             QJsonObject itemObj = item.toObject();
 | |
|             SdkForQtVersions specificVersion;
 | |
|             specificVersion.ndkPath = itemObj[NdkPathKey].toString();
 | |
|             const auto pkgs = itemObj[SdkEssentialPkgsKey].toArray();
 | |
|             for (const QJsonValue &pkg : pkgs)
 | |
|                 specificVersion.essentialPackages.append(pkg.toString());
 | |
|             const auto versions = itemObj[VersionsKey].toArray();
 | |
|             for (const QJsonValue &pkg : versions)
 | |
|                 specificVersion.versions.append(fillQtVersionsRange(pkg.toString()));
 | |
| 
 | |
|             if (itemObj[VersionsKey].toArray().first().toString() == DefaultVersionKey)
 | |
|                 m_defaultSdkDepends = specificVersion;
 | |
|             else
 | |
|                 m_specificQtVersions.append(specificVersion);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| QVector<int> AndroidConfig::availableNdkPlatforms(const BaseQtVersion *qtVersion) const
 | |
| {
 | |
|     QVector<int> availableNdkPlatforms;
 | |
| 
 | |
|     ndkLocation(qtVersion)
 | |
|         .pathAppended("platforms")
 | |
|         .iterateDirectory(
 | |
|             [&availableNdkPlatforms](const FilePath &filePath) {
 | |
|                 availableNdkPlatforms.push_back(
 | |
|                     filePath.toString()
 | |
|                         .mid(filePath.path().lastIndexOf('-') + 1)
 | |
|                         .toInt());
 | |
|                 return true;
 | |
|             },
 | |
|             {"android-*"},
 | |
|             QDir::Dirs);
 | |
| 
 | |
|     Utils::sort(availableNdkPlatforms, std::greater<>());
 | |
|     return availableNdkPlatforms;
 | |
| }
 | |
| 
 | |
| QStringList AndroidConfig::getCustomNdkList() const
 | |
| {
 | |
|     return m_customNdkList;
 | |
| }
 | |
| 
 | |
| void AndroidConfig::addCustomNdk(const QString &customNdk)
 | |
| {
 | |
|     if (!m_customNdkList.contains(customNdk))
 | |
|         m_customNdkList.append(customNdk);
 | |
| }
 | |
| 
 | |
| void AndroidConfig::removeCustomNdk(const QString &customNdk)
 | |
| {
 | |
|     m_customNdkList.removeAll(customNdk);
 | |
| }
 | |
| 
 | |
| FilePath AndroidConfig::openSslLocation() const
 | |
| {
 | |
|     return m_openSslLocation;
 | |
| }
 | |
| 
 | |
| void AndroidConfig::setOpenSslLocation(const FilePath &openSslLocation)
 | |
| {
 | |
|     m_openSslLocation = openSslLocation;
 | |
| }
 | |
| 
 | |
| QStringList AndroidConfig::apiLevelNamesFor(const SdkPlatformList &platforms)
 | |
| {
 | |
|     return Utils::transform(platforms, AndroidConfig::apiLevelNameFor);
 | |
| }
 | |
| 
 | |
| QString AndroidConfig::apiLevelNameFor(const SdkPlatform *platform)
 | |
| {
 | |
|     return platform && platform->apiLevel() > 0 ?
 | |
|                 QString("android-%1").arg(platform->apiLevel()) : "";
 | |
| }
 | |
| 
 | |
| int AndroidConfig::platformNameToApiLevel(const QString &platformName)
 | |
| {
 | |
|     int apiLevel = -1;
 | |
|     static const QRegularExpression re("(android-)(?<apiLevel>[0-9A-Z]{1,})",
 | |
|                                        QRegularExpression::CaseInsensitiveOption);
 | |
|     QRegularExpressionMatch match = re.match(platformName);
 | |
|     if (match.hasMatch()) {
 | |
|         QString apiLevelStr = match.captured("apiLevel");
 | |
|         bool isUInt;
 | |
|         apiLevel = apiLevelStr.toUInt(&isUInt);
 | |
|         if (!isUInt) {
 | |
|             if (apiLevelStr == 'Q')
 | |
|                 apiLevel = 29;
 | |
|             else if (apiLevelStr == 'R')
 | |
|                 apiLevel = 30;
 | |
|             else if (apiLevelStr == 'S')
 | |
|                 apiLevel = 31;
 | |
|         }
 | |
|     }
 | |
|     return apiLevel;
 | |
| }
 | |
| 
 | |
| bool AndroidConfig::isCmdlineSdkToolsInstalled() const
 | |
| {
 | |
|     QString toolPath("cmdline-tools/latest/bin/sdkmanager");
 | |
|     if (HostOsInfo::isWindowsHost())
 | |
|         toolPath += ANDROID_BAT_SUFFIX;
 | |
| 
 | |
|     return m_sdkLocation.pathAppended(toolPath).exists();
 | |
| }
 | |
| 
 | |
| FilePath AndroidConfig::adbToolPath() const
 | |
| {
 | |
|     return m_sdkLocation / "platform-tools/adb" QTC_HOST_EXE_SUFFIX;
 | |
| }
 | |
| 
 | |
| FilePath AndroidConfig::emulatorToolPath() const
 | |
| {
 | |
|     QString relativePath = "emulator/emulator";
 | |
|     if (sdkToolsVersion() < QVersionNumber(25, 3, 0) && !isCmdlineSdkToolsInstalled())
 | |
|         relativePath = "tools/emulator";
 | |
|     return m_sdkLocation / (relativePath + QTC_HOST_EXE_SUFFIX);
 | |
| }
 | |
| 
 | |
| FilePath AndroidConfig::sdkManagerToolPath() const
 | |
| {
 | |
|     QStringList sdkmanagerPaths = {"cmdline-tools/latest/bin/sdkmanager",
 | |
|                                    "tools/bin/sdkmanager"};
 | |
| 
 | |
|     for (QString &toolPath : sdkmanagerPaths) {
 | |
|         if (HostOsInfo::isWindowsHost())
 | |
|             toolPath += ANDROID_BAT_SUFFIX;
 | |
| 
 | |
|         const FilePath sdkmanagerPath = m_sdkLocation / toolPath;
 | |
|         if (sdkmanagerPath.exists())
 | |
|             return sdkmanagerPath;
 | |
|     }
 | |
| 
 | |
|     return FilePath();
 | |
| }
 | |
| 
 | |
| FilePath AndroidConfig::avdManagerToolPath() const
 | |
| {
 | |
|     QStringList sdkmanagerPaths = {"cmdline-tools/latest/bin/avdmanager",
 | |
|                                    "tools/bin/avdmanager"};
 | |
| 
 | |
|     for (QString &toolPath : sdkmanagerPaths) {
 | |
|         if (HostOsInfo::isWindowsHost())
 | |
|             toolPath += ANDROID_BAT_SUFFIX;
 | |
| 
 | |
|         const FilePath sdkmanagerPath = m_sdkLocation / toolPath;
 | |
|         if (sdkmanagerPath.exists())
 | |
|             return sdkmanagerPath;
 | |
|     }
 | |
| 
 | |
|     return FilePath();
 | |
| }
 | |
| 
 | |
| FilePath AndroidConfig::toolchainPathFromNdk(const FilePath &ndkLocation) const
 | |
| {
 | |
|     const FilePath tcPath = ndkLocation / "toolchains/";
 | |
|     FilePath toolchainPath;
 | |
|     QDirIterator llvmIter(tcPath.toString(), {"llvm*"}, QDir::Dirs);
 | |
|     if (llvmIter.hasNext()) {
 | |
|         llvmIter.next();
 | |
|         toolchainPath = tcPath / llvmIter.fileName() / "prebuilt/";
 | |
|     }
 | |
| 
 | |
|     // detect toolchain host
 | |
|     QStringList hostPatterns;
 | |
|     switch (HostOsInfo::hostOs()) {
 | |
|     case OsTypeLinux:
 | |
|         hostPatterns << QLatin1String("linux*");
 | |
|         break;
 | |
|     case OsTypeWindows:
 | |
|         hostPatterns << QLatin1String("windows*");
 | |
|         break;
 | |
|     case OsTypeMac:
 | |
|         hostPatterns << QLatin1String("darwin*");
 | |
|         break;
 | |
|     default: /* unknown host */ return FilePath();
 | |
|     }
 | |
| 
 | |
|     QDirIterator iter(toolchainPath.toString(), hostPatterns, QDir::Dirs);
 | |
|     if (iter.hasNext()) {
 | |
|         iter.next();
 | |
|         return toolchainPath / iter.fileName();
 | |
|     }
 | |
| 
 | |
|     return {};
 | |
| }
 | |
| 
 | |
| FilePath AndroidConfig::toolchainPath(const BaseQtVersion *qtVersion) const
 | |
| {
 | |
|     return toolchainPathFromNdk(ndkLocation(qtVersion));
 | |
| }
 | |
| 
 | |
| FilePath AndroidConfig::clangPathFromNdk(const FilePath &ndkLocation) const
 | |
| {
 | |
|     const FilePath path = toolchainPathFromNdk(ndkLocation);
 | |
|     if (path.isEmpty())
 | |
|         return {};
 | |
|     return path.pathAppended("bin/clang").withExecutableSuffix();
 | |
| }
 | |
| 
 | |
| FilePath AndroidConfig::gdbPath(const Abi &abi, const BaseQtVersion *qtVersion) const
 | |
| {
 | |
|     return gdbPathFromNdk(abi, ndkLocation(qtVersion));
 | |
| }
 | |
| 
 | |
| FilePath AndroidConfig::gdbPathFromNdk(const Abi &abi, const FilePath &ndkLocation) const
 | |
| {
 | |
|     const FilePath path = ndkLocation.pathAppended(
 | |
|         QString("prebuilt/%1/bin/gdb%2").arg(toolchainHostFromNdk(ndkLocation),
 | |
|                                              QString(QTC_HOST_EXE_SUFFIX)));
 | |
|     if (path.exists())
 | |
|         return path;
 | |
|     // fallback for old NDKs (e.g. 10e)
 | |
|     return ndkLocation.pathAppended(QString("toolchains/%1-4.9/prebuilt/%2/bin/%3-gdb%4")
 | |
|                                                    .arg(toolchainPrefix(abi),
 | |
|                                                         toolchainHostFromNdk(ndkLocation),
 | |
|                                                         toolsPrefix(abi),
 | |
|                                                         QString(QTC_HOST_EXE_SUFFIX)));
 | |
| }
 | |
| 
 | |
| FilePath AndroidConfig::makePathFromNdk(const FilePath &ndkLocation) const
 | |
| {
 | |
|     return ndkLocation.pathAppended(
 | |
|                 QString("prebuilt/%1/bin/make%2").arg(toolchainHostFromNdk(ndkLocation),
 | |
|                                                       QString(QTC_HOST_EXE_SUFFIX)));
 | |
| }
 | |
| 
 | |
| FilePath AndroidConfig::openJDKBinPath() const
 | |
| {
 | |
|     const FilePath path = m_openJDKLocation;
 | |
|     if (!path.isEmpty())
 | |
|         return path.pathAppended("bin");
 | |
|     return path;
 | |
| }
 | |
| 
 | |
| FilePath AndroidConfig::keytoolPath() const
 | |
| {
 | |
|     return openJDKBinPath().pathAppended(keytoolName);
 | |
| }
 | |
| 
 | |
| QVector<AndroidDeviceInfo> AndroidConfig::connectedDevices(QString *error) const
 | |
| {
 | |
|     QVector<AndroidDeviceInfo> devices;
 | |
|     QtcProcess adbProc;
 | |
|     adbProc.setTimeoutS(30);
 | |
|     CommandLine cmd{adbToolPath(), {"devices"}};
 | |
|     adbProc.setCommand(cmd);
 | |
|     adbProc.runBlocking();
 | |
|     if (adbProc.result() != QtcProcess::FinishedWithSuccess) {
 | |
|         if (error)
 | |
|             *error = QApplication::translate("AndroidConfiguration", "Could not run: %1")
 | |
|                 .arg(cmd.toUserOutput());
 | |
|         return devices;
 | |
|     }
 | |
|     QStringList adbDevs = adbProc.allOutput().split('\n', Qt::SkipEmptyParts);
 | |
|     if (adbDevs.empty())
 | |
|         return devices;
 | |
| 
 | |
|     for (const QString &line : adbDevs) // remove the daemon logs
 | |
|         if (line.startsWith("* daemon"))
 | |
|             adbDevs.removeOne(line);
 | |
|     adbDevs.removeFirst(); // remove "List of devices attached" header line
 | |
| 
 | |
|     // workaround for '????????????' serial numbers:
 | |
|     // can use "adb -d" when only one usb device attached
 | |
|     foreach (const QString &device, adbDevs) {
 | |
|         const QString serialNo = device.left(device.indexOf('\t')).trimmed();
 | |
|         const QString deviceType = device.mid(device.indexOf('\t')).trimmed();
 | |
|         AndroidDeviceInfo dev;
 | |
|         dev.serialNumber = serialNo;
 | |
|         dev.type = serialNo.startsWith(QLatin1String("emulator")) ? IDevice::Emulator
 | |
|                                                                   : IDevice::Hardware;
 | |
|         dev.sdk = getSDKVersion(dev.serialNumber);
 | |
|         dev.cpuAbi = getAbis(dev.serialNumber);
 | |
|         if (deviceType == QLatin1String("unauthorized"))
 | |
|             dev.state = IDevice::DeviceConnected;
 | |
|         else if (deviceType == QLatin1String("offline"))
 | |
|             dev.state = IDevice::DeviceDisconnected;
 | |
|         else
 | |
|             dev.state = IDevice::DeviceReadyToUse;
 | |
| 
 | |
|         if (dev.type == IDevice::Emulator) {
 | |
|             dev.avdname = getAvdName(dev.serialNumber);
 | |
|             if (dev.avdname.isEmpty())
 | |
|                 dev.avdname = serialNo;
 | |
|         }
 | |
| 
 | |
|         devices.push_back(dev);
 | |
|     }
 | |
| 
 | |
|     Utils::sort(devices);
 | |
|     if (devices.isEmpty() && error)
 | |
|         *error = QApplication::translate("AndroidConfiguration",
 | |
|                                          "No devices found in output of: %1")
 | |
|             .arg(cmd.toUserOutput());
 | |
|     return devices;
 | |
| }
 | |
| 
 | |
| bool AndroidConfig::isConnected(const QString &serialNumber) const
 | |
| {
 | |
|     QVector<AndroidDeviceInfo> devices = connectedDevices();
 | |
|     foreach (AndroidDeviceInfo device, devices) {
 | |
|         if (device.serialNumber == serialNumber)
 | |
|             return true;
 | |
|     }
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| QString AndroidConfig::getDeviceProperty(const QString &device, const QString &property)
 | |
| {
 | |
|     // workaround for '????????????' serial numbers
 | |
|     CommandLine cmd(AndroidConfigurations::currentConfig().adbToolPath(),
 | |
|                     AndroidDeviceInfo::adbSelector(device));
 | |
|     cmd.addArgs({"shell", "getprop", property});
 | |
| 
 | |
|     QtcProcess adbProc;
 | |
|     adbProc.setTimeoutS(10);
 | |
|     adbProc.setCommand(cmd);
 | |
|     adbProc.runBlocking();
 | |
|     if (adbProc.result() != QtcProcess::FinishedWithSuccess)
 | |
|         return QString();
 | |
| 
 | |
|     return adbProc.allOutput();
 | |
| }
 | |
| 
 | |
| int AndroidConfig::getSDKVersion(const QString &device)
 | |
| {
 | |
|     QString tmp = getDeviceProperty(device, "ro.build.version.sdk");
 | |
|     if (tmp.isEmpty())
 | |
|         return -1;
 | |
|     return tmp.trimmed().toInt();
 | |
| }
 | |
| 
 | |
| QString AndroidConfig::getAvdName(const QString &serialnumber)
 | |
| {
 | |
|     int index = serialnumber.indexOf(QLatin1String("-"));
 | |
|     if (index == -1)
 | |
|         return QString();
 | |
|     bool ok;
 | |
|     int port = serialnumber.mid(index + 1).toInt(&ok);
 | |
|     if (!ok)
 | |
|         return QString();
 | |
| 
 | |
|     const QByteArray avdName = "avd name\n";
 | |
| 
 | |
|     QTcpSocket tcpSocket;
 | |
|     tcpSocket.connectToHost(QHostAddress(QHostAddress::LocalHost), port);
 | |
|     if (!tcpSocket.waitForConnected(100)) // Don't wait more than 100ms for a local connection
 | |
|         return QString{};
 | |
| 
 | |
|     tcpSocket.write(avdName + "exit\n");
 | |
|     tcpSocket.waitForDisconnected(500);
 | |
| 
 | |
|     QByteArray name;
 | |
|     const QByteArrayList response = tcpSocket.readAll().split('\n');
 | |
|     // The input "avd name" might not be echoed as-is, but contain ASCII
 | |
|     // control sequences.
 | |
|     for (int i = response.size() - 1; i > 1; --i) {
 | |
|         if (response.at(i).startsWith("OK")) {
 | |
|             name = response.at(i - 1);
 | |
|             break;
 | |
|         }
 | |
|     }
 | |
|     return QString::fromLatin1(name).trimmed();
 | |
| }
 | |
| 
 | |
| static SdkToolResult emulatorNameAdbCommand(const QString &serialNumber)
 | |
| {
 | |
|     QStringList args = AndroidDeviceInfo::adbSelector(serialNumber);
 | |
|     args.append({"emu", "avd", "name"});
 | |
|     return AndroidManager::runAdbCommand(args);
 | |
| }
 | |
| 
 | |
| QString AndroidConfig::getRunningAvdsSerialNumber(const QString &name) const
 | |
| {
 | |
|     for (const AndroidDeviceInfo &dev : connectedDevices()) {
 | |
|         if (!dev.serialNumber.startsWith("emulator"))
 | |
|             continue;
 | |
|         SdkToolResult result = emulatorNameAdbCommand(dev.serialNumber);
 | |
|         const QString stdOut = result.stdOut();
 | |
|         if (stdOut.isEmpty())
 | |
|             continue; // Not an avd
 | |
|         const QStringList outputLines = stdOut.split('\n');
 | |
|         if (outputLines.size() > 1 && outputLines.first() == name)
 | |
|             return dev.serialNumber;
 | |
|     }
 | |
| 
 | |
|     return {};
 | |
| }
 | |
| 
 | |
| QStringList AndroidConfig::getRunningAvdsFromDevices(const QVector<AndroidDeviceInfo> &devs)
 | |
| {
 | |
|     QStringList runningDevs;
 | |
|     for (const AndroidDeviceInfo &dev : devs) {
 | |
|         if (!dev.serialNumber.startsWith("emulator"))
 | |
|             continue;
 | |
|         SdkToolResult result = emulatorNameAdbCommand(dev.serialNumber);
 | |
|         const QString stdOut = result.stdOut();
 | |
|         if (stdOut.isEmpty())
 | |
|             continue; // Not an avd
 | |
|         const QStringList outputLines = stdOut.split('\n');
 | |
|         if (outputLines.size() > 1)
 | |
|             runningDevs.append(outputLines.first());
 | |
|     }
 | |
| 
 | |
|     return runningDevs;
 | |
| }
 | |
| 
 | |
| AndroidConfig::OpenGl AndroidConfig::getOpenGLEnabled(const QString &emulator) const
 | |
| {
 | |
|     QDir dir = QDir::home();
 | |
|     if (!dir.cd(QLatin1String(".android")))
 | |
|         return OpenGl::Unknown;
 | |
|     if (!dir.cd(QLatin1String("avd")))
 | |
|         return OpenGl::Unknown;
 | |
|     if (!dir.cd(emulator + QLatin1String(".avd")))
 | |
|         return OpenGl::Unknown;
 | |
|     QFile file(dir.filePath(QLatin1String("config.ini")));
 | |
|     if (!file.exists())
 | |
|         return OpenGl::Unknown;
 | |
|     if (!file.open(QIODevice::ReadOnly | QIODevice::Text))
 | |
|         return OpenGl::Unknown;
 | |
|     while (!file.atEnd()) {
 | |
|         QByteArray line = file.readLine();
 | |
|         if (line.contains("hw.gpu.enabled") && line.contains("yes"))
 | |
|             return OpenGl::Enabled;
 | |
|     }
 | |
|     return OpenGl::Disabled;
 | |
| }
 | |
| 
 | |
| //!
 | |
| //! \brief AndroidConfigurations::getProductModel
 | |
| //! \param device serial number
 | |
| //! \return the produce model of the device or if that cannot be read the serial number
 | |
| //!
 | |
| QString AndroidConfig::getProductModel(const QString &device) const
 | |
| {
 | |
|     if (m_serialNumberToDeviceName.contains(device))
 | |
|         return m_serialNumberToDeviceName.value(device);
 | |
| 
 | |
|     QString model = getDeviceProperty(device, "ro.product.model").trimmed();
 | |
|     if (model.isEmpty())
 | |
|         return device;
 | |
| 
 | |
|     if (!device.startsWith(QLatin1String("????")))
 | |
|         m_serialNumberToDeviceName.insert(device, model);
 | |
|     return model;
 | |
| }
 | |
| 
 | |
| QStringList AndroidConfig::getAbis(const QString &device)
 | |
| {
 | |
|     const FilePath adbTool = AndroidConfigurations::currentConfig().adbToolPath();
 | |
|     QStringList result;
 | |
|     // First try via ro.product.cpu.abilist
 | |
|     QStringList arguments = AndroidDeviceInfo::adbSelector(device);
 | |
|     arguments << "shell" << "getprop" << "ro.product.cpu.abilist";
 | |
|     QtcProcess adbProc;
 | |
|     adbProc.setTimeoutS(10);
 | |
|     adbProc.setCommand({adbTool, arguments});
 | |
|     adbProc.runBlocking();
 | |
|     if (adbProc.result() != QtcProcess::FinishedWithSuccess)
 | |
|         return result;
 | |
| 
 | |
|     QString output = adbProc.allOutput().trimmed();
 | |
|     if (!output.isEmpty()) {
 | |
|         QStringList result = output.split(QLatin1Char(','));
 | |
|         if (!result.isEmpty())
 | |
|             return result;
 | |
|     }
 | |
| 
 | |
|     // Fall back to ro.product.cpu.abi, ro.product.cpu.abi2 ...
 | |
|     for (int i = 1; i < 6; ++i) {
 | |
|         QStringList arguments = AndroidDeviceInfo::adbSelector(device);
 | |
|         arguments << QLatin1String("shell") << QLatin1String("getprop");
 | |
|         if (i == 1)
 | |
|             arguments << QLatin1String("ro.product.cpu.abi");
 | |
|         else
 | |
|             arguments << QString::fromLatin1("ro.product.cpu.abi%1").arg(i);
 | |
| 
 | |
|         QtcProcess abiProc;
 | |
|         abiProc.setTimeoutS(10);
 | |
|         abiProc.setCommand({adbTool, arguments});
 | |
|         abiProc.runBlocking();
 | |
|         if (abiProc.result() != QtcProcess::FinishedWithSuccess)
 | |
|             return result;
 | |
| 
 | |
|         QString abi = abiProc.allOutput().trimmed();
 | |
|         if (abi.isEmpty())
 | |
|             break;
 | |
|         result << abi;
 | |
|     }
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| bool AndroidConfig::isValidNdk(const QString &ndkLocation) const
 | |
| {
 | |
|     auto ndkPath = Utils::FilePath::fromUserInput(ndkLocation);
 | |
| 
 | |
|     if (!ndkPath.exists())
 | |
|         return false;
 | |
| 
 | |
|     if (!ndkPath.pathAppended("toolchains").exists())
 | |
|         return false;
 | |
| 
 | |
|     const QVersionNumber version = ndkVersion(ndkPath);
 | |
|     if (ndkVersion(ndkPath).isNull())
 | |
|         return false;
 | |
| 
 | |
|     const FilePath ndkPlatformsDir = ndkPath.pathAppended("platforms");
 | |
|     if (version.majorVersion() <= 22
 | |
|             && (!ndkPlatformsDir.exists() || ndkPlatformsDir.toString().contains(' ')))
 | |
|         return false; // TODO: Adapt code that assumes the presence of a "platforms" folder
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| QString AndroidConfig::bestNdkPlatformMatch(int target, const BaseQtVersion *qtVersion) const
 | |
| {
 | |
|     target = std::max(AndroidManager::defaultMinimumSDK(qtVersion), target);
 | |
|     foreach (int apiLevel, availableNdkPlatforms(qtVersion)) {
 | |
|         if (apiLevel <= target)
 | |
|             return QString::fromLatin1("android-%1").arg(apiLevel);
 | |
|     }
 | |
|     return QString("android-%1").arg(AndroidManager::defaultMinimumSDK(qtVersion));
 | |
| }
 | |
| 
 | |
| FilePath AndroidConfig::sdkLocation() const
 | |
| {
 | |
|     return m_sdkLocation;
 | |
| }
 | |
| 
 | |
| void AndroidConfig::setSdkLocation(const FilePath &sdkLocation)
 | |
| {
 | |
|     m_sdkLocation = sdkLocation;
 | |
| }
 | |
| 
 | |
| QVersionNumber AndroidConfig::sdkToolsVersion() const
 | |
| {
 | |
|     QVersionNumber version;
 | |
|     if (m_sdkLocation.exists()) {
 | |
|         FilePath sdkToolsPropertiesPath;
 | |
|         if (isCmdlineSdkToolsInstalled())
 | |
|             sdkToolsPropertiesPath = m_sdkLocation / "cmdline-tools/latest/source.properties";
 | |
|         else
 | |
|             sdkToolsPropertiesPath = m_sdkLocation / "tools/source.properties";
 | |
|         QSettings settings(sdkToolsPropertiesPath.toString(), QSettings::IniFormat);
 | |
|         auto versionStr = settings.value(sdkToolsVersionKey).toString();
 | |
|         version = QVersionNumber::fromString(versionStr);
 | |
|     }
 | |
|     return version;
 | |
| }
 | |
| 
 | |
| QVersionNumber AndroidConfig::buildToolsVersion() const
 | |
| {
 | |
|     //TODO: return version according to qt version
 | |
|     QVersionNumber maxVersion;
 | |
|     QDir buildToolsDir(m_sdkLocation.pathAppended("build-tools").toString());
 | |
|     const auto files = buildToolsDir.entryInfoList(QDir::Dirs|QDir::NoDotAndDotDot);
 | |
|     for (const QFileInfo &file: files)
 | |
|         maxVersion = qMax(maxVersion, QVersionNumber::fromString(file.fileName()));
 | |
|     return maxVersion;
 | |
| }
 | |
| 
 | |
| QStringList AndroidConfig::sdkManagerToolArgs() const
 | |
| {
 | |
|     return m_sdkManagerToolArgs;
 | |
| }
 | |
| 
 | |
| void AndroidConfig::setSdkManagerToolArgs(const QStringList &args)
 | |
| {
 | |
|     m_sdkManagerToolArgs = args;
 | |
| }
 | |
| 
 | |
| FilePath AndroidConfig::ndkLocation(const BaseQtVersion *qtVersion) const
 | |
| {
 | |
|     return sdkLocation().pathAppended(ndkPathFromQtVersion(*qtVersion));
 | |
| }
 | |
| 
 | |
| QVersionNumber AndroidConfig::ndkVersion(const BaseQtVersion *qtVersion) const
 | |
| {
 | |
|     return ndkVersion(ndkLocation(qtVersion));
 | |
| }
 | |
| 
 | |
| QVersionNumber AndroidConfig::ndkVersion(const FilePath &ndkPath) const
 | |
| {
 | |
|     QVersionNumber version;
 | |
|     if (!ndkPath.exists()) {
 | |
|         qCDebug(avdConfigLog) << "Cannot find ndk version. Check NDK path."
 | |
|                               << ndkPath.toString();
 | |
|         return version;
 | |
|     }
 | |
| 
 | |
|     const FilePath ndkPropertiesPath = ndkPath.pathAppended("source.properties");
 | |
|     if (ndkPropertiesPath.exists()) {
 | |
|         // source.properties files exists in NDK version > 11
 | |
|         QSettings settings(ndkPropertiesPath.toString(), QSettings::IniFormat);
 | |
|         auto versionStr = settings.value(ndkRevisionKey).toString();
 | |
|         version = QVersionNumber::fromString(versionStr);
 | |
|     } else {
 | |
|         // No source.properties. There should be a file named RELEASE.TXT
 | |
|         const FilePath ndkReleaseTxtPath = ndkPath.pathAppended("RELEASE.TXT");
 | |
|         FileReader reader;
 | |
|         QString errorString;
 | |
|         if (reader.fetch(ndkReleaseTxtPath, &errorString)) {
 | |
|             // RELEASE.TXT contains the ndk version in either of the following formats:
 | |
|             // r6a
 | |
|             // r10e (64 bit)
 | |
|             QString content = QString::fromUtf8(reader.data());
 | |
|             QRegularExpression re("(r)(?<major>[0-9]{1,2})(?<minor>[a-z]{1,1})");
 | |
|             QRegularExpressionMatch match = re.match(content);
 | |
|             if (match.hasMatch()) {
 | |
|                 QString major = match.captured("major");
 | |
|                 QString minor = match.captured("minor");
 | |
|                 // Minor version: a = 0, b = 1, c = 2 and so on.
 | |
|                 // Int equivalent = minorVersionChar - 'a'. i.e. minorVersionChar - 97.
 | |
|                 version = QVersionNumber::fromString(QString("%1.%2.0").arg(major)
 | |
|                                                      .arg((int)minor[0].toLatin1() - 97));
 | |
|             } else {
 | |
|                 qCDebug(avdConfigLog) << "Cannot find ndk version. Cannot parse RELEASE.TXT."
 | |
|                                       << content;
 | |
|             }
 | |
|         } else {
 | |
|             qCDebug(avdConfigLog) << "Cannot find ndk version." << errorString;
 | |
|         }
 | |
|     }
 | |
|     return version;
 | |
| }
 | |
| 
 | |
| QStringList AndroidConfig::allEssentials() const
 | |
| {
 | |
|     QList<BaseQtVersion *> installedVersions = QtVersionManager::versions(
 | |
|         [](const BaseQtVersion *v) {
 | |
|             return v->targetDeviceTypes().contains(Android::Constants::ANDROID_DEVICE_TYPE);
 | |
|         });
 | |
| 
 | |
|     QStringList allPackages(defaultEssentials());
 | |
|     for (const BaseQtVersion *version : installedVersions)
 | |
|         allPackages.append(essentialsFromQtVersion(*version));
 | |
|     allPackages.removeDuplicates();
 | |
| 
 | |
|     return allPackages;
 | |
| }
 | |
| 
 | |
| bool AndroidConfig::allEssentialsInstalled(AndroidSdkManager *sdkManager)
 | |
| {
 | |
|     QStringList essentialPkgs(allEssentials());
 | |
|     const auto installedPkgs = sdkManager->installedSdkPackages();
 | |
|     for (const AndroidSdkPackage *pkg : installedPkgs) {
 | |
|         if (essentialPkgs.contains(pkg->sdkStylePath()))
 | |
|             essentialPkgs.removeOne(pkg->sdkStylePath());
 | |
|         if (essentialPkgs.isEmpty())
 | |
|             break;
 | |
|     }
 | |
|     return essentialPkgs.isEmpty() ? true : false;
 | |
| }
 | |
| 
 | |
| bool AndroidConfig::sdkToolsOk() const
 | |
| {
 | |
|     bool exists = sdkLocation().exists();
 | |
|     bool writable = sdkLocation().isWritableDir();
 | |
|     bool sdkToolsExist = !sdkToolsVersion().isNull();
 | |
|     return exists && writable && sdkToolsExist;
 | |
| }
 | |
| 
 | |
| QStringList AndroidConfig::essentialsFromQtVersion(const BaseQtVersion &version) const
 | |
| {
 | |
|     QtVersionNumber qtVersion = version.qtVersion();
 | |
|     for (const SdkForQtVersions &item : m_specificQtVersions)
 | |
|         if (item.containsVersion(qtVersion))
 | |
|             return item.essentialPackages;
 | |
| 
 | |
|     return m_defaultSdkDepends.essentialPackages;
 | |
| }
 | |
| 
 | |
| QString AndroidConfig::ndkPathFromQtVersion(const BaseQtVersion &version) const
 | |
| {
 | |
|     QtVersionNumber qtVersion(version.qtVersionString());
 | |
|     for (const SdkForQtVersions &item : m_specificQtVersions)
 | |
|         if (item.containsVersion(qtVersion))
 | |
|             return item.ndkPath;
 | |
| 
 | |
|     return m_defaultSdkDepends.ndkPath;
 | |
| }
 | |
| 
 | |
| QStringList AndroidConfig::defaultEssentials() const
 | |
| {
 | |
|     return m_defaultSdkDepends.essentialPackages + m_commonEssentialPkgs;
 | |
| }
 | |
| 
 | |
| bool SdkForQtVersions::containsVersion(const QtVersionNumber &qtVersion) const
 | |
| {
 | |
|     return versions.contains(qtVersion)
 | |
|            || versions.contains(QtVersionNumber(qtVersion.majorVersion, qtVersion.minorVersion));
 | |
| }
 | |
| 
 | |
| FilePath AndroidConfig::openJDKLocation() const
 | |
| {
 | |
|     return m_openJDKLocation;
 | |
| }
 | |
| 
 | |
| void AndroidConfig::setOpenJDKLocation(const FilePath &openJDKLocation)
 | |
| {
 | |
|     m_openJDKLocation = openJDKLocation;
 | |
| }
 | |
| 
 | |
| QString AndroidConfig::toolchainHost(const BaseQtVersion *qtVersion) const
 | |
| {
 | |
|     return toolchainHostFromNdk(ndkLocation(qtVersion));
 | |
| }
 | |
| 
 | |
| QString AndroidConfig::toolchainHostFromNdk(const FilePath &ndkPath) const
 | |
| {
 | |
|     // detect toolchain host
 | |
|     QString toolchainHost;
 | |
|     QStringList hostPatterns;
 | |
|     switch (HostOsInfo::hostOs()) {
 | |
|     case OsTypeLinux:
 | |
|         hostPatterns << QLatin1String("linux*");
 | |
|         break;
 | |
|     case OsTypeWindows:
 | |
|         hostPatterns << QLatin1String("windows*");
 | |
|         break;
 | |
|     case OsTypeMac:
 | |
|         hostPatterns << QLatin1String("darwin*");
 | |
|         break;
 | |
|     default: /* unknown host */
 | |
|         return toolchainHost;
 | |
|     }
 | |
| 
 | |
|     QDirIterator jt(ndkPath.pathAppended("prebuilt").toString(),
 | |
|                     hostPatterns,
 | |
|                     QDir::Dirs);
 | |
|     if (jt.hasNext()) {
 | |
|         jt.next();
 | |
|         toolchainHost = jt.fileName();
 | |
|     }
 | |
| 
 | |
|     return toolchainHost;
 | |
| }
 | |
| 
 | |
| QStringList AndroidConfig::emulatorArgs() const
 | |
| {
 | |
|     return m_emulatorArgs;
 | |
| }
 | |
| 
 | |
| void AndroidConfig::setEmulatorArgs(const QStringList &args)
 | |
| {
 | |
|     m_emulatorArgs = args;
 | |
| }
 | |
| 
 | |
| bool AndroidConfig::automaticKitCreation() const
 | |
| {
 | |
|     return m_automaticKitCreation;
 | |
| }
 | |
| 
 | |
| void AndroidConfig::setAutomaticKitCreation(bool b)
 | |
| {
 | |
|     m_automaticKitCreation = b;
 | |
| }
 | |
| 
 | |
| FilePath AndroidConfig::defaultSdkPath()
 | |
| {
 | |
|     QString sdkFromEnvVar = QString::fromLocal8Bit(getenv("ANDROID_SDK_ROOT"));
 | |
|     if (!sdkFromEnvVar.isEmpty())
 | |
|         return FilePath::fromUserInput(sdkFromEnvVar).cleanPath();
 | |
| 
 | |
|     // Set default path of SDK as used by Android Studio
 | |
|     if (HostOsInfo::isMacHost()) {
 | |
|         return FilePath::fromString(
 | |
|             QStandardPaths::writableLocation(QStandardPaths::HomeLocation) + "/Library/Android/sdk");
 | |
|     }
 | |
| 
 | |
|     if (HostOsInfo::isWindowsHost()) {
 | |
|         return FilePath::fromString(
 | |
|             QStandardPaths::writableLocation(QStandardPaths::GenericConfigLocation) + "/Android/Sdk");
 | |
|     }
 | |
| 
 | |
|     return FilePath::fromString(
 | |
|         QStandardPaths::writableLocation(QStandardPaths::HomeLocation) + "/Android/Sdk");
 | |
| }
 | |
| 
 | |
| ///////////////////////////////////
 | |
| // AndroidConfigurations
 | |
| ///////////////////////////////////
 | |
| void AndroidConfigurations::setConfig(const AndroidConfig &devConfigs)
 | |
| {
 | |
|     emit m_instance->aboutToUpdate();
 | |
|     m_instance->m_config = devConfigs;
 | |
| 
 | |
|     m_instance->save();
 | |
|     updateAndroidDevice();
 | |
|     registerNewToolChains();
 | |
|     updateAutomaticKitList();
 | |
|     removeOldToolChains();
 | |
|     emit m_instance->updated();
 | |
| }
 | |
| 
 | |
| static bool matchToolChain(const ToolChain *atc, const ToolChain *btc)
 | |
| {
 | |
|     if (atc == btc)
 | |
|         return true;
 | |
| 
 | |
|     if (!atc || !btc)
 | |
|         return false;
 | |
| 
 | |
|     if (atc->typeId() != Constants::ANDROID_TOOLCHAIN_TYPEID || btc->typeId() != Constants::ANDROID_TOOLCHAIN_TYPEID)
 | |
|         return false;
 | |
| 
 | |
|     auto aatc = static_cast<const AndroidToolChain *>(atc);
 | |
|     auto abtc = static_cast<const AndroidToolChain *>(btc);
 | |
|     return aatc->targetAbi() == abtc->targetAbi();
 | |
| }
 | |
| 
 | |
| void AndroidConfigurations::registerNewToolChains()
 | |
| {
 | |
|     const QList<ToolChain *> existingAndroidToolChains
 | |
|             = ToolChainManager::toolChains(Utils::equal(&ToolChain::typeId, Id(Constants::ANDROID_TOOLCHAIN_TYPEID)));
 | |
|     QList<ToolChain *> newToolchains = AndroidToolChainFactory::autodetectToolChains(
 | |
|         existingAndroidToolChains);
 | |
| 
 | |
|     foreach (ToolChain *tc, newToolchains)
 | |
|         ToolChainManager::registerToolChain(tc);
 | |
| 
 | |
|     registerCustomToolChainsAndDebuggers();
 | |
| }
 | |
| 
 | |
| void AndroidConfigurations::removeOldToolChains()
 | |
| {
 | |
|     const auto tcs = ToolChainManager::toolChains(Utils::equal(&ToolChain::typeId,
 | |
|                                                                Id(Constants::ANDROID_TOOLCHAIN_TYPEID)));
 | |
|     for (ToolChain *tc : tcs) {
 | |
|         if (!tc->isValid())
 | |
|             ToolChainManager::deregisterToolChain(tc);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void AndroidConfigurations::removeUnusedDebuggers()
 | |
| {
 | |
|     const QList<BaseQtVersion*> qtVersions = QtVersionManager::versions([](const BaseQtVersion *v) {
 | |
|         return v->type() == Constants::ANDROID_QT_TYPE;
 | |
|     });
 | |
| 
 | |
|     QVector<FilePath> uniqueNdks;
 | |
|     for (const BaseQtVersion *qt : qtVersions) {
 | |
|         FilePath ndkLocation = currentConfig().ndkLocation(qt);
 | |
|         if (!uniqueNdks.contains(ndkLocation))
 | |
|             uniqueNdks.append(ndkLocation);
 | |
|     }
 | |
| 
 | |
|     uniqueNdks.append(Utils::transform(currentConfig().getCustomNdkList(),
 | |
|                                        FilePath::fromString).toVector());
 | |
| 
 | |
|     const QList<Debugger::DebuggerItem> allDebuggers = Debugger::DebuggerItemManager::debuggers();
 | |
|     for (const Debugger::DebuggerItem &debugger : allDebuggers) {
 | |
|         if (!debugger.displayName().contains("Android"))
 | |
|             continue;
 | |
| 
 | |
|         bool isChildOfNdk = false;
 | |
|         for (const FilePath &path : uniqueNdks) {
 | |
|             if (debugger.command().isChildOf(path)) {
 | |
|                 isChildOfNdk = true;
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         const bool isMultiAbiNdkGdb = debugger.command().fileName().startsWith("gdb");
 | |
|         const bool hasMultiAbiName = debugger.displayName().contains("Multi-Abi");
 | |
| 
 | |
|         if (debugger.isAutoDetected() && (!isChildOfNdk || (isMultiAbiNdkGdb && !hasMultiAbiName)))
 | |
|             Debugger::DebuggerItemManager::deregisterDebugger(debugger.id());
 | |
|     }
 | |
| }
 | |
| 
 | |
| static QStringList allSupportedAbis()
 | |
| {
 | |
|     return QStringList{
 | |
|         ProjectExplorer::Constants::ANDROID_ABI_ARMEABI_V7A,
 | |
|         ProjectExplorer::Constants::ANDROID_ABI_ARM64_V8A,
 | |
|         ProjectExplorer::Constants::ANDROID_ABI_X86,
 | |
|         ProjectExplorer::Constants::ANDROID_ABI_X86_64,
 | |
|     };
 | |
| }
 | |
| 
 | |
| static bool containsAllAbis(const QStringList &abis)
 | |
| {
 | |
|     QStringList supportedAbis{allSupportedAbis()};
 | |
|     for (const QString &abi : abis)
 | |
|         if (supportedAbis.contains(abi))
 | |
|             supportedAbis.removeOne(abi);
 | |
| 
 | |
|     return supportedAbis.isEmpty();
 | |
| }
 | |
| 
 | |
| static QString getMultiOrSingleAbiString(const QStringList &abis)
 | |
| {
 | |
|     return containsAllAbis(abis) ? "Multi-Abi" : abis.join(",");
 | |
| }
 | |
| 
 | |
| static QVariant findOrRegisterDebugger(ToolChain *tc,
 | |
|                                        const QStringList &abisList,
 | |
|                                        bool customDebugger = false)
 | |
| {
 | |
|     const auto ¤tConfig = AndroidConfigurations::currentConfig();
 | |
|     const FilePath ndk = static_cast<AndroidToolChain *>(tc)->ndkLocation();
 | |
|     const FilePath command = currentConfig.gdbPathFromNdk(tc->targetAbi(), ndk);
 | |
| 
 | |
|     // check if the debugger is already registered, but ignoring the display name
 | |
|     const Debugger::DebuggerItem *existing = Debugger::DebuggerItemManager::findByCommand(command);
 | |
| 
 | |
|     // Return existing debugger with same command
 | |
|     if (existing && existing->engineType() == Debugger::GdbEngineType
 | |
|             && existing->isAutoDetected()) {
 | |
|         return existing->id();
 | |
|     }
 | |
| 
 | |
|     // debugger not found, register a new one
 | |
|     Debugger::DebuggerItem debugger;
 | |
|     debugger.setCommand(command);
 | |
|     debugger.setEngineType(Debugger::GdbEngineType);
 | |
| 
 | |
|     // NDK 10 and older have multiple gdb versions per ABI, so check for that.
 | |
|     const bool oldNdkVersion = currentConfig.ndkVersion(ndk) <= QVersionNumber{11};
 | |
|     QString mainName = AndroidConfigurations::tr("Android Debugger (%1, NDK %2)");
 | |
|     if (customDebugger)
 | |
|         mainName.prepend("Custom ");
 | |
|     debugger.setUnexpandedDisplayName(mainName
 | |
|             .arg(getMultiOrSingleAbiString(oldNdkVersion ? abisList : allSupportedAbis()))
 | |
|             .arg(AndroidConfigurations::currentConfig().ndkVersion(ndk).toString()));
 | |
|     debugger.setAutoDetected(true);
 | |
|     debugger.reinitializeFromFile();
 | |
|     return Debugger::DebuggerItemManager::registerDebugger(debugger);
 | |
| }
 | |
| 
 | |
| void AndroidConfigurations::registerCustomToolChainsAndDebuggers()
 | |
| {
 | |
|     const QList<ToolChain *> existingAndroidToolChains = ToolChainManager::toolChains(
 | |
|         Utils::equal(&ToolChain::typeId, Utils::Id(Constants::ANDROID_TOOLCHAIN_TYPEID)));
 | |
|     QList<FilePath> customNdks = Utils::transform(currentConfig().getCustomNdkList(),
 | |
|                                                   FilePath::fromString);
 | |
|     QList<ToolChain *> customToolchains
 | |
|         = AndroidToolChainFactory::autodetectToolChainsFromNdks(existingAndroidToolChains,
 | |
|                                                                 customNdks,
 | |
|                                                                 true);
 | |
|     for (ToolChain *tc : customToolchains) {
 | |
|         ToolChainManager::registerToolChain(tc);
 | |
|         const auto androidToolChain = static_cast<AndroidToolChain *>(tc);
 | |
|         QString abiStr;
 | |
|         if (androidToolChain)
 | |
|             abiStr = androidToolChain->platformLinkerFlags().at(1).split('-').first();
 | |
|         findOrRegisterDebugger(tc, {abiStr}, true);
 | |
|     }
 | |
| }
 | |
| void AndroidConfigurations::updateAutomaticKitList()
 | |
| {
 | |
|     for (Kit *k : KitManager::kits()) {
 | |
|         if (DeviceTypeKitAspect::deviceTypeId(k) == Constants::ANDROID_DEVICE_TYPE) {
 | |
|             if (k->value(Constants::ANDROID_KIT_NDK).isNull() || k->value(Constants::ANDROID_KIT_SDK).isNull()) {
 | |
|                 if (BaseQtVersion *qt = QtKitAspect::qtVersion(k)) {
 | |
|                     k->setValueSilently(Constants::ANDROID_KIT_NDK, currentConfig().ndkLocation(qt).toString());
 | |
|                     k->setValue(Constants::ANDROID_KIT_SDK, currentConfig().sdkLocation().toString());
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     const QList<Kit *> existingKits = Utils::filtered(KitManager::kits(), [](Kit *k) {
 | |
|         Id deviceTypeId = DeviceTypeKitAspect::deviceTypeId(k);
 | |
|         if (k->isAutoDetected() && !k->isSdkProvided()
 | |
|                 && deviceTypeId == Constants::ANDROID_DEVICE_TYPE) {
 | |
|             return true;
 | |
|         }
 | |
|         return false;
 | |
|     });
 | |
| 
 | |
|     removeUnusedDebuggers();
 | |
| 
 | |
|     QHash<Abi, QList<const BaseQtVersion *> > qtVersionsForArch;
 | |
|     const QList<BaseQtVersion*> qtVersions = QtVersionManager::versions([](const BaseQtVersion *v) {
 | |
|         return v->type() == Constants::ANDROID_QT_TYPE;
 | |
|     });
 | |
|     for (const BaseQtVersion *qtVersion : qtVersions) {
 | |
|         const Abis qtAbis = qtVersion->qtAbis();
 | |
|         if (qtAbis.empty())
 | |
|             continue;
 | |
|         qtVersionsForArch[qtAbis.first()].append(qtVersion);
 | |
|     }
 | |
| 
 | |
|     // register new kits
 | |
|     const QList<ToolChain *> toolchains = ToolChainManager::toolChains([](const ToolChain *tc) {
 | |
|         return tc->isAutoDetected()
 | |
|             && tc->isValid()
 | |
|             && tc->typeId() == Constants::ANDROID_TOOLCHAIN_TYPEID;
 | |
|     });
 | |
|     QList<Kit *> unhandledKits = existingKits;
 | |
|     for (ToolChain *tc : toolchains) {
 | |
|         if (tc->language() != ProjectExplorer::Constants::CXX_LANGUAGE_ID)
 | |
|             continue;
 | |
| 
 | |
|         for (const BaseQtVersion *qt : qtVersionsForArch.value(tc->targetAbi())) {
 | |
|             FilePath tcNdk = static_cast<const AndroidToolChain *>(tc)->ndkLocation();
 | |
|             if (tcNdk != currentConfig().ndkLocation(qt))
 | |
|                 continue;
 | |
| 
 | |
|             const QList<ToolChain *> allLanguages
 | |
|                 = Utils::filtered(toolchains, [tc, tcNdk](ToolChain *otherTc) {
 | |
|                       FilePath otherNdk = static_cast<const AndroidToolChain *>(otherTc)->ndkLocation();
 | |
|                       return tc->targetAbi() == otherTc->targetAbi() && tcNdk == otherNdk;
 | |
|                   });
 | |
| 
 | |
|             QHash<Id, ToolChain *> toolChainForLanguage;
 | |
|             for (ToolChain *tc : allLanguages)
 | |
|                 toolChainForLanguage[tc->language()] = tc;
 | |
| 
 | |
|             Kit *existingKit = Utils::findOrDefault(existingKits, [&](const Kit *b) {
 | |
|                 if (qt != QtKitAspect::qtVersion(b))
 | |
|                     return false;
 | |
|                 return matchToolChain(toolChainForLanguage[ProjectExplorer::Constants::CXX_LANGUAGE_ID],
 | |
|                                       ToolChainKitAspect::cxxToolChain(b))
 | |
|                         && matchToolChain(toolChainForLanguage[ProjectExplorer::Constants::C_LANGUAGE_ID],
 | |
|                                           ToolChainKitAspect::cToolChain(b));
 | |
|             });
 | |
| 
 | |
|             const auto initializeKit = [allLanguages, tc, qt](Kit *k) {
 | |
|                 k->setAutoDetected(true);
 | |
|                 k->setAutoDetectionSource("AndroidConfiguration");
 | |
|                 DeviceTypeKitAspect::setDeviceTypeId(k, Constants::ANDROID_DEVICE_TYPE);
 | |
|                 for (ToolChain *tc : allLanguages)
 | |
|                     ToolChainKitAspect::setToolChain(k, tc);
 | |
|                 QtKitAspect::setQtVersion(k, qt);
 | |
|                 QStringList abis = static_cast<const AndroidQtVersion *>(qt)->androidAbis();
 | |
|                 Debugger::DebuggerKitAspect::setDebugger(k, findOrRegisterDebugger(tc, abis));
 | |
| 
 | |
|                 BuildDeviceKitAspect::setDeviceId(k, DeviceManager::defaultDesktopDevice()->id());
 | |
|                 k->setSticky(QtKitAspect::id(), true);
 | |
|                 k->setMutable(DeviceKitAspect::id(), true);
 | |
|                 k->setSticky(DeviceTypeKitAspect::id(), true);
 | |
| 
 | |
|                 QString versionStr = QLatin1String("Qt %{Qt:Version}");
 | |
|                 if (!qt->isAutodetected())
 | |
|                     versionStr = QString("%1").arg(qt->displayName());
 | |
|                 k->setUnexpandedDisplayName(tr("Android %1 Clang %2")
 | |
|                                                 .arg(versionStr)
 | |
|                                                 .arg(getMultiOrSingleAbiString(abis)));
 | |
|                 k->setValueSilently(Constants::ANDROID_KIT_NDK, currentConfig().ndkLocation(qt).toString());
 | |
|                 k->setValueSilently(Constants::ANDROID_KIT_SDK, currentConfig().sdkLocation().toString());
 | |
|             };
 | |
| 
 | |
|             if (existingKit) {
 | |
|                 initializeKit(existingKit); // Update the existing kit with new data.
 | |
|                 unhandledKits.removeOne(existingKit);
 | |
|             } else {
 | |
|                 KitManager::registerKit(initializeKit);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     // cleanup any mess that might have existed before, by removing all Android kits that
 | |
|     // existed before, but weren't re-used
 | |
|     for (Kit *k : unhandledKits)
 | |
|         KitManager::deregisterKit(k);
 | |
| }
 | |
| 
 | |
| bool AndroidConfigurations::force32bitEmulator()
 | |
| {
 | |
|     return m_instance->m_force32bit;
 | |
| }
 | |
| 
 | |
| Environment AndroidConfigurations::toolsEnvironment(const AndroidConfig &config)
 | |
| {
 | |
|     Environment env = Environment::systemEnvironment();
 | |
|     FilePath jdkLocation = config.openJDKLocation();
 | |
|     if (!jdkLocation.isEmpty()) {
 | |
|         env.set("JAVA_HOME", jdkLocation.toUserOutput());
 | |
|         env.prependOrSetPath(jdkLocation.pathAppended("bin").toUserOutput());
 | |
|     }
 | |
|     return env;
 | |
| }
 | |
| 
 | |
| AndroidConfig &AndroidConfigurations::currentConfig()
 | |
| {
 | |
|     return m_instance->m_config; // ensure that m_instance is initialized
 | |
| }
 | |
| 
 | |
| AndroidSdkManager *AndroidConfigurations::sdkManager()
 | |
| {
 | |
|     return m_instance->m_sdkManager.get();
 | |
| }
 | |
| 
 | |
| AndroidConfigurations *AndroidConfigurations::instance()
 | |
| {
 | |
|     return m_instance;
 | |
| }
 | |
| 
 | |
| void AndroidConfigurations::save()
 | |
| {
 | |
|     QSettings *settings = Core::ICore::settings();
 | |
|     settings->beginGroup(SettingsGroup);
 | |
|     m_config.save(*settings);
 | |
|     settings->endGroup();
 | |
| }
 | |
| 
 | |
| AndroidConfigurations::AndroidConfigurations()
 | |
|     : m_sdkManager(new AndroidSdkManager(m_config))
 | |
| {
 | |
|     load();
 | |
|     connect(DeviceManager::instance(), &DeviceManager::devicesLoaded,
 | |
|             this, &AndroidConfigurations::updateAndroidDevice);
 | |
| 
 | |
|     m_force32bit = is32BitUserSpace();
 | |
|     m_instance = this;
 | |
| }
 | |
| 
 | |
| AndroidConfigurations::~AndroidConfigurations() = default;
 | |
| 
 | |
| static FilePath androidStudioPath()
 | |
| {
 | |
| #if defined(Q_OS_WIN)
 | |
|     const QLatin1String registryKey("HKEY_LOCAL_MACHINE\\SOFTWARE\\Android Studio");
 | |
|     const QLatin1String valueName("Path");
 | |
|     const QSettings settings64(registryKey, QSettings::Registry64Format);
 | |
|     const QSettings settings32(registryKey, QSettings::Registry32Format);
 | |
|     return FilePath::fromUserInput(
 | |
|                 settings64.value(valueName, settings32.value(valueName).toString()).toString());
 | |
| #endif
 | |
|     return {}; // TODO non-Windows
 | |
| }
 | |
| 
 | |
| FilePath AndroidConfig::getJdkPath()
 | |
| {
 | |
|     FilePath jdkHome;
 | |
| 
 | |
|     if (HostOsInfo::isWindowsHost()) {
 | |
|         QStringList allVersions;
 | |
|         std::unique_ptr<QSettings> settings(
 | |
|             new QSettings(jdk8SettingsPath, QSettings::NativeFormat));
 | |
|         allVersions = settings->childGroups();
 | |
| #ifdef Q_OS_WIN
 | |
|         if (allVersions.isEmpty()) {
 | |
|             settings.reset(new QSettings(jdk8SettingsPath, QSettings::Registry64Format));
 | |
|             allVersions = settings->childGroups();
 | |
|         }
 | |
| #endif // Q_OS_WIN
 | |
| 
 | |
|         // If no jdk 1.8 can be found, look for jdk versions above 1.8
 | |
|         // Android section would warn if sdkmanager cannot run with newer jdk versions
 | |
|         if (allVersions.isEmpty()) {
 | |
|             settings.reset(new QSettings(jdkLatestSettingsPath, QSettings::NativeFormat));
 | |
|             allVersions = settings->childGroups();
 | |
| #ifdef Q_OS_WIN
 | |
|             if (allVersions.isEmpty()) {
 | |
|                 settings.reset(new QSettings(jdkLatestSettingsPath, QSettings::Registry64Format));
 | |
|                 allVersions = settings->childGroups();
 | |
|             }
 | |
| #endif // Q_OS_WIN
 | |
|         }
 | |
| 
 | |
|         for (const QString &version : qAsConst(allVersions)) {
 | |
|             settings->beginGroup(version);
 | |
|             jdkHome = FilePath::fromUserInput(settings->value("JavaHome").toString());
 | |
|             settings->endGroup();
 | |
|             if (version.startsWith("1.8")) {
 | |
|                 if (!jdkHome.exists())
 | |
|                     continue;
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // Nothing found yet? Let's try finding Android Studio's jdk
 | |
|         if (jdkHome.isEmpty()) {
 | |
|             const FilePath androidStudioSdkPath = androidStudioPath();
 | |
|             if (!androidStudioSdkPath.isEmpty()) {
 | |
|                 const FilePath androidStudioSdkJrePath = androidStudioSdkPath / "jre";
 | |
|                 if (androidStudioSdkJrePath.exists())
 | |
|                     jdkHome = androidStudioSdkJrePath;
 | |
|             }
 | |
|         }
 | |
|     } else {
 | |
|         QStringList args;
 | |
|         if (HostOsInfo::isMacHost())
 | |
|             args << "-c"
 | |
|                  << "/usr/libexec/java_home";
 | |
|         else
 | |
|             args << "-c"
 | |
|                  << "readlink -f $(which java)";
 | |
| 
 | |
|         QProcess findJdkPathProc;
 | |
|         findJdkPathProc.start("sh", args);
 | |
|         findJdkPathProc.waitForFinished();
 | |
|         QByteArray jdkPath = findJdkPathProc.readAllStandardOutput().trimmed();
 | |
| 
 | |
|         if (HostOsInfo::isMacHost()) {
 | |
|             jdkHome = FilePath::fromUtf8(jdkPath);
 | |
|         } else {
 | |
|             jdkPath.replace("bin/java", ""); // For OpenJDK 11
 | |
|             jdkPath.replace("jre", "");
 | |
|             jdkPath.replace("//", "/");
 | |
|             jdkHome = FilePath::fromUtf8(jdkPath);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return jdkHome;
 | |
| }
 | |
| 
 | |
| void AndroidConfigurations::load()
 | |
| {
 | |
|     QSettings *settings = Core::ICore::settings();
 | |
|     settings->beginGroup(SettingsGroup);
 | |
|     m_config.load(*settings);
 | |
|     settings->endGroup();
 | |
| }
 | |
| 
 | |
| void AndroidConfigurations::updateAndroidDevice()
 | |
| {
 | |
|     // Remove any dummy Android device, because it won't be usable.
 | |
|     DeviceManager *const devMgr = DeviceManager::instance();
 | |
|     IDevice::ConstPtr dev = devMgr->find(Constants::ANDROID_DEVICE_ID);
 | |
|     if (dev)
 | |
|         devMgr->removeDevice(dev->id());
 | |
| 
 | |
|     AndroidDeviceManager::instance()->setupDevicesWatcher();
 | |
| }
 | |
| 
 | |
| AndroidConfigurations *AndroidConfigurations::m_instance = nullptr;
 | |
| 
 | |
| } // namespace Android
 |