summaryrefslogtreecommitdiffstats
path: root/mobile/android/gradle/with_gecko_binaries.gradle
diff options
context:
space:
mode:
Diffstat (limited to 'mobile/android/gradle/with_gecko_binaries.gradle')
-rw-r--r--mobile/android/gradle/with_gecko_binaries.gradle320
1 files changed, 320 insertions, 0 deletions
diff --git a/mobile/android/gradle/with_gecko_binaries.gradle b/mobile/android/gradle/with_gecko_binaries.gradle
new file mode 100644
index 0000000000..3dbdfe0990
--- /dev/null
+++ b/mobile/android/gradle/with_gecko_binaries.gradle
@@ -0,0 +1,320 @@
+/* -*- Mode: Groovy; c-basic-offset: 4; tab-width: 20; indent-tabs-mode: nil; -*-
+ * This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+// The JNI wrapper generation tasks depend on the JAR creation task of the :annotations project.
+evaluationDependsOn(':annotations')
+
+import groovy.util.FileNameFinder
+import groovy.transform.Memoized
+
+import java.nio.file.Path
+import java.nio.file.Paths
+import java.security.MessageDigest
+
+// To find the LLVM tools from the Android NDK, there are a few wrinkles. In a compiled build,
+// we can use our own `ANDROID_NDK` configure option. But in an artifact build, that isn't
+// defined, so we fall back to `android.ndkDirectory` -- but that's defined in
+// `local.properties`, which may not define it. In that case, fall back to crawling the
+// filesystem.
+@Memoized
+def getNDKDirectory() {
+ if (project.mozconfig.substs.ANDROID_NDK) {
+ return project.mozconfig.substs.ANDROID_NDK
+ }
+ if (project.android.ndkDirectory) {
+ return project.android.ndkDirectory
+ }
+ def mozbuild = System.getenv('MOZBUILD_STATE_PATH') ?: "${System.getProperty('user.home')}/.mozbuild"
+ def files = new FileNameFinder().getFileNames(mozbuild, "android-ndk-*/source.properties")
+ if (files) {
+ // It would be nice to sort these by version, but that's too much effort right now.
+ return project.file(files.first()).parentFile.toString()
+ }
+ return null
+}
+
+ext.getNDKDirectory = {
+ // This is the easiest way to expose the memoized function to consumers.
+ getNDKDirectory()
+}
+
+// Bug 1657190: This task works around a bug in the Android-Gradle plugin. When using SNAPSHOT
+// builds (and possibly in other situations) the JNI library invalidation process isn't correct. If
+// there are two JNI libs `a.so` and `b.so` and a new snapshot updates only `a.so`, then the
+// resulting AAR will include both JNI libs but the consuming project's resulting APK will include
+// only the modified `a.so`. For GeckoView, it's usually `libxul.so` that is updated. For a
+// consumer (like Fenix), this leads to a hard startup crash because `libmozglue.so` will be missing
+// when it is read (first, before `libxul.so`)
+//
+// For !MOZILLA_OFFICIAL builds, we work around this issue to ensure that when *any* library is
+// updated then *every* library is updated. We use the ELF build ID baked into each library to
+// determine whether any library is updated. We digest (SHA256, for now) all of the ELF build IDs
+// and use this as our own Mozilla-specific "library generation ID". We then add our own
+// Mozilla-specific ELF section to every library so that they will all be invalidated by the
+// Android-Gradle plugin of a consuming project.
+class SyncLibsAndUpdateGenerationID extends DefaultTask {
+ @InputDirectory
+ File source
+
+ @InputFile
+ File extraSource
+
+ @OutputDirectory
+ File destinationDir
+
+ // The `**` in the pattern depends on the host architecture. We could compute them or bake them
+ // in, but this is easy enough. `FileNameFinder` won't return a directory, so we find a file
+ // and return its parent directory.
+ @Input
+ File llvmBin = project.file(new FileNameFinder().getFileNames(project.ext.getNDKDirectory(), "toolchains/llvm/prebuilt/**/bin/llvm-*")
+ .first()).parentFile
+
+ // Sibling to `.note.gnu.build-id`.
+ @Input
+ String newElfSection = ".note.mozilla.build-id"
+
+ @TaskAction
+ void execute() {
+ Path root = Paths.get(source.toString())
+
+ def libs = project.fileTree(source.toString()).files.collect {
+ Path lib = Paths.get(it.toString())
+ root.relativize(lib).toString()
+ }
+
+ def generationId = new ByteArrayOutputStream().withStream { os ->
+ // Start with the hash of any extra source.
+ def digest = MessageDigest.getInstance("SHA-256")
+ extraSource.eachByte(64 * 1024) { byte[] buf, int bytesRead ->
+ digest.update(buf, 0, bytesRead);
+ }
+ def extraSourceHex = new BigInteger(1, digest.digest()).toString(16).padLeft(64, '0')
+ os.write("${extraSource.toString()} ${extraSourceHex}\n".getBytes("utf-8"));
+
+ // Follow with all the ordered build ID section dumps.
+ libs.each { lib ->
+ // This should never fail. If it does, there's a real problem, so an exception is
+ // reasonable.
+ project.exec {
+ commandLine "${llvmBin}/llvm-readobj"
+ args '--hex-dump=.note.gnu.build-id'
+ args "${source}/${lib}"
+ standardOutput = os
+ }
+ }
+
+ def allBuildIDs = os.toString()
+
+ // For detailed debugging.
+ new File("${destinationDir}/${newElfSection}-details").setText(allBuildIDs, 'utf-8')
+
+ allBuildIDs.sha256()
+ }
+
+ logger.info("Mozilla-specific library generation ID is ${generationId} (see ${destinationDir}/${newElfSection}-details)")
+
+ def generationIdIsStale = libs.any { lib ->
+ new ByteArrayOutputStream().withStream { os ->
+ // This can fail, but there's little value letting stderr go to the console in
+ // general, since it's just noise after a clobber build.
+ def execResult = project.exec {
+ ignoreExitValue true
+ commandLine "${llvmBin}/llvm-readobj"
+ args "--hex-dump=${newElfSection}"
+ args "${destinationDir}/${lib}"
+ standardOutput = os
+ errorOutput = new ByteArrayOutputStream()
+ }
+
+ if (execResult.exitValue != 0) {
+ logger.info("Mozilla-specific library generation ID is missing: ${lib}")
+ } else if (!os.toString().contains(generationId)) {
+ logger.info("Mozilla-specific library generation ID is stale: ${lib}")
+ } else {
+ logger.debug("Mozilla-specific library generation ID is fresh: ${lib}")
+ }
+ execResult.exitValue != 0 || !os.toString().contains(generationId)
+ }
+ }
+
+ if (generationIdIsStale) {
+ project.mkdir destinationDir
+ new File("${destinationDir}/${newElfSection}").setText(generationId, 'utf-8')
+
+ libs.each { lib ->
+ project.mkdir project.file("${destinationDir}/${lib}").parent
+
+ new ByteArrayOutputStream().withStream { os ->
+ // This should never fail: if it does, there's a real problem (again).
+ project.exec {
+ commandLine "${llvmBin}/llvm-objcopy"
+ args "--add-section=${newElfSection}=${destinationDir}/${newElfSection}"
+ args "${source}/${lib}"
+ args "${destinationDir}/${lib}"
+ standardOutput = os
+ }
+ }
+ }
+ } else {
+ logger.info("Mozilla-specific library generation ID is fresh!")
+ }
+ }
+}
+
+ext.configureVariantWithGeckoBinaries = { variant ->
+ def omnijarDir = "${topobjdir}/dist/geckoview"
+ def distDir = "${topobjdir}/dist/geckoview"
+
+ def syncOmnijarFromDistDir = task("syncOmnijarFromDistDirFor${variant.name.capitalize()}", type: Sync) {
+ onlyIf {
+ if (source.empty) {
+ throw new StopExecutionException("Required omnijar not found in ${omnijarDir}/{omni.ja,assets/omni.ja}. Have you built and packaged?")
+ }
+ return true
+ }
+
+ into("${project.buildDir}/moz.build/src/${variant.name}/omnijar")
+ from("${omnijarDir}/omni.ja",
+ "${omnijarDir}/assets/omni.ja") {
+ // Throw an exception if we find multiple, potentially conflicting omni.ja files.
+ duplicatesStrategy 'fail'
+ }
+ }
+
+ // For !MOZILLA_OFFICIAL builds, work around an Android-Gradle plugin bug that causes startup
+ // crashes with local substitution. But -- we want to allow artifact builds that don't have the
+ // NDK installed. See class comment above.
+ def shouldUpdateGenerationID = {
+ if (mozconfig.substs.MOZILLA_OFFICIAL) {
+ return false
+ } else if (mozconfig.substs.MOZ_ANDROID_FAT_AAR_ARCHITECTURES) {
+ return false
+ } else if (ext.getNDKDirectory() == null) {
+ logger.warn("Could not determine Android NDK directory.")
+ logger.warn("Set `ndk.dir` in `${project.topsrcdir}/local.properties` to avoid startup crashes when using `substitute-local-geckoview.gradle`.")
+ logger.warn("See https://bugzilla.mozilla.org/show_bug.cgi?id=1657190.")
+ return false
+ }
+ return true
+ }()
+
+ def syncLibsFromDistDir = { if (shouldUpdateGenerationID) {
+ def jarTask = tasks["bundleLibRuntime${variant.name.capitalize()}"]
+ def bundleJar = jarTask.outputs.files.find({ it.name == 'classes.jar' })
+
+ task("syncLibsAndUpdateGenerationIDFromDistDirFor${variant.name.capitalize()}", type: SyncLibsAndUpdateGenerationID) {
+ source file("${distDir}/lib")
+ destinationDir file("${project.buildDir}/moz.build/src/${variant.name}/jniLibs")
+ // Include the hash of classes.jar as well, so that JVM-only changes don't leave every
+ // JNI library unchanged and therefore invalidate all of the JNI libraries in a consumer
+ // doing local substitution.
+ extraSource bundleJar
+ dependsOn jarTask
+ }
+ } else {
+ task("syncLibsFromDistDirFor${variant.name.capitalize()}", type: Sync) {
+ into("${project.buildDir}/moz.build/src/${variant.name}/jniLibs")
+ from("${distDir}/lib")
+ }
+ } }()
+
+ syncLibsFromDistDir.onlyIf { task ->
+ if (!mozconfig.substs.COMPILE_ENVIRONMENT && !mozconfig.substs.MOZ_ARTIFACT_BUILDS) {
+ // We won't have JNI libraries if we're not compiling and we're not downloading
+ // artifacts. Such a configuration is used for running lints, generating docs, etc.
+ return true
+ }
+ if (files(task.source).empty) {
+ throw new StopExecutionException("Required JNI libraries not found in ${task.source}. Have you built and packaged?")
+ }
+ return true
+ }
+
+ def syncAssetsFromDistDir = task("syncAssetsFromDistDirFor${variant.name.capitalize()}", type: Sync) {
+ into("${project.buildDir}/moz.build/src/${variant.name}/assets")
+ from("${distDir}/assets") {
+ exclude 'omni.ja'
+ }
+ }
+
+ // Local (read, not 'official') builds want to reflect developer changes to
+ // the omnijar sources, and (when compiling) to reflect developer changes to
+ // the native binaries. To do this, the Gradle build calls out to the
+ // moz.build system, which can be re-entrant. Official builds are driven by
+ // the moz.build system and should never be re-entrant in this way.
+ if (!mozconfig.substs.MOZILLA_OFFICIAL) {
+ syncOmnijarFromDistDir.dependsOn rootProject.machStagePackage
+ syncLibsFromDistDir.dependsOn rootProject.machStagePackage
+ syncAssetsFromDistDir.dependsOn rootProject.machStagePackage
+ }
+
+ def assetGenTask = tasks.findByName("generate${variant.name.capitalize()}Assets")
+ def jniLibFoldersTask = tasks.findByName("merge${variant.name.capitalize()}JniLibFolders")
+ if ((variant.productFlavors*.name).contains('withGeckoBinaries')) {
+ assetGenTask.dependsOn syncOmnijarFromDistDir
+ assetGenTask.dependsOn syncAssetsFromDistDir
+ jniLibFoldersTask.dependsOn syncLibsFromDistDir
+
+ android.sourceSets."${variant.name}".assets.srcDir syncOmnijarFromDistDir.destinationDir
+ android.sourceSets."${variant.name}".assets.srcDir syncAssetsFromDistDir.destinationDir
+
+ if (!mozconfig.substs.MOZ_ANDROID_FAT_AAR_ARCHITECTURES) {
+ android.sourceSets."${variant.name}".jniLibs.srcDir syncLibsFromDistDir.destinationDir
+ } else {
+ android.sourceSets."${variant.name}".jniLibs.srcDir "${topobjdir}/dist/fat-aar/output/jni"
+ }
+ }
+}
+
+ext.configureLibraryVariantWithJNIWrappers = { variant, module ->
+ // BundleLibRuntime prepares the library for further processing to be
+ // incorporated in an app. We use this version to create the JNI wrappers.
+ def jarTask = tasks["bundleLibRuntime${variant.name.capitalize()}"]
+ def bundleJar = jarTask.outputs.files.find({ it.name == 'classes.jar' })
+
+ def annotationProcessorsJarTask = project(':annotations').jar
+
+ def wrapperTask
+ if (System.env.IS_LANGUAGE_REPACK == '1') {
+ // Single-locale l10n repacks set `IS_LANGUAGE_REPACK=1` and don't
+ // really have a build environment.
+ wrapperTask = task("generateJNIWrappersFor${module}${variant.name.capitalize()}")
+ } else {
+ wrapperTask = task("generateJNIWrappersFor${module}${variant.name.capitalize()}", type: JavaExec) {
+ classpath annotationProcessorsJarTask.archivePath
+
+ // Configure the classpath at evaluation-time, not at
+ // configuration-time: see above comment.
+ doFirst {
+ classpath variant.javaCompileProvider.get().classpath
+ // Include android.jar.
+ classpath variant.javaCompileProvider.get().options.bootstrapClasspath
+ }
+
+ main = 'org.mozilla.gecko.annotationProcessors.AnnotationProcessor'
+ args module
+ args bundleJar
+
+ workingDir "${topobjdir}/widget/android"
+
+ inputs.file(bundleJar)
+ inputs.file(annotationProcessorsJarTask.archivePath)
+ inputs.property("module", module)
+
+ outputs.file("${topobjdir}/widget/android/GeneratedJNINatives.h")
+ outputs.file("${topobjdir}/widget/android/GeneratedJNIWrappers.cpp")
+ outputs.file("${topobjdir}/widget/android/GeneratedJNIWrappers.h")
+
+ dependsOn jarTask
+ dependsOn annotationProcessorsJarTask
+ }
+ }
+
+ if (module == 'Generated') {
+ tasks["bundle${variant.name.capitalize()}Aar"].dependsOn wrapperTask
+ } else {
+ tasks["assemble${variant.name.capitalize()}"].dependsOn wrapperTask
+ }
+}