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

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

相关文章

Nginx分布式部署流程分析

《Nginx分布式部署流程分析》文章介绍Nginx在分布式部署中的反向代理和负载均衡作用,用于分发请求、减轻服务器压力及解决session共享问题,涵盖配置方法、策略及Java项目应用,并提及分布式事... 目录分布式部署NginxJava中的代理代理分为正向代理和反向代理正向代理反向代理Nginx应用场景

Redis中的有序集合zset从使用到原理分析

《Redis中的有序集合zset从使用到原理分析》Redis有序集合(zset)是字符串与分值的有序映射,通过跳跃表和哈希表结合实现高效有序性管理,适用于排行榜、延迟队列等场景,其时间复杂度低,内存占... 目录开篇:排行榜背后的秘密一、zset的基本使用1.1 常用命令1.2 Java客户端示例二、zse

Redis中的AOF原理及分析

《Redis中的AOF原理及分析》Redis的AOF通过记录所有写操作命令实现持久化,支持always/everysec/no三种同步策略,重写机制优化文件体积,与RDB结合可平衡数据安全与恢复效率... 目录开篇:从日记本到AOF一、AOF的基本执行流程1. 命令执行与记录2. AOF重写机制二、AOF的

MyBatis Plus大数据量查询慢原因分析及解决

《MyBatisPlus大数据量查询慢原因分析及解决》大数据量查询慢常因全表扫描、分页不当、索引缺失、内存占用高及ORM开销,优化措施包括分页查询、流式读取、SQL优化、批处理、多数据源、结果集二次... 目录大数据量查询慢的常见原因优化方案高级方案配置调优监控与诊断总结大数据量查询慢的常见原因MyBAT

分析 Java Stream 的 peek使用实践与副作用处理方案

《分析JavaStream的peek使用实践与副作用处理方案》StreamAPI的peek操作是中间操作,用于观察元素但不终止流,其副作用风险包括线程安全、顺序混乱及性能问题,合理使用场景有限... 目录一、peek 操作的本质:有状态的中间操作二、副作用的定义与风险场景1. 并行流下的线程安全问题2. 顺

MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决

《MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决》MyBatis默认开启一级缓存,同一事务中循环调用查询方法时会重复使用缓存数据,导致获取的序列主键值均为1,... 目录问题原因解决办法如果是存储过程总结问题myBATis有如下代码获取序列作为主键IdMappe

Java中最全最基础的IO流概述和简介案例分析

《Java中最全最基础的IO流概述和简介案例分析》JavaIO流用于程序与外部设备的数据交互,分为字节流(InputStream/OutputStream)和字符流(Reader/Writer),处理... 目录IO流简介IO是什么应用场景IO流的分类流的超类类型字节文件流应用简介核心API文件输出流应用文

java 恺撒加密/解密实现原理(附带源码)

《java恺撒加密/解密实现原理(附带源码)》本文介绍Java实现恺撒加密与解密,通过固定位移量对字母进行循环替换,保留大小写及非字母字符,由于其实现简单、易于理解,恺撒加密常被用作学习加密算法的入... 目录Java 恺撒加密/解密实现1. 项目背景与介绍2. 相关知识2.1 恺撒加密算法原理2.2 Ja

Android 缓存日志Logcat导出与分析最佳实践

《Android缓存日志Logcat导出与分析最佳实践》本文全面介绍AndroidLogcat缓存日志的导出与分析方法,涵盖按进程、缓冲区类型及日志级别过滤,自动化工具使用,常见问题解决方案和最佳实... 目录android 缓存日志(Logcat)导出与分析全攻略为什么要导出缓存日志?按需过滤导出1. 按

通过配置nginx访问服务器静态资源的过程

《通过配置nginx访问服务器静态资源的过程》文章介绍了图片存储路径设置、Nginx服务器配置及通过http://192.168.206.170:8007/a.png访问图片的方法,涵盖图片管理与服务... 目录1.图片存储路径2.nginx配置3.访问图片方式总结1.图片存储路径2.nginx配置