Skip to content

Commit

Permalink
fix: added another enum restore pattern (#926)
Browse files Browse the repository at this point in the history
  • Loading branch information
skylot committed Oct 2, 2020
1 parent e2b4280 commit aa8fd3c
Show file tree
Hide file tree
Showing 3 changed files with 234 additions and 3 deletions.
17 changes: 14 additions & 3 deletions jadx-core/src/main/java/jadx/core/dex/visitors/EnumVisitor.java
Original file line number Diff line number Diff line change
Expand Up @@ -167,6 +167,8 @@ private boolean convertToEnum(ClassNode cls) {
InsnRemover.removeAllAndUnbind(classInitMth, staticBlock, toRemove);
if (classInitMth.countInsns() == 0) {
classInitMth.add(AFlag.DONT_GENERATE);
} else if (!toRemove.isEmpty()) {
CodeShrinkVisitor.shrinkMethod(classInitMth);
}
removeEnumMethods(cls, clsType, valuesField);
return true;
Expand Down Expand Up @@ -263,13 +265,14 @@ private List<EnumField> extractEnumFieldsFromFilledArray(ClassNode cls, InsnNode
InsnNode wrappedInsn = ((InsnWrapArg) arg).getWrapInsn();
field = processEnumFieldByField(cls, wrappedInsn, staticBlock, toRemove);
} else if (arg.isRegister()) {
field = processEnumFiledByRegister(cls, (RegisterArg) arg, toRemove);
field = processEnumFiledByRegister(cls, (RegisterArg) arg, staticBlock, toRemove);
}
if (field == null) {
return null;
}
enumFields.add(field);
}
toRemove.add(arrFillInsn);
return enumFields;
}

Expand All @@ -292,13 +295,21 @@ private EnumField processEnumFieldByField(ClassNode cls, InsnNode sgetInsn, Bloc
if (co == null) {
return null;
}
toRemove.add(sgetInsn);
RegisterArg sgetResult = sgetInsn.getResult();
if (sgetResult == null || sgetResult.getSVar().getUseCount() == 1) {
toRemove.add(sgetInsn);
}
toRemove.add(sputInsn);
return createEnumFieldByConstructor(cls, enumFieldNode, co);
}

