【手游】少年西游记 美术资源加密分析

2024-02-17 08:10

本文主要是介绍【手游】少年西游记 美术资源加密分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

有个网友让我帮忙分析下《少年西游记》这款手游,分析过程和大家分享一下


0x00 先用WinHex看看少年西游记资源包package.was

粗略的分析可以把PackageWas的结构看成为这样:

┌─PackageWas

┊        ├─文件头

┊        ├─文件1

┊        ├─     ┊

┊        ├─文件n


详细的解构分析如下:

was文件头 32字节

12字节文件标识
4字节未知
4字节未知
4字节未知
4字节子文件个数
4字节未知

子块文件头 32字节

4字节文件块偏移
4字节当前文件数据长度
4字节原始文件数据长度
4字节压缩标志 值=1为压缩
4字节加密标志 值=1为加密
4字节未知标志 值默认为0
4字节文件块长度
4字节文件名长度

子块文件数据

n字节文件名
1字节文件名结束标志
4字节原始文件数据长度
n字节文件数据

0x01 在IDA中反汇编\lib\armeabi\libcocos2dlua.so文件,来分析它是怎样读取资源包数据的





主要看下图片中用红框标注的地方,逻辑其实很简单,关键是在对子文件数据的处理上 先是判断有没有用xxTea加密 然后判断有没有用zlib压缩

关于xxTea的key的寻找,可在IDA中 搜索SetXXTeaKey函数,然后查看该函数的引用位置就能找到key,可参考我的另一片文章 http://blog.csdn.net/blueeffie/article/details/51208285


