// Copyright (C) 2016 BogDan Vatra // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0 #include "androidconfigurations.h" #include "androidconstants.h" #include "androidmanager.h" #include "androidqtversion.h" #include "androidtr.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace ProjectExplorer; namespace Android { namespace Internal { AndroidQtVersion::AndroidQtVersion() : QtSupport::QtVersion() , m_guard(std::make_unique()) { QObject::connect(AndroidConfigurations::instance(), &AndroidConfigurations::aboutToUpdate, m_guard.get(), [this] { resetCache(); }); } bool AndroidQtVersion::isValid() const { if (!QtVersion::isValid()) return false; if (qtAbis().isEmpty()) return false; return true; } QString AndroidQtVersion::invalidReason() const { QString tmp = QtVersion::invalidReason(); if (tmp.isEmpty()) { if (AndroidConfigurations::currentConfig().ndkLocation(this).isEmpty()) return Tr::tr("NDK is not configured in Devices > Android."); if (AndroidConfigurations::currentConfig().sdkLocation().isEmpty()) return Tr::tr("SDK is not configured in Devices > Android."); if (qtAbis().isEmpty()) return Tr::tr("Failed to detect the ABIs used by the Qt version. Check the settings in " "Devices > Android for errors."); } return tmp; } bool AndroidQtVersion::supportsMultipleQtAbis() const { return qtVersion() >= QVersionNumber(5, 14) && qtVersion() < QVersionNumber(6, 0); } Abis AndroidQtVersion::detectQtAbis() const { const bool conf = AndroidConfigurations::currentConfig().sdkFullyConfigured(); return conf ? Utils::transform(androidAbis(), &AndroidManager::androidAbi2Abi) : Abis(); } void AndroidQtVersion::addToEnvironment(const Kit *k, Utils::Environment &env) const { QtVersion::addToEnvironment(k, env); const AndroidConfig &config = AndroidConfigurations::currentConfig(); // this env vars are used by qmake mkspecs to generate makefiles (check QTDIR/mkspecs/android-g++/qmake.conf for more info) env.set(QLatin1String("ANDROID_NDK_HOST"), config.toolchainHost(this)); env.set(QLatin1String("ANDROID_NDK_ROOT"), config.ndkLocation(this).toUserOutput()); env.set(QLatin1String("ANDROID_NDK_PLATFORM"), config.bestNdkPlatformMatch(qMax(minimumNDK(), AndroidManager::minimumSDK(k)), this)); } void AndroidQtVersion::setupQmakeRunEnvironment(Utils::Environment &env) const { env.set(QLatin1String("ANDROID_NDK_ROOT"), AndroidConfigurations::currentConfig().ndkLocation(this).toUserOutput()); } QString AndroidQtVersion::description() const { //: Qt Version is meant for Android return Tr::tr("Android"); } const QStringList &AndroidQtVersion::androidAbis() const { ensureMkSpecParsed(); return m_androidAbis; } int AndroidQtVersion::minimumNDK() const { ensureMkSpecParsed(); return m_minNdk; } Utils::FilePath AndroidQtVersion::androidDeploymentSettings(const Target *target) { // Try to fetch the file name from node data as provided by qmake and Qbs QString buildKey = target->activeBuildKey(); const ProjectNode *node = target->project()->findNodeForBuildKey(buildKey); if (node) { const QString nameFromData = node->data(Constants::AndroidDeploySettingsFile).toString(); if (!nameFromData.isEmpty()) return Utils::FilePath::fromUserInput(nameFromData); } // If unavailable, construct the name by ourselves (CMake) const BuildSystem *bs = target->buildSystem(); if (!bs) return {}; const QString displayName = bs->buildTarget(buildKey).displayName; return AndroidManager::buildDirectory(target).pathAppended( AndroidManager::isQt5CmakeProject(target) ? QLatin1String("android_deployment_settings.json") : QString::fromLatin1("android-%1-deployment-settings.json") .arg(displayName)); } void AndroidQtVersion::parseMkSpec(ProFileEvaluator *evaluator) const { m_androidAbis = evaluator->values("ALL_ANDROID_ABIS"); if (m_androidAbis.isEmpty()) m_androidAbis = QStringList{evaluator->value(Constants::ANDROID_TARGET_ARCH)}; const QString androidPlatform = evaluator->value("ANDROID_PLATFORM"); if (!androidPlatform.isEmpty()) { const QRegularExpression regex("android-(\\d+)"); const QRegularExpressionMatch match = regex.match(androidPlatform); if (match.hasMatch()) { bool ok = false; int tmp = match.captured(1).toInt(&ok); if (ok) m_minNdk = tmp; } } QtVersion::parseMkSpec(evaluator); } QSet AndroidQtVersion::availableFeatures() const { QSet features = QtSupport::QtVersion::availableFeatures(); features.insert(QtSupport::Constants::FEATURE_MOBILE); features.remove(QtSupport::Constants::FEATURE_QT_CONSOLE); features.remove(QtSupport::Constants::FEATURE_QT_WEBKIT); return features; } QSet AndroidQtVersion::targetDeviceTypes() const { return {Constants::ANDROID_DEVICE_TYPE}; } // Factory AndroidQtVersionFactory::AndroidQtVersionFactory() { setQtVersionCreator([] { return new AndroidQtVersion; }); setSupportedType(Constants::ANDROID_QT_TYPE); setPriority(90); setRestrictionChecker([](const SetupData &setup) { return !setup.config.contains("android-no-sdk") && (setup.config.contains("android") || setup.platforms.contains("android")); }); } } // Internal } // Android