cocos2dx游戏加密

在手机游戏当中,游戏的资源加密保护是一件很重要的事情。

我花了两天的时间整理了自己在游戏当中的资源加密问题,实现了跨平台的资源流加密,这个都是巨人的肩膀之上的。

大概的思路是这样的,游戏资源通过XXTEA加密方法对流的加密方式,有自己的密钥和标识,通过标识可知是否有加密,密钥是自己程序当中的。除非有密钥,否则很难通过解出正确的文件。经过加密后,加密文件也就是游戏资源放在resource的自己文件夹中,否则在xcode编译到趁机是会识别不了文件。在程序中cocos2dx底层加入解密过程,就可以把文件正确读取出来,让程序显示。经试验,已经可以读取,png,plist,json文件。

现在记录下实现的步骤

链接: http://pan.baidu.com/s/1ntx98VZ 密码: qyqe去下载加密资源的脚本,这个是quick-cocos2d-x提取出来的打包工具。

pack_files.sh -i olddir -o newdir -ek XXTEA -es decodetest

把ResourcesDecode和xxtea四个文件加入到cocos2dx/platform下;

把platform/ResourcesDecode.cpp \
platform/xxtea.c \加入到cocos2dx/platform的android.mk文件中,加入android编译。

写一个单例用来保存密码和对流解密过程,代码如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
CCAssert(buf != NULL, "decodeData buf not NULL" );
unsigned char * buffer = NULL;
ResourcesDecode* decode = ResourcesDecode::sharedDecode();
bool isXXTEA = decode && decode->m_xxteaEnabled;
for (unsigned int i = 0; isXXTEA && i < decode->m_xxteaSignLen && i < size; ++i)
{
isXXTEA = buf[i] == decode->m_xxteaSign[i];
}
if (isXXTEA)
{
//decrypt XXTEA
xxtea_long len = 0;
buffer = xxtea_decrypt(buf+decode->m_xxteaSignLen,(xxtea_long)size -(xxtea_long)decode->m_xxteaSignLen,(unsigned *)decode->m_xxteaKey,(xxtea_long)decode->m_xxteaKeyLen,&len);
delete [] buf;
buf = NULL;
size = len;
}
else
{
buffer = buf;
}
(pSize)
{
*pSize = size;
}
return buffer;


buffer就是经过XXTEA解密后正确的流。

CCFileUtils::getFileData()当中return返回之前调用解密pBuffer =ResourcesDecode::sharedDecode()->decodeData(pBuffer,size,pSize);这里是跨平台的读取资源的方法。

ZipFile::getFileData()当中也加入解密方法pBuffer =ResourcesDecode::sharedDecode()->decodeData(pBuffer,fileInfo.uncompressed_size,pSize);这个是android读取plist的地方,我也不太清楚为什么android会在这里读取资源。

在boolCCSAXParser::parse(constchar*pszFile)中把原先的rt改为rb :char* pBuffer = (char*)CCFileUtils::sharedFileUtils()->getFileData(pszFile,/*"rt"*/"rb",&size);

ios的修改地方 不一样

在CCFileUtilsIOS中的createCCDictionaryWithContentsOfFile修改如下,注释掉的是原先的,后面是新增的。

?
18
CCDictionary* CCFileUtilsIOS::createCCDictionaryWithContentsOfFile( const std::string& filename)
{
std::string fullPath = CCFileUtils::sharedFileUtils()->fullPathForFilename(filename.c_str());
// NSString* pPath = [NSString stringWithUTF8String:fullPath.c_str()];
// NSDictionary* pDict = [NSDictionary dictionaryWithContentsOfFile:pPath];
long fileSize = 0;
* pFileData = CCFileUtils::sharedFileUtils()->getFileData(fullPath.c_str(),monospace!important; border:0px!important; color:blue!important; bottom:auto!important; float:none!important; height:auto!important; left:auto!important; line-height:1.1em!important; outline:0px!important; overflow:visible!important; position:static!important; right:auto!important; top:auto!important; vertical-align:baseline!important; width:auto!important; font-size:1em!important; min-height:inherit!important; background:none!important">"rb" ,&fileSize);
NSData *data = [[[NSData alloc] initWithBytes:pFileData length:fileSize] autorelease];
[]pFileData;
NSPropertyListFormat format;
NSString *error;
NSMutableDictionary *pDict = (NSMutableDictionary *)[
NSPropertyListSerialization propertyListFromData:data
mutabilityOption:NSPropertyListMutableContainersAndLeaves
format:&format
errorDescription:&error];




在CCImage.mm当中修改,同样是注释是原先的,后面是新增的。


static _initWithFile( const * path,tImageInfo *pImageinfo)
CGImageRef CGImage;
UIImage *jpg;
UIImage *png;
ret;
// convert jpg to png before loading the texture
// NSString *fullPath = [NSString stringWithUTF8String:path];
// jpg = [[UIImage alloc] initWithContentsOfFile: fullPath];
fileSize = 0;
* pFileData = cocos2d::CCFileUtils::sharedFileUtils()->getFileData(path,&fileSize);
NSData *adata = [[NSData alloc] initWithBytes:pFileData length:fileSize];
[]pFileData;
jpg = [[UIImage alloc] initWithData:adata];

android平台

在CCImageCommon_cpp当中修改如下

8
CCImage::initWithImageFileThreadSafe( *fullpath,EImageFormat imageType)
bRet = false ;
nSize = 0;
#if (CC_TARGET_PLATFORM == CC_PLATFORM_ANDROID)
CCFileUtilsAndroid *fileUitls = (CCFileUtilsAndroid*)CCFileUtils::sharedFileUtils();
// unsigned char *pBuffer = fileUitls->getFileDataForAsync(fullpath,"rb",&nSize);
* pBuffer = CCFileUtils::sharedFileUtils()->getFileData(fullpath,&nSize);

到此,基本结束了。 在自己程序当中加入资源前把设置密钥和标识和自己加密资源时的一样:ResourcesDecode::sharedDecode()->setXXTeaKey("XXTEA",strlen("XXTEA"),"decodetest",strlen("decodetest"));

其它就正常的读取和显示。

参考网址:http://my.oschina.net/SunLightJuly/blog/184061

http://my.oschina.net/SunLightJuly/blog/189971

http://my.oschina.net/SunLightJuly/blog/184179

原文地址:点击查看原文

相关文章

    本文实践自 RayWenderlich、Ali Hafizji 的文章《...
Cocos-code-ide使用入门学习地点:杭州滨江邮箱:appdevzw@1...
第一次開始用手游引擎挺激动!!!进入正题。下载资源1:从C...
    Cocos2d-x是一款强大的基于OpenGLES的跨平台游戏开发...
1.  来源 QuickV3sample项目中的2048样例游戏,以及最近《...
   Cocos2d-x3.x已经支持使用CMake来进行构建了,这里尝试...