407 lines
15 KiB
Plaintext
407 lines
15 KiB
Plaintext
/*
|
|
* This file is part of LSPosed.
|
|
*
|
|
* LSPosed is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* LSPosed is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with LSPosed. If not, see <https://www.gnu.org/licenses/>.
|
|
*
|
|
* Copyright (C) 2021 LSPosed Contributors
|
|
*/
|
|
|
|
import com.android.build.api.component.analytics.AnalyticsEnabledApplicationVariant
|
|
import com.android.build.api.variant.impl.ApplicationVariantImpl
|
|
import com.android.build.gradle.BaseExtension
|
|
import com.android.ide.common.signing.KeystoreHelper
|
|
import org.apache.commons.codec.binary.Hex
|
|
import org.apache.tools.ant.filters.FixCrLfFilter
|
|
import org.apache.tools.ant.filters.ReplaceTokens
|
|
import java.io.ByteArrayOutputStream
|
|
import java.io.FileOutputStream
|
|
import java.io.PrintStream
|
|
import java.security.MessageDigest
|
|
import java.util.jar.JarFile
|
|
import java.util.zip.ZipOutputStream
|
|
|
|
plugins {
|
|
id("com.android.application")
|
|
}
|
|
|
|
val moduleName = "LSPosed"
|
|
val moduleBaseId = "lsposed"
|
|
val authors = "LSPosed Developers"
|
|
|
|
val riruModuleId = "lsposed"
|
|
val moduleMinRiruApiVersion = 25
|
|
val moduleMinRiruVersionName = "25.0.1"
|
|
val moduleMaxRiruApiVersion = 25
|
|
|
|
val injectedPackageName = "com.android.shell"
|
|
val injectedPackageUid = 2000
|
|
|
|
val agpVersion: String by rootProject.extra
|
|
|
|
val defaultManagerPackageName: String by rootProject.extra
|
|
val apiCode: Int by rootProject.extra
|
|
val verCode: Int by rootProject.extra
|
|
val verName: String by rootProject.extra
|
|
|
|
val androidTargetSdkVersion: Int by rootProject.extra
|
|
val androidMinSdkVersion: Int by rootProject.extra
|
|
val androidBuildToolsVersion: String by rootProject.extra
|
|
val androidCompileSdkVersion: Int by rootProject.extra
|
|
val androidCompileNdkVersion: String by rootProject.extra
|
|
val androidSourceCompatibility: JavaVersion by rootProject.extra
|
|
val androidTargetCompatibility: JavaVersion by rootProject.extra
|
|
|
|
android {
|
|
compileSdk = androidCompileSdkVersion
|
|
ndkVersion = androidCompileNdkVersion
|
|
buildToolsVersion = androidBuildToolsVersion
|
|
|
|
flavorDimensions += "api"
|
|
|
|
buildFeatures {
|
|
prefab = true
|
|
}
|
|
|
|
defaultConfig {
|
|
applicationId = "org.lsposed.lspd"
|
|
minSdk = androidMinSdkVersion
|
|
targetSdk = androidTargetSdkVersion
|
|
versionCode = verCode
|
|
versionName = verName
|
|
multiDexEnabled = false
|
|
|
|
externalNativeBuild {
|
|
ndkBuild {
|
|
arguments += "INJECTED_AID=$injectedPackageUid"
|
|
arguments += "VERSION_CODE=$verCode"
|
|
arguments += "VERSION_NAME=$verName"
|
|
arguments += "-j${Runtime.getRuntime().availableProcessors()}"
|
|
}
|
|
}
|
|
|
|
buildConfigField("int", "API_CODE", "$apiCode")
|
|
buildConfigField(
|
|
"String",
|
|
"DEFAULT_MANAGER_PACKAGE_NAME",
|
|
""""$defaultManagerPackageName""""
|
|
)
|
|
buildConfigField("String", "MANAGER_INJECTED_PKG_NAME", """"$injectedPackageName"""")
|
|
buildConfigField("int", "MANAGER_INJECTED_UID", """$injectedPackageUid""")
|
|
}
|
|
|
|
lint {
|
|
isAbortOnError = true
|
|
isCheckReleaseBuilds = false
|
|
}
|
|
|
|
buildTypes {
|
|
release {
|
|
isMinifyEnabled = true
|
|
proguardFiles("proguard-rules.pro")
|
|
}
|
|
}
|
|
externalNativeBuild {
|
|
ndkBuild {
|
|
path("src/main/cpp/Android.mk")
|
|
}
|
|
}
|
|
|
|
compileOptions {
|
|
targetCompatibility(androidTargetCompatibility)
|
|
sourceCompatibility(androidSourceCompatibility)
|
|
}
|
|
|
|
buildTypes {
|
|
all {
|
|
externalNativeBuild {
|
|
ndkBuild {
|
|
arguments += "NDK_OUT=${File(buildDir, ".cxx/$name").absolutePath}"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
productFlavors {
|
|
all {
|
|
externalNativeBuild {
|
|
ndkBuild {
|
|
arguments += "MODULE_NAME=${name.toLowerCase()}_$moduleBaseId"
|
|
arguments += "API=${name.toLowerCase()}"
|
|
}
|
|
}
|
|
buildConfigField("String", "API", """"$name"""")
|
|
}
|
|
|
|
create("Riru") {
|
|
dimension = "api"
|
|
externalNativeBuild {
|
|
ndkBuild {
|
|
arguments += "API_VERSION=$moduleMaxRiruApiVersion"
|
|
}
|
|
}
|
|
}
|
|
|
|
create("Zygisk") {
|
|
dimension = "api"
|
|
externalNativeBuild {
|
|
ndkBuild {
|
|
arguments += "API_VERSION=1"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
|
|
dependencies {
|
|
// keep this dep since it affects ccache
|
|
implementation("dev.rikka.ndk:riru:26.0.0")
|
|
implementation("dev.rikka.ndk.thirdparty:cxx:1.2.0")
|
|
implementation("io.github.vvb2060.ndk:dobby:1.2")
|
|
implementation("com.android.tools.build:apksig:$agpVersion")
|
|
implementation("org.apache.commons:commons-lang3:3.12.0")
|
|
implementation("de.upb.cs.swt:axml:2.1.1")
|
|
compileOnly("androidx.annotation:annotation:1.2.0")
|
|
compileOnly(project(":hiddenapi-stubs"))
|
|
implementation(project(":hiddenapi-bridge"))
|
|
implementation(project(":manager-service"))
|
|
android.applicationVariants.all {
|
|
"${name}Implementation"(files(File(project.buildDir, "tmp/${name}R.jar")) {
|
|
builtBy("generateApp${name}RFile")
|
|
})
|
|
}
|
|
}
|
|
|
|
val zipAll = task("zipAll", Task::class) {
|
|
|
|
}
|
|
|
|
androidComponents.onVariants { v ->
|
|
val variant: ApplicationVariantImpl =
|
|
if (v is ApplicationVariantImpl) v
|
|
else (v as AnalyticsEnabledApplicationVariant).delegate as ApplicationVariantImpl
|
|
val variantCapped = variant.name.capitalize()
|
|
val variantLowered = variant.name.toLowerCase()
|
|
val buildTypeCapped = variant.buildType!!.capitalize()
|
|
val buildTypeLowered = variant.buildType!!.toLowerCase()
|
|
val flavorCapped = variant.flavorName!!.capitalize()
|
|
val flavorLowered = variant.flavorName!!.toLowerCase()
|
|
|
|
val magiskDir = "$buildDir/magisk/$variantLowered"
|
|
|
|
task("generateApp${variantCapped}RFile", Jar::class) {
|
|
dependsOn(":app:process${buildTypeCapped}Resources")
|
|
doLast {
|
|
val rFile = JarFile(
|
|
File(
|
|
project(":app").buildDir,
|
|
"intermediates/compile_and_runtime_not_namespaced_r_class_jar/${buildTypeLowered}/R.jar"
|
|
)
|
|
)
|
|
ZipOutputStream(FileOutputStream(File(project.buildDir, "tmp/${variantCapped}R.jar"))).use {
|
|
for (entry in rFile.entries()) {
|
|
if (entry.name.startsWith("org/lsposed/manager")) {
|
|
it.putNextEntry(entry)
|
|
rFile.getInputStream(entry).transferTo(it)
|
|
it.closeEntry()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
afterEvaluate {
|
|
val app = rootProject.project(":app").extensions.getByName<BaseExtension>("android")
|
|
val outSrcDir = file("$buildDir/generated/source/signInfo/${variantLowered}")
|
|
val outSrc = file("$outSrcDir/org/lsposed/lspd/util/SignInfo.java")
|
|
val signInfoTask = tasks.register("generate${variantCapped}SignInfo") {
|
|
dependsOn(":app:validateSigning${buildTypeCapped}")
|
|
outputs.file(outSrc)
|
|
doLast {
|
|
val sign = app.buildTypes.named(buildTypeLowered).get().signingConfig
|
|
outSrc.parentFile.mkdirs()
|
|
val certificateInfo = KeystoreHelper.getCertificateInfo(
|
|
sign?.storeType,
|
|
sign?.storeFile,
|
|
sign?.storePassword,
|
|
sign?.keyPassword,
|
|
sign?.keyAlias
|
|
)
|
|
PrintStream(outSrc).apply {
|
|
println("package org.lsposed.lspd.util;")
|
|
println("public final class SignInfo {")
|
|
print("public static final byte[] CERTIFICATE = {")
|
|
val bytes = certificateInfo.certificate.encoded
|
|
print(bytes.joinToString(",") { it.toString() })
|
|
println("};")
|
|
println("}")
|
|
}
|
|
}
|
|
}
|
|
variant.variantData.registerJavaGeneratingTask(signInfoTask, arrayListOf(outSrcDir))
|
|
}
|
|
|
|
val moduleId = "${flavorLowered}_$moduleBaseId"
|
|
val zipFileName = "$moduleName-v$verName-$verCode-${flavorLowered}-$buildTypeLowered.zip"
|
|
|
|
val prepareMagiskFilesTask = task("prepareMagiskFiles$variantCapped", Sync::class) {
|
|
dependsOn("assemble$variantCapped")
|
|
dependsOn(":app:assemble$buildTypeCapped")
|
|
into(magiskDir)
|
|
from("${rootProject.projectDir}/README.md")
|
|
from("$projectDir/magisk_module") {
|
|
exclude("riru.sh", "module.prop", "customize.sh", "sepolicy.rule")
|
|
}
|
|
from("$projectDir/magisk_module") {
|
|
include("module.prop")
|
|
expand(
|
|
"moduleId" to moduleId,
|
|
"versionName" to "v$verName",
|
|
"versionCode" to verCode,
|
|
"authorList" to authors,
|
|
"requirement" to when (flavorLowered) {
|
|
"riru" -> "Requires Riru $moduleMinRiruVersionName or above installed"
|
|
"zygisk" -> "Requires Magisk 24.0+ and Zygisk enabled"
|
|
else -> "No further requirements"
|
|
},
|
|
"api" to flavorCapped
|
|
)
|
|
filter<FixCrLfFilter>("eol" to FixCrLfFilter.CrLf.newInstance("lf"))
|
|
}
|
|
from("$projectDir/magisk_module") {
|
|
include("customize.sh")
|
|
val tokens = mapOf("FLAVOR" to flavorLowered)
|
|
filter<ReplaceTokens>("tokens" to tokens)
|
|
filter<FixCrLfFilter>("eol" to FixCrLfFilter.CrLf.newInstance("lf"))
|
|
}
|
|
if (flavorLowered == "riru") {
|
|
from("${projectDir}/magisk_module") {
|
|
include("riru.sh", "sepolicy.rule")
|
|
val tokens = mapOf(
|
|
"RIRU_MODULE_LIB_NAME" to "lspd",
|
|
"RIRU_MODULE_API_VERSION" to moduleMaxRiruApiVersion.toString(),
|
|
"RIRU_MODULE_MIN_API_VERSION" to moduleMinRiruApiVersion.toString(),
|
|
"RIRU_MODULE_MIN_RIRU_VERSION_NAME" to moduleMinRiruVersionName,
|
|
"RIRU_MODULE_DEBUG" to if (buildTypeLowered == "debug") "true" else "false",
|
|
)
|
|
filter<ReplaceTokens>("tokens" to tokens)
|
|
filter<FixCrLfFilter>("eol" to FixCrLfFilter.CrLf.newInstance("lf"))
|
|
}
|
|
}
|
|
from("${project(":app").buildDir}/outputs/apk/${buildTypeLowered}") {
|
|
include("*.apk")
|
|
rename(".*\\.apk", "manager.apk")
|
|
}
|
|
into("lib") {
|
|
from("${buildDir}/intermediates/stripped_native_libs/$variantCapped/out/lib")
|
|
}
|
|
val dexOutPath = if (buildTypeLowered == "release")
|
|
"$buildDir/intermediates/dex/$variantCapped/minify${variantCapped}WithR8" else
|
|
"$buildDir/intermediates/dex/$variantCapped/mergeDex$variantCapped"
|
|
into("framework") {
|
|
from(dexOutPath)
|
|
rename("classes.dex", "lspd.dex")
|
|
}
|
|
doLast {
|
|
fileTree(magiskDir).visit {
|
|
if (isDirectory) return@visit
|
|
val md = MessageDigest.getInstance("SHA-256")
|
|
file.forEachBlock(4096) { bytes, size ->
|
|
md.update(bytes, 0, size)
|
|
}
|
|
file(file.path + ".sha256").writeText(Hex.encodeHexString(md.digest()))
|
|
}
|
|
}
|
|
}
|
|
|
|
val zipTask = task("zip${variantCapped}", Zip::class) {
|
|
dependsOn(prepareMagiskFilesTask)
|
|
archiveFileName.set(zipFileName)
|
|
destinationDirectory.set(file("$projectDir/release"))
|
|
from(magiskDir)
|
|
}
|
|
|
|
zipAll.dependsOn(zipTask)
|
|
|
|
val adb: String = androidComponents.sdkComponents.adb.get().asFile.absolutePath
|
|
val pushTask = task("push${variantCapped}", Exec::class) {
|
|
dependsOn(zipTask)
|
|
workingDir("${projectDir}/release")
|
|
commandLine(adb, "push", zipFileName, "/data/local/tmp/")
|
|
}
|
|
val flashTask = task("flash${variantCapped}", Exec::class) {
|
|
dependsOn(pushTask)
|
|
commandLine(
|
|
adb, "shell", "su", "-c",
|
|
"magisk --install-module /data/local/tmp/${zipFileName}"
|
|
)
|
|
}
|
|
task("flashAndReboot${variantCapped}", Exec::class) {
|
|
dependsOn(flashTask)
|
|
commandLine(adb, "shell", "reboot")
|
|
}
|
|
}
|
|
|
|
val adb: String = androidComponents.sdkComponents.adb.get().asFile.absolutePath
|
|
val killLspd = task("killLspd", Exec::class) {
|
|
commandLine(adb, "shell", "su", "-c", "killall", "lspd")
|
|
isIgnoreExitValue = true
|
|
}
|
|
val pushLspd = task("pushLspd", Exec::class) {
|
|
dependsOn("mergeDexRiruDebug")
|
|
workingDir("$buildDir/intermediates/dex/RiruDebug/mergeDexRiruDebug")
|
|
commandLine(adb, "push", "classes.dex", "/data/local/tmp/lspd.dex")
|
|
}
|
|
val pushLspdNative = task("pushLspdNative", Exec::class) {
|
|
dependsOn("mergeRiruDebugNativeLibs")
|
|
doFirst {
|
|
val abi: String = ByteArrayOutputStream().use { outputStream ->
|
|
exec {
|
|
commandLine(adb, "shell", "getprop", "ro.product.cpu.abi")
|
|
standardOutput = outputStream
|
|
}
|
|
outputStream.toString().trim()
|
|
}
|
|
workingDir("$buildDir/intermediates/merged_native_libs/RiruDebug/out/lib/$abi")
|
|
}
|
|
commandLine(adb, "push", "libdaemon.so", "/data/local/tmp/libdaemon.so")
|
|
}
|
|
val reRunLspd = task("reRunLspd", Exec::class) {
|
|
dependsOn(pushLspd)
|
|
dependsOn(pushLspdNative)
|
|
dependsOn(killLspd)
|
|
commandLine(adb, "shell", "su", "-c", "sh /data/adb/modules/*_lsposed/service.sh&")
|
|
isIgnoreExitValue = true
|
|
}
|
|
val tmpApk = "/data/local/tmp/lsp.apk"
|
|
val pushApk = task("pushApk", Exec::class) {
|
|
dependsOn(":app:assembleDebug")
|
|
workingDir("${project(":app").buildDir}/outputs/apk/debug")
|
|
commandLine(adb, "push", "LSPosedManager-v$verName-$verCode-debug.apk", tmpApk)
|
|
}
|
|
val openApp = task("openApp", Exec::class) {
|
|
commandLine(
|
|
adb, "shell", "am start -a android.intent.action.MAIN " +
|
|
"-c org.lsposed.manager.LAUNCH_MANAGER " +
|
|
"com.android.shell/.BugreportWarningActivity"
|
|
)
|
|
}
|
|
task("reRunApp", Exec::class) {
|
|
dependsOn(pushApk)
|
|
commandLine(adb, "shell", "su", "-c", "mv -f $tmpApk /data/adb/lspd/manager.apk")
|
|
isIgnoreExitValue = true
|
|
finalizedBy(reRunLspd)
|
|
}
|