mirror of
https://github.com/revanced/smali.git
synced 2025-05-09 10:54:29 +02:00
Track register narrowing via instance-of after a move
This commit is contained in:
parent
ca48e6f7d0
commit
db49ae1d03
@ -1208,8 +1208,27 @@ public class MethodAnalyzer {
|
|||||||
AnalyzedInstruction prevAnalyzedInstruction = analyzedInstructions.valueAt(instructionIndex - 1);
|
AnalyzedInstruction prevAnalyzedInstruction = analyzedInstructions.valueAt(instructionIndex - 1);
|
||||||
if (prevAnalyzedInstruction.instruction.getOpcode() == Opcode.INSTANCE_OF) {
|
if (prevAnalyzedInstruction.instruction.getOpcode() == Opcode.INSTANCE_OF) {
|
||||||
if (canNarrowAfterInstanceOf(prevAnalyzedInstruction, analyzedInstruction, classPath)) {
|
if (canNarrowAfterInstanceOf(prevAnalyzedInstruction, analyzedInstruction, classPath)) {
|
||||||
|
List<Integer> narrowingRegisters = Lists.newArrayList();
|
||||||
|
|
||||||
|
if (instructionIndex > 1) {
|
||||||
|
// If we have something like:
|
||||||
|
// move-object/from16 v0, p1
|
||||||
|
// instance-of v2, v0, Lblah;
|
||||||
|
// if-eqz v2, :target
|
||||||
|
// Then we need to narrow both v0 AND p1
|
||||||
|
AnalyzedInstruction prevPrevAnalyzedInstruction =
|
||||||
|
analyzedInstructions.valueAt(instructionIndex - 2);
|
||||||
|
Opcode opcode = prevPrevAnalyzedInstruction.instruction.getOpcode();
|
||||||
|
if (opcode == Opcode.MOVE_OBJECT || opcode == Opcode.MOVE_OBJECT_16 ||
|
||||||
|
opcode == Opcode.MOVE_OBJECT_FROM16) {
|
||||||
|
narrowingRegisters.add(
|
||||||
|
((TwoRegisterInstruction)prevPrevAnalyzedInstruction.instruction).getRegisterB());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Propagate the original type to the failing branch, and the new type to the successful branch
|
// Propagate the original type to the failing branch, and the new type to the successful branch
|
||||||
int narrowingRegister = ((Instruction22c)prevAnalyzedInstruction.instruction).getRegisterB();
|
int narrowingRegister = ((Instruction22c)prevAnalyzedInstruction.instruction).getRegisterB();
|
||||||
|
narrowingRegisters.add(narrowingRegister);
|
||||||
RegisterType originalType = analyzedInstruction.getPreInstructionRegisterType(narrowingRegister);
|
RegisterType originalType = analyzedInstruction.getPreInstructionRegisterType(narrowingRegister);
|
||||||
RegisterType newType = RegisterType.getRegisterType(classPath,
|
RegisterType newType = RegisterType.getRegisterType(classPath,
|
||||||
(TypeReference)((Instruction22c)prevAnalyzedInstruction.instruction).getReference());
|
(TypeReference)((Instruction22c)prevAnalyzedInstruction.instruction).getReference());
|
||||||
@ -1221,16 +1240,18 @@ public class MethodAnalyzer {
|
|||||||
((Instruction21t)analyzedInstruction.instruction).getCodeOffset();
|
((Instruction21t)analyzedInstruction.instruction).getCodeOffset();
|
||||||
AnalyzedInstruction branchInstruction = analyzedInstructions.get(nextAddress);
|
AnalyzedInstruction branchInstruction = analyzedInstructions.get(nextAddress);
|
||||||
|
|
||||||
if (analyzedInstruction.instruction.getOpcode() == Opcode.IF_EQZ) {
|
for (int register: narrowingRegisters) {
|
||||||
overridePredecessorRegisterTypeAndPropagateChanges(fallthroughInstruction, analyzedInstruction,
|
if (analyzedInstruction.instruction.getOpcode() == Opcode.IF_EQZ) {
|
||||||
narrowingRegister, newType);
|
overridePredecessorRegisterTypeAndPropagateChanges(fallthroughInstruction, analyzedInstruction,
|
||||||
overridePredecessorRegisterTypeAndPropagateChanges(branchInstruction, analyzedInstruction,
|
register, newType);
|
||||||
narrowingRegister, originalType);
|
overridePredecessorRegisterTypeAndPropagateChanges(branchInstruction, analyzedInstruction,
|
||||||
} else {
|
register, originalType);
|
||||||
overridePredecessorRegisterTypeAndPropagateChanges(fallthroughInstruction, analyzedInstruction,
|
} else {
|
||||||
narrowingRegister, originalType);
|
overridePredecessorRegisterTypeAndPropagateChanges(fallthroughInstruction, analyzedInstruction,
|
||||||
overridePredecessorRegisterTypeAndPropagateChanges(branchInstruction, analyzedInstruction,
|
register, originalType);
|
||||||
narrowingRegister, newType);
|
overridePredecessorRegisterTypeAndPropagateChanges(branchInstruction, analyzedInstruction,
|
||||||
|
register, newType);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,156 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2016, Google Inc.
|
||||||
|
* All rights reserved.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are
|
||||||
|
* met:
|
||||||
|
*
|
||||||
|
* Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* Redistributions in binary form must reproduce the above
|
||||||
|
* copyright notice, this list of conditions and the following disclaimer
|
||||||
|
* in the documentation and/or other materials provided with the
|
||||||
|
* distribution.
|
||||||
|
* Neither the name of Google Inc. nor the names of its
|
||||||
|
* contributors may be used to endorse or promote products derived from
|
||||||
|
* this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||||
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||||
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||||
|
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||||
|
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||||
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||||
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||||
|
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||||
|
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||||
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.jf.dexlib2.analysis;
|
||||||
|
|
||||||
|
import org.jf.dexlib2.AccessFlags;
|
||||||
|
import org.jf.dexlib2.Opcode;
|
||||||
|
import org.jf.dexlib2.Opcodes;
|
||||||
|
import org.jf.dexlib2.builder.MethodImplementationBuilder;
|
||||||
|
import org.jf.dexlib2.builder.instruction.BuilderInstruction10x;
|
||||||
|
import org.jf.dexlib2.builder.instruction.BuilderInstruction12x;
|
||||||
|
import org.jf.dexlib2.builder.instruction.BuilderInstruction21t;
|
||||||
|
import org.jf.dexlib2.builder.instruction.BuilderInstruction22c;
|
||||||
|
import org.jf.dexlib2.iface.ClassDef;
|
||||||
|
import org.jf.dexlib2.iface.DexFile;
|
||||||
|
import org.jf.dexlib2.iface.Method;
|
||||||
|
import org.jf.dexlib2.iface.MethodImplementation;
|
||||||
|
import org.jf.dexlib2.immutable.ImmutableClassDef;
|
||||||
|
import org.jf.dexlib2.immutable.ImmutableDexFile;
|
||||||
|
import org.jf.dexlib2.immutable.ImmutableMethod;
|
||||||
|
import org.jf.dexlib2.immutable.ImmutableMethodParameter;
|
||||||
|
import org.jf.dexlib2.immutable.reference.ImmutableTypeReference;
|
||||||
|
import org.junit.Assert;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
public class MethodAnalyzerTest {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testInstanceOfNarrowingEqz() throws IOException {
|
||||||
|
MethodImplementationBuilder builder = new MethodImplementationBuilder(2);
|
||||||
|
|
||||||
|
builder.addInstruction(new BuilderInstruction22c(Opcode.INSTANCE_OF, 0, 1,
|
||||||
|
new ImmutableTypeReference("Lmain;")));
|
||||||
|
builder.addInstruction(new BuilderInstruction21t(Opcode.IF_EQZ, 0, builder.getLabel("not_instance_of")));
|
||||||
|
builder.addInstruction(new BuilderInstruction10x(Opcode.RETURN_VOID));
|
||||||
|
|
||||||
|
builder.addLabel("not_instance_of");
|
||||||
|
builder.addInstruction(new BuilderInstruction10x(Opcode.RETURN_VOID));
|
||||||
|
|
||||||
|
MethodImplementation methodImplementation = builder.getMethodImplementation();
|
||||||
|
|
||||||
|
Method method = new ImmutableMethod("Lmain;", "narrowing",
|
||||||
|
Collections.singletonList(new ImmutableMethodParameter("Ljava/lang/Object;", null, null)), "V",
|
||||||
|
AccessFlags.PUBLIC.getValue(), null, methodImplementation);
|
||||||
|
ClassDef classDef = new ImmutableClassDef("Lmain;", AccessFlags.PUBLIC.getValue(), "Ljava/lang/Object;", null,
|
||||||
|
null, null, null, Collections.singletonList(method));
|
||||||
|
DexFile dexFile = new ImmutableDexFile(Opcodes.forApi(15), Collections.singletonList(classDef));
|
||||||
|
|
||||||
|
ClassPath classPath = new ClassPath(new DexClassProvider(dexFile));
|
||||||
|
MethodAnalyzer methodAnalyzer = new MethodAnalyzer(classPath, method, null, false);
|
||||||
|
|
||||||
|
List<AnalyzedInstruction> analyzedInstructions = methodAnalyzer.getAnalyzedInstructions();
|
||||||
|
Assert.assertEquals("Lmain;", analyzedInstructions.get(2).getPreInstructionRegisterType(1).type.getType());
|
||||||
|
|
||||||
|
Assert.assertEquals("Ljava/lang/Object;",
|
||||||
|
analyzedInstructions.get(3).getPreInstructionRegisterType(1).type.getType());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testInstanceOfNarrowingNez() throws IOException {
|
||||||
|
MethodImplementationBuilder builder = new MethodImplementationBuilder(2);
|
||||||
|
|
||||||
|
builder.addInstruction(new BuilderInstruction22c(Opcode.INSTANCE_OF, 0, 1,
|
||||||
|
new ImmutableTypeReference("Lmain;")));
|
||||||
|
builder.addInstruction(new BuilderInstruction21t(Opcode.IF_NEZ, 0, builder.getLabel("instance_of")));
|
||||||
|
builder.addInstruction(new BuilderInstruction10x(Opcode.RETURN_VOID));
|
||||||
|
|
||||||
|
builder.addLabel("instance_of");
|
||||||
|
builder.addInstruction(new BuilderInstruction10x(Opcode.RETURN_VOID));
|
||||||
|
|
||||||
|
MethodImplementation methodImplementation = builder.getMethodImplementation();
|
||||||
|
|
||||||
|
Method method = new ImmutableMethod("Lmain;", "narrowing",
|
||||||
|
Collections.singletonList(new ImmutableMethodParameter("Ljava/lang/Object;", null, null)), "V",
|
||||||
|
AccessFlags.PUBLIC.getValue(), null, methodImplementation);
|
||||||
|
ClassDef classDef = new ImmutableClassDef("Lmain;", AccessFlags.PUBLIC.getValue(), "Ljava/lang/Object;", null,
|
||||||
|
null, null, null, Collections.singletonList(method));
|
||||||
|
DexFile dexFile = new ImmutableDexFile(Opcodes.forApi(15), Collections.singletonList(classDef));
|
||||||
|
|
||||||
|
ClassPath classPath = new ClassPath(new DexClassProvider(dexFile));
|
||||||
|
MethodAnalyzer methodAnalyzer = new MethodAnalyzer(classPath, method, null, false);
|
||||||
|
|
||||||
|
List<AnalyzedInstruction> analyzedInstructions = methodAnalyzer.getAnalyzedInstructions();
|
||||||
|
Assert.assertEquals("Ljava/lang/Object;",
|
||||||
|
analyzedInstructions.get(2).getPreInstructionRegisterType(1).type.getType());
|
||||||
|
|
||||||
|
Assert.assertEquals("Lmain;", analyzedInstructions.get(3).getPreInstructionRegisterType(1).type.getType());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testInstanceOfNarrowingAfterMove() throws IOException {
|
||||||
|
MethodImplementationBuilder builder = new MethodImplementationBuilder(3);
|
||||||
|
|
||||||
|
builder.addInstruction(new BuilderInstruction12x(Opcode.MOVE_OBJECT, 1, 2));
|
||||||
|
builder.addInstruction(new BuilderInstruction22c(Opcode.INSTANCE_OF, 0, 1,
|
||||||
|
new ImmutableTypeReference("Lmain;")));
|
||||||
|
builder.addInstruction(new BuilderInstruction21t(Opcode.IF_EQZ, 0, builder.getLabel("not_instance_of")));
|
||||||
|
builder.addInstruction(new BuilderInstruction10x(Opcode.RETURN_VOID));
|
||||||
|
|
||||||
|
builder.addLabel("not_instance_of");
|
||||||
|
builder.addInstruction(new BuilderInstruction10x(Opcode.RETURN_VOID));
|
||||||
|
|
||||||
|
MethodImplementation methodImplementation = builder.getMethodImplementation();
|
||||||
|
|
||||||
|
Method method = new ImmutableMethod("Lmain;", "narrowing",
|
||||||
|
Collections.singletonList(new ImmutableMethodParameter("Ljava/lang/Object;", null, null)), "V",
|
||||||
|
AccessFlags.PUBLIC.getValue(), null, methodImplementation);
|
||||||
|
ClassDef classDef = new ImmutableClassDef("Lmain;", AccessFlags.PUBLIC.getValue(), "Ljava/lang/Object;", null,
|
||||||
|
null, null, null, Collections.singletonList(method));
|
||||||
|
DexFile dexFile = new ImmutableDexFile(Opcodes.forApi(15), Collections.singletonList(classDef));
|
||||||
|
|
||||||
|
ClassPath classPath = new ClassPath(new DexClassProvider(dexFile));
|
||||||
|
MethodAnalyzer methodAnalyzer = new MethodAnalyzer(classPath, method, null, false);
|
||||||
|
|
||||||
|
List<AnalyzedInstruction> analyzedInstructions = methodAnalyzer.getAnalyzedInstructions();
|
||||||
|
Assert.assertEquals("Lmain;", analyzedInstructions.get(3).getPreInstructionRegisterType(1).type.getType());
|
||||||
|
Assert.assertEquals("Lmain;", analyzedInstructions.get(3).getPreInstructionRegisterType(2).type.getType());
|
||||||
|
|
||||||
|
Assert.assertEquals("Ljava/lang/Object;",
|
||||||
|
analyzedInstructions.get(4).getPreInstructionRegisterType(1).type.getType());
|
||||||
|
Assert.assertEquals("Ljava/lang/Object;",
|
||||||
|
analyzedInstructions.get(4).getPreInstructionRegisterType(2).type.getType());
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user