0x02 分析完资源包结构和文件的解析处理后 那就直接上代码 解包吧(C# 代码片段)

//读取Was资源包文件
private void ReadPackageFile(FileInfo f)
{UpdateText("开始解析资源包" + Path.GetFileName(f.FullName));//使用委托开启异步,防止解析数据时界面假死Func<bool> Analysis = new Func<bool>(() =>{return AnalysisPackageFile(f);});Analysis.BeginInvoke((isAnalysis) =>{this.BeginInvoke(new Action(() =>{if (Analysis.EndInvoke(isAnalysis)){UpdateText("开始导出资源包...");OutPackageFile(f);UpdateText("资源包导出完成!");}else{UpdateText("解析资源包失败!");}}), null);}, null);
}//分析Was资源包文件
private bool AnalysisPackageFile(FileInfo f)
{byte[] bytes;using (FileStream inStream = new FileStream(f.FullName, FileMode.Open, FileAccess.ReadWrite)){bytes = new byte[inStream.Length];inStream.Read(bytes, 0, bytes.Length);}if (bytes != null && PacketUtil.IsWasFile(bytes)){try{int fileDataLength = 0;for (int i = 32; i < bytes.Length; i += fileDataLength){int Deviation = BitConverter.ToInt32(bytes, i); //获取当前文件块偏移if (i == Deviation){fileDataLength = BitConverter.ToInt32(bytes, i + 24); //获取当前文件块长度byte[] fileByte = new byte[fileDataLength];Array.Copy(bytes, i, fileByte, 0, fileDataLength);WasData wasData = new WasData();FileData fileData = wasData.intData(fileByte); //解析文件块fileDataList.Add(fileData);}}return true;}catch (Exception e){UpdateText(e.ToString());return false;}}else{return false;}}//导出包文件
private void OutPackageFile(FileInfo f)
{for (int i = 0; i < fileDataList.Count; i++){string path = Path.Combine(f.DirectoryName, Path.GetFileNameWithoutExtension(f.FullName) + "/" + fileDataList[i].name);OutResFile(fileDataList[i].data, path);}fileDataList.Clear();
}//输出文件
private void OutResFile(byte[] bytes, string path)
{if (!Directory.Exists(Path.GetDirectoryName(path))){Directory.CreateDirectory(Path.GetDirectoryName(path));}using (FileStream outStream = new FileStream(path, FileMode.OpenOrCreate, FileAccess.ReadWrite)){outStream.Write(bytes, 0, bytes.Length);UpdateText("生成:" + path);}
}


public FileData intData(byte[] bytes)
{FileData fileData = new FileData();FileHead fileHead = (FileHead)PacketUtil.BytesToStuct(bytes, typeof(FileHead));fileData.name = GetFileName(bytes, fileHead);fileData.data = GetFileData(bytes, fileHead);return fileData;
}private string GetFileName(byte[] bytes, FileHead fileHead)
{byte[] nameByte = new byte[fileHead.FileNameLength];Array.Copy(bytes, Marshal.SizeOf(fileHead), nameByte, 0, nameByte.Length);return Encoding.Default.GetString(nameByte);
}private byte[] GetFileData(byte[] bytes, FileHead fileHead)
{byte[] dataByte = new byte[fileHead.CurrentFileLength - 4];Array.Copy(bytes, fileHead.FileDataLength - dataByte.Length, dataByte, 0, dataByte.Length);if (fileHead.EncryptFlag == 1) //需要解密{byte[] decryptByte = DecryptData(dataByte);if (fileHead.CompressFlag == 1) //需要解压{return ZlibStream.UncompressBuffer(decryptByte);}else{return decryptByte;}}else{if (fileHead.CompressFlag == 1) //需要解压{return ZlibStream.UncompressBuffer(dataByte);}else{return dataByte;}}
}private byte[] DecryptData(byte[] bytes)
{byte[] keyByte = Encoding.Default.GetBytes("27efb289-bc7f-3636-ae74-e747b1bea17c");byte[] decryptByte = XXTea.Decrypt(bytes, keyByte);return decryptByte;
}

zilb库 我用的 http://dotnetzip.codeplex.com




0x03 注意 .\package\spine 中的PNG图片,它本身并不是png图片而是JFIF图片,可以用XnView看图软件直接打开,或是把下图红框的内容删除后可以直接打开查看



资源提取工具源码下载:

链接:http://pan.baidu.com/s/1slEQ54p 密码:6dag

这篇关于【手游】少年西游记 美术资源加密分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/717182

相关文章

python实现对数据公钥加密与私钥解密

《python实现对数据公钥加密与私钥解密》这篇文章主要为大家详细介绍了如何使用python实现对数据公钥加密与私钥解密,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录公钥私钥的生成使用公钥加密使用私钥解密公钥私钥的生成这一部分,使用python生成公钥与私钥,然后保存在两个文

MySQL中的表连接原理分析

《MySQL中的表连接原理分析》:本文主要介绍MySQL中的表连接原理分析,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、背景2、环境3、表连接原理【1】驱动表和被驱动表【2】内连接【3】外连接【4编程】嵌套循环连接【5】join buffer4、总结1、背景

python中Hash使用场景分析

《python中Hash使用场景分析》Python的hash()函数用于获取对象哈希值,常用于字典和集合,不可变类型可哈希,可变类型不可,常见算法包括除法、乘法、平方取中和随机数哈希,各有优缺点,需根... 目录python中的 Hash除法哈希算法乘法哈希算法平方取中法随机数哈希算法小结在Python中,

Java Stream的distinct去重原理分析

《JavaStream的distinct去重原理分析》Javastream中的distinct方法用于去除流中的重复元素,它返回一个包含过滤后唯一元素的新流,该方法会根据元素的hashcode和eq... 目录一、distinct 的基础用法与核心特性二、distinct 的底层实现原理1. 顺序流中的去重

关于MyISAM和InnoDB对比分析

《关于MyISAM和InnoDB对比分析》:本文主要介绍关于MyISAM和InnoDB对比分析,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录开篇:从交通规则看存储引擎选择理解存储引擎的基本概念技术原理对比1. 事务支持:ACID的守护者2. 锁机制:并发控制的艺

C++中RAII资源获取即初始化

《C++中RAII资源获取即初始化》RAII通过构造/析构自动管理资源生命周期,确保安全释放,本文就来介绍一下C++中的RAII技术及其应用,具有一定的参考价值,感兴趣的可以了解一下... 目录一、核心原理与机制二、标准库中的RAII实现三、自定义RAII类设计原则四、常见应用场景1. 内存管理2. 文件操

MyBatis Plus 中 update_time 字段自动填充失效的原因分析及解决方案(最新整理)

《MyBatisPlus中update_time字段自动填充失效的原因分析及解决方案(最新整理)》在使用MyBatisPlus时,通常我们会在数据库表中设置create_time和update... 目录前言一、问题现象二、原因分析三、总结:常见原因与解决方法对照表四、推荐写法前言在使用 MyBATis

Python主动抛出异常的各种用法和场景分析

《Python主动抛出异常的各种用法和场景分析》在Python中,我们不仅可以捕获和处理异常,还可以主动抛出异常,也就是以类的方式自定义错误的类型和提示信息,这在编程中非常有用,下面我将详细解释主动抛... 目录一、为什么要主动抛出异常?二、基本语法:raise关键字基本示例三、raise的多种用法1. 抛

github打不开的问题分析及解决

《github打不开的问题分析及解决》:本文主要介绍github打不开的问题分析及解决,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、找到github.com域名解析的ip地址二、找到github.global.ssl.fastly.net网址解析的ip地址三

Mysql的主从同步/复制的原理分析

《Mysql的主从同步/复制的原理分析》:本文主要介绍Mysql的主从同步/复制的原理分析,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录为什么要主从同步?mysql主从同步架构有哪些?Mysql主从复制的原理/整体流程级联复制架构为什么好?Mysql主从复制注意