@Nullable
private EnumField processEnumFiledByRegister(ClassNode cls, RegisterArg arg, List<InsnNode> toRemove) {
private EnumField processEnumFiledByRegister(ClassNode cls, RegisterArg arg, BlockNode staticBlock, List<InsnNode> toRemove) {
InsnNode assignInsn = arg.getAssignInsn();
if (assignInsn != null && assignInsn.getType() == InsnType.SGET) {
return processEnumFieldByField(cls, assignInsn, staticBlock, toRemove);
}

SSAVar ssaVar = arg.getSVar();
if (ssaVar.getUseCount() == 1) {
return null;
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
package jadx.tests.integration.enums;

import org.junit.jupiter.api.Test;

import jadx.tests.api.SmaliTest;

import static jadx.tests.api.utils.assertj.JadxAssertions.assertThat;

public class TestEnums8 extends SmaliTest {

@Test
public void test() {
assertThat(getClassNodeFromSmali())
.code()
.containsOne("enum TestEnums8");
}
}
203 changes: 203 additions & 0 deletions jadx-core/src/test/smali/enums/TestEnums8.smali
Original file line number Diff line number Diff line change
@@ -0,0 +1,203 @@
.class public final enum Lenums/TestEnums8;
.super Ljava/lang/Enum;
.source "SourceFile"


# annotations
.annotation system Ldalvik/annotation/Signature;
value = {
"Ljava/lang/Enum<",
"Lenums/TestEnums8;",
">;"
}
.end annotation


# static fields
.field private static final synthetic $VALUES:[Lenums/TestEnums8;

.field private static final FOR_BITS:[Lenums/TestEnums8;

.field public static final enum H:Lenums/TestEnums8;

.field public static final enum L:Lenums/TestEnums8;

.field public static final enum M:Lenums/TestEnums8;

.field public static final enum Q:Lenums/TestEnums8;


# instance fields
.field private final bits:I


# direct methods
.method static constructor <clinit>()V
.locals 10

.line 28
new-instance v0, Lenums/TestEnums8;

const/4 v1, 0x1

const/4 v2, 0x0

const-string v3, "L"

invoke-direct {v0, v3, v2, v1}, Lenums/TestEnums8;-><init>(Ljava/lang/String;II)V

sput-object v0, Lenums/TestEnums8;->L:Lenums/TestEnums8;

.line 30
new-instance v0, Lenums/TestEnums8;

const-string v3, "M"

invoke-direct {v0, v3, v1, v2}, Lenums/TestEnums8;-><init>(Ljava/lang/String;II)V

sput-object v0, Lenums/TestEnums8;->M:Lenums/TestEnums8;

.line 32
new-instance v0, Lenums/TestEnums8;

const/4 v3, 0x3

const/4 v4, 0x2

const-string v5, "Q"

invoke-direct {v0, v5, v4, v3}, Lenums/TestEnums8;-><init>(Ljava/lang/String;II)V

sput-object v0, Lenums/TestEnums8;->Q:Lenums/TestEnums8;

.line 34
new-instance v0, Lenums/TestEnums8;

const-string v5, "H"

invoke-direct {v0, v5, v3, v4}, Lenums/TestEnums8;-><init>(Ljava/lang/String;II)V

sput-object v0, Lenums/TestEnums8;->H:Lenums/TestEnums8;

const/4 v0, 0x4

new-array v5, v0, [Lenums/TestEnums8;

.line 25
sget-object v6, Lenums/TestEnums8;->L:Lenums/TestEnums8;

aput-object v6, v5, v2

sget-object v7, Lenums/TestEnums8;->M:Lenums/TestEnums8;

aput-object v7, v5, v1

sget-object v8, Lenums/TestEnums8;->Q:Lenums/TestEnums8;

aput-object v8, v5, v4

sget-object v9, Lenums/TestEnums8;->H:Lenums/TestEnums8;

aput-object v9, v5, v3

sput-object v5, Lenums/TestEnums8;->$VALUES:[Lenums/TestEnums8;

new-array v0, v0, [Lenums/TestEnums8;

aput-object v7, v0, v2

aput-object v6, v0, v1

aput-object v9, v0, v4

aput-object v8, v0, v3

.line 36
sput-object v0, Lenums/TestEnums8;->FOR_BITS:[Lenums/TestEnums8;

return-void
.end method

.method private constructor <init>(Ljava/lang/String;II)V
.locals 0
.annotation system Ldalvik/annotation/Signature;
value = {
"(I)V"
}
.end annotation

.line 40
invoke-direct {p0, p1, p2}, Ljava/lang/Enum;-><init>(Ljava/lang/String;I)V

.line 41
iput p3, p0, Lenums/TestEnums8;->bits:I

return-void
.end method

.method public static forBits(I)Lenums/TestEnums8;
.locals 2

if-ltz p0, :cond_0

.line 53
sget-object v0, Lenums/TestEnums8;->FOR_BITS:[Lenums/TestEnums8;

array-length v1, v0

if-ge p0, v1, :cond_0

.line 56
aget-object p0, v0, p0

return-object p0

.line 54
:cond_0
new-instance p0, Ljava/lang/IllegalArgumentException;

invoke-direct {p0}, Ljava/lang/IllegalArgumentException;-><init>()V

throw p0
.end method

.method public static valueOf(Ljava/lang/String;)Lenums/TestEnums8;
.locals 1

.line 25
const-class v0, Lenums/TestEnums8;

invoke-static {v0, p0}, Ljava/lang/Enum;->valueOf(Ljava/lang/Class;Ljava/lang/String;)Ljava/lang/Enum;

move-result-object p0

check-cast p0, Lenums/TestEnums8;

return-object p0
.end method

.method public static values()[Lenums/TestEnums8;
.locals 1

.line 25
sget-object v0, Lenums/TestEnums8;->$VALUES:[Lenums/TestEnums8;

invoke-virtual {v0}, [Lenums/TestEnums8;->clone()Ljava/lang/Object;

move-result-object v0

check-cast v0, [Lenums/TestEnums8;

return-object v0
.end method


# virtual methods
.method public getBits()I
.locals 1

.line 45
iget v0, p0, Lenums/TestEnums8;->bits:I

return v0
.end method

0 comments on commit aa8fd3c

Please sign in to comment.