Magisk/buildSrc/src/main/java/Codegen.kt

247 lines
7.2 KiB
Kotlin
Raw Normal View History

2021-09-02 21:31:33 -07:00
import java.io.ByteArrayOutputStream
import java.io.File
2021-12-14 21:30:15 +08:00
import java.io.InputStream
2021-09-02 21:31:33 -07:00
import java.io.PrintStream
import java.security.SecureRandom
import java.util.*
import javax.crypto.Cipher
import javax.crypto.CipherOutputStream
import javax.crypto.spec.IvParameterSpec
import javax.crypto.spec.SecretKeySpec
import kotlin.random.asKotlinRandom
2021-09-02 21:31:33 -07:00
// Set non-zero value here to fix the random seed for reproducible builds
2022-06-20 05:34:13 +08:00
// CI builds are always reproducible
val RAND_SEED = if (System.getenv("CI") != null) 42 else 0
2021-09-02 21:31:33 -07:00
private lateinit var RANDOM: Random
private val kRANDOM get() = RANDOM.asKotlinRandom()
2021-09-02 21:31:33 -07:00
private val c1 = mutableListOf<String>()
private val c2 = mutableListOf<String>()
private val c3 = mutableListOf<String>()
fun initRandom(dict: File) {
RANDOM = if (RAND_SEED != 0) Random(RAND_SEED.toLong()) else SecureRandom()
c1.clear()
c2.clear()
c3.clear()
for (a in chain('a'..'z', 'A'..'Z')) {
if (a != 'a' && a != 'A') {
c1.add("$a")
}
for (b in chain('a'..'z', 'A'..'Z', '0'..'9')) {
c2.add("$a$b")
for (c in chain('a'..'z', 'A'..'Z', '0'..'9')) {
c3.add("$a$b$c")
}
}
}
c1.shuffle(RANDOM)
c2.shuffle(RANDOM)
c3.shuffle(RANDOM)
PrintStream(dict).use {
for (c in chain(c1, c2, c3)) {
it.println(c)
}
}
}
private fun <T> chain(vararg iters: Iterable<T>) = sequence {
iters.forEach { it.forEach { v -> yield(v) } }
}
private fun PrintStream.byteField(name: String, bytes: ByteArray) {
println("public static byte[] $name() {")
print("byte[] buf = {")
2021-12-14 21:30:15 +08:00
print(bytes.joinToString(",") { it.toString() })
2021-09-02 21:31:33 -07:00
println("};")
println("return buf;")
println("}")
}
fun genKeyData(keysDir: File, outSrc: File) {
outSrc.parentFile.mkdirs()
PrintStream(outSrc).use {
it.println("package com.topjohnwu.magisk.signing;")
it.println("public final class KeyData {")
it.byteField("verityCert", File(keysDir, "verity.x509.pem").readBytes())
it.byteField("verityKey", File(keysDir, "verity.pk8").readBytes())
it.println("}")
}
}
fun genStubManifest(srcDir: File, outDir: File): String {
outDir.deleteRecursively()
fun String.ind(level: Int) = replaceIndentByMargin(" ".repeat(level))
2022-02-13 06:22:42 -08:00
val cmpList = mutableListOf<String>()
cmpList.add(
2021-09-02 21:31:33 -07:00
"""
|<provider
| android:name="%s"
| android:authorities="${'$'}{applicationId}.provider"
| android:directBootAware="true"
| android:exported="false"
| android:grantUriPermissions="true" />""".ind(2)
2022-02-13 06:22:42 -08:00
)
2021-09-02 21:31:33 -07:00
2022-02-13 06:22:42 -08:00
cmpList.add(
2021-09-02 21:31:33 -07:00
"""
|<receiver
| android:name="%s"
| android:exported="false">
| <intent-filter>
| <action android:name="android.intent.action.LOCALE_CHANGED" />
| <action android:name="android.intent.action.UID_REMOVED" />
2022-02-13 19:54:59 -08:00
| <action android:name="android.intent.action.MY_PACKAGE_REPLACED" />
2021-09-02 21:31:33 -07:00
| </intent-filter>
| <intent-filter>
| <action android:name="android.intent.action.PACKAGE_REPLACED" />
| <action android:name="android.intent.action.PACKAGE_FULLY_REMOVED" />
|
| <data android:scheme="package" />
| </intent-filter>
|</receiver>""".ind(2)
2022-02-13 06:22:42 -08:00
)
2021-09-02 21:31:33 -07:00
2022-02-13 06:22:42 -08:00
cmpList.add(
2021-09-02 21:31:33 -07:00
"""
|<activity
| android:name="%s"
| android:exported="true">
| <intent-filter>
| <action android:name="android.intent.action.MAIN" />
| <category android:name="android.intent.category.LAUNCHER" />
| </intent-filter>
|</activity>""".ind(2)
2022-02-13 06:22:42 -08:00
)
2021-09-02 21:31:33 -07:00
2022-02-13 06:22:42 -08:00
cmpList.add(
2021-09-02 21:31:33 -07:00
"""
|<activity
| android:name="%s"
| android:directBootAware="true"
2022-06-06 05:26:53 -07:00
| android:exported="false"
2022-06-09 19:39:43 +08:00
| android:taskAffinity=""
2022-06-06 05:26:53 -07:00
| tools:ignore="AppLinkUrlError">
2021-09-02 21:31:33 -07:00
| <intent-filter>
| <action android:name="android.intent.action.VIEW"/>
| <category android:name="android.intent.category.DEFAULT"/>
| </intent-filter>
|</activity>""".ind(2)
2022-02-13 06:22:42 -08:00
)
2021-09-02 21:31:33 -07:00
2022-02-13 06:22:42 -08:00
cmpList.add(
2021-11-04 23:39:35 -07:00
"""
|<service
| android:name="%s"
2022-02-13 06:22:42 -08:00
| android:exported="false" />""".ind(2)
)
2021-09-02 21:31:33 -07:00
2022-02-13 06:22:42 -08:00
cmpList.add(
2021-09-02 21:31:33 -07:00
"""
|<service
| android:name="%s"
2021-11-04 23:39:35 -07:00
| android:exported="false"
2021-09-02 21:31:33 -07:00
| android:permission="android.permission.BIND_JOB_SERVICE" />""".ind(2)
2022-02-13 06:22:42 -08:00
)
2021-09-02 21:31:33 -07:00
val names = mutableListOf<String>()
names.addAll(c1)
names.addAll(c2.subList(0, 10))
names.addAll(c3.subList(0, 10))
names.shuffle(RANDOM)
2022-02-13 06:22:42 -08:00
val pkgNames = names
2022-02-06 07:12:26 -08:00
// Distinct by lower case to support case insensitive file systems
.distinctBy { it.toLowerCase(Locale.ROOT) }
// Old Android does not support capitalized package names
// Check Android 7.0.0 PackageParser#buildClassName
.map { it.decapitalize(Locale.ROOT) }
fun isJavaKeyword(name: String) = when (name) {
"do", "if", "for", "int", "new", "try" -> true
else -> false
}
2022-02-13 06:22:42 -08:00
val cmps = mutableListOf<String>()
val usedNames = mutableListOf<String>()
fun genCmpName(): String {
var pkgName: String
do {
2022-02-13 06:22:42 -08:00
pkgName = pkgNames.random(kRANDOM)
} while (isJavaKeyword(pkgName))
2022-02-13 06:22:42 -08:00
var clzName: String
do {
clzName = names.random(kRANDOM)
} while (isJavaKeyword(clzName))
2022-02-13 06:22:42 -08:00
val cmp = "${pkgName}.${clzName}"
usedNames.add(cmp)
return cmp
}
2021-09-02 21:31:33 -07:00
2022-02-13 06:22:42 -08:00
fun genClass(type: String) {
val clzName = genCmpName()
2021-09-02 21:31:33 -07:00
val (pkg, name) = clzName.split('.')
val pkgDir = File(outDir, pkg)
2022-02-13 14:23:06 -08:00
pkgDir.mkdirs()
PrintStream(File(pkgDir, "$name.java")).use {
2021-09-02 21:31:33 -07:00
it.println("package $pkg;")
it.println("public class $name extends com.topjohnwu.magisk.$type {}")
}
}
2022-02-13 06:22:42 -08:00
// Generate 2 non redirect-able classes
genClass("DelegateComponentFactory")
genClass("DelegateApplication")
2021-09-02 21:31:33 -07:00
for (gen in cmpList) {
val name = genCmpName()
2022-02-13 06:22:42 -08:00
cmps.add(gen.format(name))
2021-09-02 21:31:33 -07:00
}
// Shuffle the order of the components
cmps.shuffle(RANDOM)
2022-02-13 06:22:42 -08:00
val xml = File(srcDir, "AndroidManifest.xml").readText()
return xml.format(usedNames[0], usedNames[1], cmps.joinToString("\n\n"))
2021-09-02 21:31:33 -07:00
}
2021-12-14 21:30:15 +08:00
fun genEncryptedResources(res: InputStream, outDir: File) {
2021-09-02 21:31:33 -07:00
val mainPkgDir = File(outDir, "com/topjohnwu/magisk")
2022-02-13 14:23:06 -08:00
mainPkgDir.mkdirs()
2021-09-02 21:31:33 -07:00
// Generate iv and key
val iv = ByteArray(16)
val key = ByteArray(32)
RANDOM.nextBytes(iv)
RANDOM.nextBytes(key)
val cipher = Cipher.getInstance("AES/CBC/PKCS5Padding")
cipher.init(Cipher.ENCRYPT_MODE, SecretKeySpec(key, "AES"), IvParameterSpec(iv))
val bos = ByteArrayOutputStream()
2021-12-14 21:30:15 +08:00
res.use {
CipherOutputStream(bos, cipher).use { os ->
2021-09-02 21:31:33 -07:00
it.transferTo(os)
}
}
PrintStream(File(mainPkgDir, "Bytes.java")).use {
it.println("package com.topjohnwu.magisk;")
it.println("public final class Bytes {")
it.byteField("key", key)
it.byteField("iv", iv)
it.byteField("res", bos.toByteArray())
it.println("}")
}
}