教你用 Java 字节码做点有趣的事 ——无痛破解 Java 系软件

在软件开发领域,Java 是一种非常流行的编程语言,由于其跨平台的特性和简洁的语法,它被广泛应用于各种应用程序和系统开发中。然而,Java 系软件的一大特点是其安全性,为了保护软件的知识产权和防止盗版行为,开发者常常对软件进行加密和保护。本篇文章将教你如何利用 Java 字节码技术,来进行无痛破解 Java 系软件,带你领略 Java 字节码的神奇之处!

本文会用到之前讲过的javaagent, 字节码修改框架ASM.不清楚的童鞋可以看下字节码合集里面前面的文章.

项目准备

假如我们有一个swing写了一个图形化程序demo.jar, 注意我们是没有源代码的. 使用java -jar运行这个demo.jar程序会先去校验license是否到期, 到期的话则会弹窗提示:

jar 包本质上就是一个 zip 压缩包,用 unzip 命令将 jar 包解压到一个临时文件夹 tmp 中,对应的目录结构如下所示:

.
├── META-INF
│   ├── MANIFEST.MF
│   └── maven
│       └── LicenseCheckSwing
│           └── LicenseCheckSwing
│               ├── pom.properties
│               └── pom.xml
└── me
    └── ya
        └── swing
            ├── AppMain.class
            └── StartupChecks.class

借助JD-GUI等反编译工具去查看StartupChecks.class反编译后的源码可以看到,这里判断 license 是否过期的方法比较简单,是拿当前时间与过期时间做对比,如果当前时间大于过期时间,就返回 license 已过期

public class StartupChecks {
  private static int getDayOfMonth() {
    return 7;
  }
  
  private static int getMonthOfYear() {
    return 0;
  }
  
  private static int getYear() {
    return 2019;
  }
  
  public static boolean canLoad() {
    validateLicensing();
    GregorianCalendar currentDate = new GregorianCalendar();
    GregorianCalendar expiryDate = getExiryDate();
    if (currentDate.after(expiryDate)) {
      return false;
    }
    return true;
  }
  
  private static void validateLicensing() {}
  
  public static GregorianCalendar getExiryDate() {
    return new GregorianCalendar(getYear(), getMonthOfYear(), getDayOfMonth());
  }
}

无痛破解

有了之前的字节码知识可以知道, 我们只需要将验证license是否过期的方法通过字节码修改的框架进行修改, 在canLoad()方法里面插入"return true;"让这个方法始终返回true即可.

public static boolean canLoad() {
    // 在这里强行插入 return true;
    return true;
    // 下面的语句不会执行到
    validateLicensing();
    GregorianCalendar currentDate = new GregorianCalendar();
    GregorianCalendar expiryDate = getExiryDate();
    if (currentDate.after(expiryDate)) {
        return false;
    }
    return true;
}

"return true;" 语句对应的字节码语句如下所示。

ICONST_1
IRETURN

下面我们使用ASM字节码改写框架来看具体的代码.

  1. 首先实现一个自定义的 MethodVisitor,在方法开始处插入 "return true;" 逻辑,代码如下所示。
public static class MyMethodVisitor extends AdviceAdapter {
    @Override
    protected void onMethodEnter() {
        // 强行插入 return true;
        mv.visitInsn(ICONST_1);
        mv.visitInsn(IRETURN);
    }
}

  1. 接下来实现一个自定义的 ClassVisitor,只处理 canLoad 方法,代码如下所示。
public static class MyClassVisitor extends ClassVisitor {
    @Override
    public MethodVisitor visitMethod(int access, String name, String desc, String signature, String[] exceptions) {
        MethodVisitor mv = super.visitMethod(access, name, desc, signature, exceptions);
        // 只注入 canLoad 方法
        if (name.equals("canLoad")) {
            return new MyMethodVisitor(mv, access, name, desc);
        }
        return mv;
    }
}

  1. 随后实现一个自定义的 ClassFileTransformer,在 transform 方法中进行字节码改写,代码如下所示。
public static class MyClassFileTransformer implements ClassFileTransformer {
    @Override
    public byte[] transform(ClassLoader loader, String className, Class<?> classBeingRedefined, ProtectionDomain protectionDomain, byte[] classBytes) throws IllegalClassFormatException {
        // 只注入 StartupChecks 类
        if (className.equals("me/ya/swing/StartupChecks")) {
            ClassReader cr = new ClassReader(classBytes);
            ClassWriter cw = new ClassWriter(cr, ClassWriter.COMPUTE_FRAMES);
            ClassVisitor cv = new MyClassVisitor(cw);
            cr.accept(cv, ClassReader.SKIP_FRAMES | ClassReader.SKIP_DEBUG);
            return cw.toByteArray();
        }
        return classBytes;
    }
}

  1. 执行 mvn clean package 编译生成 my-crack-agent.jar,执行 java -javaagent:/path/to/my-crack-agent.jar -jar crack-demo.jar. 此时发现已经成功地绕过了过期检查,弹出了 license 合法的提示框


改写后的字节码如下所示。

public static boolean canLoad();
Code:
  stack=1, locals=2, args_size=0
     0: iconst_1
     1: ireturn
     2: nop
     3: nop
     4: nop
     ...

可以看到经过改写以后 canLoad 在字节码开始处插入了 "return true;",旧指令被替换为了无用的 nop 指令。

总结

这篇文章,我们讲解了如何通过 javaagent 和 ASM 的方式来破解软件,回顾一下重点:要通过反编译工具找到相关的 license 检查函数在哪里,然后通过 javaagent 的 premain 函数在类加载之前动态修改字节码,绕过 license 检查机制。
希望本文能够帮助到对 Java 字节码感兴趣的读者,并能够进一步拓展你对 Java 系软件破解的知识和技能。Java 字节码是一种强大而神奇的编程工具,通过深入理解和应用,我们可以实现更多有趣的事情!

本文由mdnice多平台发布

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,270评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,489评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,630评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,906评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,928评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,718评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,442评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,345评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,802评论 1 317
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,984评论 3 337
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,117评论 1 351
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,810评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,462评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,011评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,139评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,377评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,060评论 2 355

推荐阅读更多精彩内容