1
mirror of https://github.com/revanced/revanced-patcher synced 2024-11-20 16:19:21 +01:00

fix: Merge extension only when patch executes (#315)

This commit is contained in:
oSumAtrIX 2024-10-27 16:00:30 +01:00 committed by GitHub
parent ab624f04f6
commit aa472eb985
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
4 changed files with 36 additions and 37 deletions

View File

@ -91,19 +91,15 @@ class Patcher(private val config: PatcherConfig) : Closeable {
}.also { executedPatches[this] = it }
}
// Prevent from decoding the app manifest twice if it is not needed.
// Prevent decoding the app manifest twice if it is not needed.
if (config.resourceMode != ResourcePatchContext.ResourceMode.NONE) {
context.resourceContext.decodeResources(config.resourceMode)
}
logger.info("Merging extensions")
logger.info("Initializing lookup maps")
with(context.bytecodeContext) {
context.executablePatches.mergeExtensions()
// Initialize lookup maps.
lookupMaps
}
// Accessing the lazy lookup maps to initialize them.
context.bytecodeContext.lookupMaps
logger.info("Executing patches")

View File

@ -59,42 +59,33 @@ class BytecodePatchContext internal constructor(private val config: PatcherConfi
internal val lookupMaps by lazy { LookupMaps(classes) }
/**
* Merge the extensions for this set of patches.
* Merge the extension of this patch.
*/
internal fun Set<Patch<*>>.mergeExtensions() {
// Lookup map to check if a class exists by its type quickly.
val classesByType = mutableMapOf<String, ClassDef>().apply {
classes.forEach { classDef -> put(classDef.type, classDef) }
}
internal fun BytecodePatch.mergeExtension() {
extension?.use { extensionStream ->
RawDexIO.readRawDexFile(extensionStream, 0, null).classes.forEach { classDef ->
val existingClass = lookupMaps.classesByType[classDef.type] ?: run {
logger.fine("Adding class \"$classDef\"")
forEachRecursively { patch ->
if (patch !is BytecodePatch) return@forEachRecursively
classes += classDef
lookupMaps.classesByType[classDef.type] = classDef
patch.extension?.use { extensionStream ->
RawDexIO.readRawDexFile(extensionStream, 0, null).classes.forEach { classDef ->
val existingClass = classesByType[classDef.type] ?: run {
logger.fine("Adding class \"$classDef\"")
return@forEach
}
classes += classDef
classesByType[classDef.type] = classDef
logger.fine("Class \"$classDef\" exists already. Adding missing methods and fields.")
return@forEach
existingClass.merge(classDef, this@BytecodePatchContext).let { mergedClass ->
// If the class was merged, replace the original class with the merged class.
if (mergedClass === existingClass) {
return@let
}
logger.fine("Class \"$classDef\" exists already. Adding missing methods and fields.")
existingClass.merge(classDef, this@BytecodePatchContext).let { mergedClass ->
// If the class was merged, replace the original class with the merged class.
if (mergedClass === existingClass) {
return@let
}
classes -= existingClass
classes += mergedClass
}
classes -= existingClass
classes += mergedClass
}
}
}
} ?: return logger.fine("No extension to merge")
}
/**
@ -185,6 +176,11 @@ class BytecodePatchContext internal constructor(private val config: PatcherConfi
*/
internal val methodsByStrings = MethodClassPairsLookupMap()
// Lookup map for fast checking if a class exists by its type.
val classesByType = mutableMapOf<String, ClassDef>().apply {
classes.forEach { classDef -> put(classDef.type, classDef) }
}
init {
classes.forEach { classDef ->
classDef.methods.forEach { method ->
@ -231,6 +227,7 @@ class BytecodePatchContext internal constructor(private val config: PatcherConfi
override fun close() {
methodsByStrings.clear()
classesByType.clear()
}
}

View File

@ -158,7 +158,13 @@ class BytecodePatch internal constructor(
finalizeBlock,
) {
override fun execute(context: PatcherContext) = with(context.bytecodeContext) {
fingerprints.forEach { it.match(this) }
with(context.bytecodeContext) {
mergeExtension()
}
fingerprints.forEach {
it.match(this)
}
execute(this)
}

View File

@ -195,7 +195,7 @@ internal object PatcherTest {
private operator fun Set<Patch<*>>.invoke(): List<PatchResult> {
every { patcher.context.executablePatches } returns toMutableSet()
every { patcher.context.bytecodeContext.lookupMaps } returns LookupMaps(patcher.context.bytecodeContext.classes)
every { with(patcher.context.bytecodeContext) { any<Set<Patch<*>>>().mergeExtensions() } } just runs
every { with(patcher.context.bytecodeContext) { any<BytecodePatch>().mergeExtension() } } just runs
return runBlocking { patcher().toList() }
}