《超星图书阅览器》图书目录文件格式介绍

《超星图书阅览器》图书目录文件格式介绍

========================================
《超星图书阅览器》图书目录文件格式探秘
========================================
版本 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 字节为单位顺序进行,最后不足单位的字节忽略。

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;
}
}

_____________________________________________________________________
“超星图书阅览器” 是北京世纪超星信息技术发展公司的注册商标
<<< 全文结束