|
《超星图书阅览器》图书目录文件格式探秘
--------------------------------------------------------------------------------
发表时间:2004-8-10
作者:未知[获得此文档时候没有作者记录,深感抱歉,本文档全为转载]
========================================
《超星图书阅览器》图书目录文件格式探秘
========================================
版本 1.1
Shun Cox (shuncox@263.net)
注意:
本文可以任意复制、转换和发布,但必须保持文件内容的一致和完整。
有任何问题请到 bookexpress.iscool.net 或邮件 shuncox@263.net
一、.dat 文件
_____________________________________________________________________
整体结构为
+-----------+
¦ 头 ¦
+-----------+
¦ 记录 1 ¦
+-----------+
¦ 记录 2 ¦
+-----------+
¦ ... ¦
+-----------+
¦ 记录 n ¦
+-----------+
.dat 文件的头结构
struct DatHead {
DWORD Mark; // 标志,'SSDT'
WORD Ver; // 可能是版本号,一般为 2
WORD Type; // 类型,在 site 和 address 中都为 2,
// bktree 中为 0(V3.51 中为 2),
// bklist 中为 3
DWORD NumOfRec; // 含有多少记录
DWORD LenPerRec; // 每个纪录的长度
DWORD LenOfHead; // 头的长度
DWORD EncOffset; // 加密段开始处,只在 bklist 中有效
// (在 V3.51 中此变量无效)
};
二、不同 .dat 文件的记录结构
_____________________________________________________________________
site.dat
--------
记录完全加密
struct SiteInfo {
char Path[32]; // 本地路径
char Name[64]; // 图书馆名称
WORD Rev1; // 未知
char Date[22]; // 更新日期
char URL[132]; // 更新书库的 URL
DWORD Rev2; // 未知
};
address.dat
-----------
记录完全加密
struct AddrInfo {
DWORD Rev; // 未知
char Addr[]; // 下载主机地址
};
bktree.dat
----------
V3.51 之前记录没有加密,V3.51 对整个记录完全加密
struct TreeInfo {
char Name[100]; // 分类名称
WORD Rev; // 未知
WORD Level; // 分类所处的层数
DWORD Start; // 在相应 bklist 中的起始记录号
DWORD Num; // 图书数量
};
bklist.dat
----------
bklist 的记录相对复杂
偏移 0 处(V3.51 以前不加密,V3.51 对整个记录完全加密):
struct BookInfo {
char Name[100]; // 书名
DWORD Pages; // 页数
WORD AddrNo; // 主机编号(从 1 开始,主机地址在
address 中)
};
偏移 DatHead::EncOffset 处为加密段(V3.51 从 0x6a 开始):
* 对于 SSREADER V3.4
struct EncInfo {
WORD EncLen; // 加密长度
// 加密开始
WORD Copyright; // 版权,如果(Copyright >> 8 == 0)则可以下载
char Path[100]; // 路径和文件名
char Author[]; // 作者,如果(DatHead::LenPerRec != 0x110)则无
};
* 对于 SSREADER V3.5 - V3.51
struct EncInfo {
WORD EncLen; // 加密长度(在 V3.51 中此变量无效)
// 加密开始
WORD Copyright; // 版权,如果(Copyright >> 8 == 0)则可以下载
char Rev[58]; // 未知
DWORD SN; // 超星书号
char Path[]; // 路径和文件名
};
三、解密算法
_____________________________________________________________________
解密以 0x10 [wiki]字节[/wiki]为单位顺序进行,最后不足单位的字节忽略。
void Decrypt(DWORD *s)
{
DWORD v = 0xe3779b90;
DWORD d = 0x9e3779b9;
static DWORD k[] = {0x3f65496d, 0x61737745, 0x2e2c2e5e,
0x38375f2f};
int i;
for (i = 0; i < 0x10; i++) {
s[3] -= ((s[0] << 4) + k[2]) ^ (s[0] + v) ^
((s[0] >> 5) + k[1]);
s[2] -= ((s[3] << 4) + k[0]) ^ (s[3] + v) ^
((s[3] >> 5) + k[3]);
s[1] -= ((s[2] << 4) + k[2]) ^ (s[2] + v) ^
((s[2] >> 5) + k[3]);
s[0] -= ((s[1] << 4) + k[0]) ^ (s[1] + v) ^
((s[1] >> 5) + k[1]);
v -= d;
}
}
请注意:本文并非原创,是转载,因已丢失原有效链接地址,直接就贴上来了. |
|