mirror of
https://github.com/inotia00/revanced-patches.git
synced 2025-05-06 09:34:37 +02:00
feat(YouTube/Spoof format stream data): improve hook method, fetch to ANDROID_TESTSUITE
client
This commit is contained in:
parent
0f1473ab80
commit
05e3b9a0b1
@ -3,21 +3,27 @@ package app.revanced.patches.youtube.utils.fix.formatstream
|
|||||||
import app.revanced.patcher.data.BytecodeContext
|
import app.revanced.patcher.data.BytecodeContext
|
||||||
import app.revanced.patcher.extensions.InstructionExtensions.addInstruction
|
import app.revanced.patcher.extensions.InstructionExtensions.addInstruction
|
||||||
import app.revanced.patcher.extensions.InstructionExtensions.getInstruction
|
import app.revanced.patcher.extensions.InstructionExtensions.getInstruction
|
||||||
|
import app.revanced.patcher.extensions.InstructionExtensions.replaceInstruction
|
||||||
import app.revanced.patcher.patch.PatchException
|
import app.revanced.patcher.patch.PatchException
|
||||||
import app.revanced.patcher.util.proxy.mutableTypes.MutableMethod
|
|
||||||
import app.revanced.patches.youtube.utils.compatibility.Constants
|
import app.revanced.patches.youtube.utils.compatibility.Constants
|
||||||
import app.revanced.patches.youtube.utils.fix.formatstream.fingerprints.FormatStreamModelConstructorFingerprint
|
import app.revanced.patches.youtube.utils.fix.formatstream.fingerprints.FormatStreamModelConstructorFingerprint
|
||||||
import app.revanced.patches.youtube.utils.fix.formatstream.fingerprints.VideoStreamingDataConstructorFingerprint
|
import app.revanced.patches.youtube.utils.fix.formatstream.fingerprints.PlaybackStartFingerprint
|
||||||
import app.revanced.patches.youtube.utils.integrations.Constants.MISC_PATH
|
import app.revanced.patches.youtube.utils.integrations.Constants.MISC_PATH
|
||||||
import app.revanced.patches.youtube.utils.settings.SettingsPatch
|
import app.revanced.patches.youtube.utils.settings.SettingsPatch
|
||||||
import app.revanced.patches.youtube.video.information.VideoInformationPatch
|
import app.revanced.patches.youtube.video.information.VideoInformationPatch
|
||||||
import app.revanced.patches.youtube.video.playerresponse.PlayerResponseMethodHookPatch
|
import app.revanced.patches.youtube.video.playerresponse.PlayerResponseMethodHookPatch
|
||||||
import app.revanced.patches.youtube.video.videoid.VideoIdPatch
|
import app.revanced.patches.youtube.video.videoid.VideoIdPatch
|
||||||
import app.revanced.util.addFieldAndInstructions
|
import app.revanced.util.getReference
|
||||||
|
import app.revanced.util.getStringInstructionIndex
|
||||||
|
import app.revanced.util.indexOfFirstInstruction
|
||||||
import app.revanced.util.patch.BaseBytecodePatch
|
import app.revanced.util.patch.BaseBytecodePatch
|
||||||
import app.revanced.util.resultOrThrow
|
import app.revanced.util.resultOrThrow
|
||||||
|
import com.android.tools.smali.dexlib2.Opcode
|
||||||
|
import com.android.tools.smali.dexlib2.iface.instruction.OneRegisterInstruction
|
||||||
import com.android.tools.smali.dexlib2.iface.instruction.ReferenceInstruction
|
import com.android.tools.smali.dexlib2.iface.instruction.ReferenceInstruction
|
||||||
|
import com.android.tools.smali.dexlib2.iface.instruction.TwoRegisterInstruction
|
||||||
import com.android.tools.smali.dexlib2.iface.reference.FieldReference
|
import com.android.tools.smali.dexlib2.iface.reference.FieldReference
|
||||||
|
import com.android.tools.smali.dexlib2.iface.reference.MethodReference
|
||||||
|
|
||||||
@Suppress("unused")
|
@Suppress("unused")
|
||||||
object SpoofFormatStreamDataPatch : BaseBytecodePatch(
|
object SpoofFormatStreamDataPatch : BaseBytecodePatch(
|
||||||
@ -32,71 +38,95 @@ object SpoofFormatStreamDataPatch : BaseBytecodePatch(
|
|||||||
compatiblePackages = Constants.COMPATIBLE_PACKAGE,
|
compatiblePackages = Constants.COMPATIBLE_PACKAGE,
|
||||||
fingerprints = setOf(
|
fingerprints = setOf(
|
||||||
FormatStreamModelConstructorFingerprint,
|
FormatStreamModelConstructorFingerprint,
|
||||||
VideoStreamingDataConstructorFingerprint
|
PlaybackStartFingerprint
|
||||||
)
|
)
|
||||||
) {
|
) {
|
||||||
private const val INTEGRATIONS_CLASS_DESCRIPTOR =
|
private const val INTEGRATIONS_CLASS_DESCRIPTOR =
|
||||||
"$MISC_PATH/SpoofFormatStreamDataPatch;"
|
"$MISC_PATH/SpoofFormatStreamDataPatch;"
|
||||||
|
|
||||||
|
private const val INTEGRATIONS_METHOD_DESCRIPTOR =
|
||||||
|
"hookStreamData"
|
||||||
|
|
||||||
|
private const val INTEGRATIONS_METHOD_CALL =
|
||||||
|
INTEGRATIONS_CLASS_DESCRIPTOR +
|
||||||
|
"->" +
|
||||||
|
INTEGRATIONS_METHOD_DESCRIPTOR +
|
||||||
|
"(Ljava/lang/Object;)V"
|
||||||
|
|
||||||
|
private const val STREAMING_DATA_OUTER_CLASS =
|
||||||
|
"Lcom/google/protos/youtube/api/innertube/StreamingDataOuterClass\$StreamingData;"
|
||||||
|
|
||||||
override fun execute(context: BytecodeContext) {
|
override fun execute(context: BytecodeContext) {
|
||||||
|
|
||||||
// hook player response video id, to start loading format stream data sooner in the background.
|
// Hook player response video id, to start loading format stream data sooner in the background.
|
||||||
VideoIdPatch.hookPlayerResponseVideoId("$INTEGRATIONS_CLASS_DESCRIPTOR->newPlayerResponseVideoId(Ljava/lang/String;Z)V")
|
VideoIdPatch.hookPlayerResponseVideoId("$INTEGRATIONS_CLASS_DESCRIPTOR->newPlayerResponseVideoId(Ljava/lang/String;Z)V")
|
||||||
|
|
||||||
// TODO: Check if all instructions need to be spoofed.
|
|
||||||
FormatStreamModelConstructorFingerprint.resultOrThrow().let {
|
FormatStreamModelConstructorFingerprint.resultOrThrow().let {
|
||||||
it.mutableMethod.apply {
|
it.mutableMethod.apply {
|
||||||
val formatStreamDataIndex = it.scanResult.patternScanResult!!.startIndex
|
|
||||||
val formatStreamDataReference = getInstruction<ReferenceInstruction>(formatStreamDataIndex).reference
|
|
||||||
val formatStreamDataClass = context.findClass((formatStreamDataReference as FieldReference).definingClass)!!.mutableClass
|
|
||||||
|
|
||||||
formatStreamDataClass.methods.find { method -> method.name == "<init>" }
|
// Find the field name that will be used for reflection.
|
||||||
|
val streamDataIndex = it.scanResult.patternScanResult!!.startIndex
|
||||||
|
val streamDataReference = getInstruction<ReferenceInstruction>(streamDataIndex).reference
|
||||||
|
val streamDataFieldName = (streamDataReference as FieldReference).name
|
||||||
|
|
||||||
|
// Found field name is reflected in the integration.
|
||||||
|
context.findClass(INTEGRATIONS_CLASS_DESCRIPTOR)!!
|
||||||
|
.mutableClass.methods.find { method -> method.name == INTEGRATIONS_METHOD_DESCRIPTOR }
|
||||||
?.apply {
|
?.apply {
|
||||||
val getSmaliInstructions =
|
val index = getStringInstructionIndex("replaceMeWithFieldName")
|
||||||
"""
|
val register = getInstruction<OneRegisterInstruction>(index).registerA
|
||||||
if-eqz v0, :ignore
|
|
||||||
iget-object v0, v0, $formatStreamDataReference
|
|
||||||
if-eqz v0, :ignore
|
|
||||||
return-object v0
|
|
||||||
:ignore
|
|
||||||
const-string v0, ""
|
|
||||||
return-object v0
|
|
||||||
"""
|
|
||||||
val setSmaliInstructions =
|
|
||||||
"""
|
|
||||||
if-eqz p0, :ignore
|
|
||||||
if-eqz v0, :ignore
|
|
||||||
iput-object p0, v0, $formatStreamDataReference
|
|
||||||
:ignore
|
|
||||||
return-void
|
|
||||||
"""
|
|
||||||
|
|
||||||
val integrationMutableClass =
|
replaceInstruction(
|
||||||
context.findClass(INTEGRATIONS_CLASS_DESCRIPTOR)!!.mutableClass
|
index,
|
||||||
|
"const-string v$register, \"$streamDataFieldName\""
|
||||||
integrationMutableClass.addFieldAndInstructions(
|
|
||||||
context,
|
|
||||||
"getFormatStreamData",
|
|
||||||
"formatStreamDataClass",
|
|
||||||
definingClass,
|
|
||||||
getSmaliInstructions,
|
|
||||||
true
|
|
||||||
)
|
)
|
||||||
integrationMutableClass.addFieldAndInstructions(
|
} ?: throw PatchException("SpoofFormatStreamDataPatch not found")
|
||||||
context,
|
|
||||||
"setFormatStreamData",
|
|
||||||
"formatStreamDataClass",
|
|
||||||
definingClass,
|
|
||||||
setSmaliInstructions,
|
|
||||||
true
|
|
||||||
)
|
|
||||||
} ?: throw PatchException("FormatStreamDataClass not found!")
|
|
||||||
|
|
||||||
hook()
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
VideoStreamingDataConstructorFingerprint.resultOrThrow().mutableMethod.hook()
|
PlaybackStartFingerprint.resultOrThrow().mutableMethod.apply {
|
||||||
|
|
||||||
|
// Type of object being invoked is protobufList.
|
||||||
|
// Find the class name of protobufList.
|
||||||
|
val protobufListIndex = indexOfFirstInstruction {
|
||||||
|
opcode == Opcode.INVOKE_STATIC
|
||||||
|
&& getReference<MethodReference>()?.definingClass == STREAMING_DATA_OUTER_CLASS
|
||||||
|
&& getReference<MethodReference>()?.name == "emptyProtobufList"
|
||||||
|
}
|
||||||
|
if (protobufListIndex <= 0)
|
||||||
|
throw PatchException("ProtobufList index not found")
|
||||||
|
|
||||||
|
val protobufListReference = getInstruction<ReferenceInstruction>(protobufListIndex).reference
|
||||||
|
val protobufListClass = (protobufListReference as MethodReference).returnType
|
||||||
|
|
||||||
|
// Hooks all instructions that load or save protobufList.
|
||||||
|
for (index in implementation!!.instructions.size - 1 downTo 0) {
|
||||||
|
val instruction = getInstruction(index)
|
||||||
|
|
||||||
|
if (instruction.opcode != Opcode.IGET_OBJECT
|
||||||
|
&& instruction.opcode != Opcode.IPUT_OBJECT)
|
||||||
|
continue
|
||||||
|
|
||||||
|
val fieldReference = instruction.getReference<FieldReference>()
|
||||||
|
if (fieldReference?.definingClass != STREAMING_DATA_OUTER_CLASS)
|
||||||
|
continue
|
||||||
|
if (fieldReference.type != protobufListClass)
|
||||||
|
continue
|
||||||
|
|
||||||
|
val insertRegister = getInstruction<TwoRegisterInstruction>(index).registerA
|
||||||
|
val insertIndex =
|
||||||
|
if (instruction.opcode == Opcode.IPUT_OBJECT)
|
||||||
|
index
|
||||||
|
else
|
||||||
|
index + 1
|
||||||
|
|
||||||
|
addInstruction(
|
||||||
|
insertIndex,
|
||||||
|
"invoke-static { v$insertRegister }, " +
|
||||||
|
INTEGRATIONS_METHOD_CALL
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Add settings
|
* Add settings
|
||||||
@ -110,10 +140,4 @@ object SpoofFormatStreamDataPatch : BaseBytecodePatch(
|
|||||||
|
|
||||||
SettingsPatch.updatePatchStatus(this)
|
SettingsPatch.updatePatchStatus(this)
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun MutableMethod.hook() =
|
|
||||||
addInstruction(
|
|
||||||
1,
|
|
||||||
"invoke-static { }, $INTEGRATIONS_CLASS_DESCRIPTOR->hookFormatStreamData()V"
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,13 @@
|
|||||||
|
package app.revanced.patches.youtube.utils.fix.formatstream.fingerprints
|
||||||
|
|
||||||
|
import app.revanced.patcher.extensions.or
|
||||||
|
import app.revanced.patcher.fingerprint.MethodFingerprint
|
||||||
|
import com.android.tools.smali.dexlib2.AccessFlags
|
||||||
|
|
||||||
|
internal object PlaybackStartFingerprint : MethodFingerprint(
|
||||||
|
returnType = "Lcom/google/android/libraries/youtube/innertube/model/media/VideoStreamingData;",
|
||||||
|
accessFlags = AccessFlags.PUBLIC or AccessFlags.FINAL,
|
||||||
|
parameters = emptyList(),
|
||||||
|
strings = listOf("Invalid playback type; streaming data is not playable"),
|
||||||
|
)
|
||||||
|
|
@ -1,14 +0,0 @@
|
|||||||
package app.revanced.patches.youtube.utils.fix.formatstream.fingerprints
|
|
||||||
|
|
||||||
import app.revanced.patcher.extensions.or
|
|
||||||
import app.revanced.patcher.fingerprint.MethodFingerprint
|
|
||||||
import com.android.tools.smali.dexlib2.AccessFlags
|
|
||||||
|
|
||||||
internal object VideoStreamingDataConstructorFingerprint : MethodFingerprint(
|
|
||||||
accessFlags = AccessFlags.PUBLIC or AccessFlags.CONSTRUCTOR,
|
|
||||||
returnType = "V",
|
|
||||||
customFingerprint = { _, classDef ->
|
|
||||||
classDef.type == "Lcom/google/android/libraries/youtube/innertube/model/media/VideoStreamingData;"
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
Loading…
x
Reference in New Issue
Block a user