import java.io.ByteArrayOutputStream import java.io.File import java.io.InputStream 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 // Set non-zero value here to fix the random seed for reproducible builds // CI builds are always reproducible val RAND_SEED = if (System.getenv("CI") != null) 42 else 0 private lateinit var RANDOM: Random private val kRANDOM get() = RANDOM.asKotlinRandom() private val c1 = mutableListOf() private val c2 = mutableListOf() private val c3 = mutableListOf() 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 chain(vararg iters: Iterable) = sequence { iters.forEach { it.forEach { v -> yield(v) } } } private fun PrintStream.byteField(name: String, bytes: ByteArray) { println("public static byte[] $name() {") print("byte[] buf = {") print(bytes.joinToString(",") { it.toString() }) 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)) val cmpList = mutableListOf() cmpList.add( """ |""".ind(2) ) cmpList.add( """ | | | | | | | | | | | | |""".ind(2) ) cmpList.add( """ | | | | | |""".ind(2) ) cmpList.add( """ | | | | | |""".ind(2) ) cmpList.add( """ |""".ind(2) ) cmpList.add( """ |""".ind(2) ) val names = mutableListOf() names.addAll(c1) names.addAll(c2.subList(0, 10)) names.addAll(c3.subList(0, 10)) names.shuffle(RANDOM) val pkgNames = names // 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 } val cmps = mutableListOf() val usedNames = mutableListOf() fun genCmpName(): String { var pkgName: String do { pkgName = pkgNames.random(kRANDOM) } while (isJavaKeyword(pkgName)) var clzName: String do { clzName = names.random(kRANDOM) } while (isJavaKeyword(clzName)) val cmp = "${pkgName}.${clzName}" usedNames.add(cmp) return cmp } fun genClass(type: String) { val clzName = genCmpName() val (pkg, name) = clzName.split('.') val pkgDir = File(outDir, pkg) pkgDir.mkdirs() PrintStream(File(pkgDir, "$name.java")).use { it.println("package $pkg;") it.println("public class $name extends com.topjohnwu.magisk.$type {}") } } // Generate 2 non redirect-able classes genClass("DelegateComponentFactory") genClass("DelegateApplication") for (gen in cmpList) { val name = genCmpName() cmps.add(gen.format(name)) } // Shuffle the order of the components cmps.shuffle(RANDOM) val xml = File(srcDir, "AndroidManifest.xml").readText() return xml.format(usedNames[0], usedNames[1], cmps.joinToString("\n\n")) } fun genEncryptedResources(res: InputStream, outDir: File) { val mainPkgDir = File(outDir, "com/topjohnwu/magisk") mainPkgDir.mkdirs() // 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() res.use { CipherOutputStream(bos, cipher).use { os -> 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("}") } }