|
本帖最后由 银河漫步 于 2011-9-30 23:11 编辑 - l6 n' g$ g5 z E
2 u/ m8 J9 P( B& R/ z
基地址的查找与特征数据
3 a& o, J! ?( I2 [# o(指针部分的使用说明在二楼,对基地址没兴趣的可跳过)
% k3 F: Q0 ~) H/ F, C2 V( G$ M- S+ c2 \4 H' V3 `
所谓基地址其实就是指针,其用法和指针的用法几乎相同,但是只所以区别开来还是因为模拟器这种东西,每次模拟游戏时游戏里的数据都是相对于某一个地址的偏移地址,所以与其每次都取写这个指针,不如把这个地址记录下来作为这个模拟器的基地址,则可以在做相同模拟器的游戏的时候方便很多/ M* p- }8 T E
指针怎么查找?CM其实目前没有反汇编的功能,所以关于指针的查找还是建议使用CheatEngine或者OllyDBG,我个人推荐前者. u( x$ ^& W# P) t: @9 ^) j/ P
CheatEngine简称CE,这软件功能十分强大,但是直到6.0版本,对于修改器界面的设计始终功能比较简单,无法做出比较随心所欲的修改器,而纯粹以编程的方式去处理修改器界面则门槛相对较高,所以也有了CheatMaker这一类的中间软件层的存在,所以就笔者我而言,是属于用CE寻址,然后用CM实现界面设计的这一类人(另一说,CE 6.1的版本已经可以推出了比较好的修改器设计功能,但是到目前为止笔者尚未研究出头绪…)
F9 l5 h* H9 j. Z5 l
: o# p) o7 @5 H, z; l以SFC模拟器:Snex9X为例子,模拟器版本Ver 1.52,游戏三国志III,如下图查找金钱
1 a& y, s8 {4 X* x7 O, V
" ]! c" b; b4 z) ?' r4 B+ `# |7 z {: ~
4 x7 z- D: T3 s8 [4 u Q得到了唯一地址:013BF460
4 \; k }5 i7 _! o( ~- G8 E% Q2 h# o: ^- G: _; V8 s* z
' `7 a# P6 e. `+ q& a
0 F* B8 L) S! t9 p* I: G然后对这个数据跟踪,得到一个可能的地址指针值,注意这个地址指针值的013BCBA0是对应ESI的值,而EBX的值是28C0,这个是指的平原金钱的偏移地址
( Z4 f3 v0 F0 v' `' A9 E值得注意的是[ebx+esi]的模式esi通常将不是固定值,这种时候,指针的好处就特别明显了,因为每次用模拟器模拟同一个游戏,对应的数据的搜索的绝对地址往往都在变动0 l# m S$ S+ d0 B' [/ H3 g
但是相对esi的ebx的偏移地址则是固定的5 C" a3 I2 T+ q8 h6 D4 C
* r- r g/ @( w' p
' v$ r6 q/ c! y& c6 d5 z1 \; ?4 k+ M F, J3 L/ b, L2 G
直接搜索相关这个地址的内容可以得到以下结果
- R1 Y" U4 d8 F
4 S% O1 S( b3 o; j! a( ^+ ]: W# k9 O: ~$ v
: A, a& j9 k6 D6 ]' l$ |- n; I/ C* r
如果得到的结果很多需要做一些进一步的测试和跟踪, 其实这里搜索到的这些都可以作为这个版本模拟器的基地址,比如任意选择一个0076448C0 M4 {' C K) ~, z! Y; \. a+ ~2 D2 w1 P
. H2 @% H/ ~: B: B6 P
8 G, Z! O7 j$ W0 Z
( _$ k, D N9 L. l) d2 k$ m我们可以看到,指针指向的地址的数据已经和原地址数据一致了 ,这里的28C0是之前的ebx的值,就是刚才平原城金钱的相对偏移地址. S5 C0 A7 c! k4 v
那么我们把这个指针添加到CM的支持程序管理3 Q, w/ Z) }( l3 U- v
. ~. B( _. H7 j% Q2 s* w$ z+ A6 J! ?3 E) `
* Z: B- L% a" ]: x9 ?8 m
) q6 M2 |; ^# E0 ]; Y% v
这里的最大偏移量是指的相对于这个基地址所指向的地址,最大允许的偏移量是多少,SFC游戏几乎不会超过20000,但其实这里填0表示不限制,也没有关系; j* I: U" u1 C2 D" H: E8 Z( O
添加完毕后,我在CM里搜索平原城的金钱地址得到如下图片* p5 u; H. X" G/ z$ n, k* t
6 F" p3 V5 j7 S: ]6 _
/ V( P' d* B" \, C$ _1 Z) o/ O& B% g8 \/ W. j! w
这个时候,对应的CM地址等于了原来的ebx的值,即这个金钱地址的相对偏移地址,我们的基地址添加成功了$ M$ L& ?# w5 O" m. U6 E7 [/ `
1 a3 j0 \# E# ^8 `* X( H! [但是这么添加基地址,仅仅只是支持了Snes9X Ver 1.52版,如果用户换了一个版本的模拟器,则将出现不支持的情况,这是因为一个程序的升级通常会让基地址发生变化,所以在模拟器里要能够支持一个系列全版本的模拟器,就需要使用动态定位基地址的方法,于是采用了“基址特征数据”这个概念9 l" q0 i! {9 h4 y4 a+ Y2 W
( |( f. [1 f- h0 P- y3 A回到我们之前的搜索出很多基地址的那个步骤,这里选择基地址其实是有诀窍的,我们启动另一个版本的Snes9X模拟器,模拟同样的游戏,同样搜索平原金钱的地址指针8 _, A3 t2 r, ^9 ~3 B, y) B
我们保留之前的模拟器进程和调试器进程,同时启动不同版本的模拟器进程双开另一个调试进程来进行跟踪调试以方便对比
) k$ n6 ]# g5 `# R) J1 A7 y4 w* X9 M, Z9 Z9 d
# P, n" w" z' c _9 w& y1 z4 U* K+ s# J% f
采用同样的方法得到了 也可以得到基地址列表1 }# S4 n6 U0 r, o% F4 Q
那么现在需要从这些基地址里面找到可以通用的特征码,这个比较考验经验,查看特数据的方式可以右键点击任意基地址,点击“反汇编该内存区域”7 f u9 q' A X3 C/ l. o! g
然后找到两个不同版本的模拟器之间的通用的一个基地址特征数据,如果找不到甚至要开始进行2级或者多级指针查找. s4 d: f: _5 a2 z0 G2 W
所谓的基址特征数据就是指的通过一段内存数据的搜索可以定位到一个固定地址,然后通过这个固定地址增加一个偏移值就可以定位到基地址
9 S: t- K5 V* V) r7 k; p比如这里SFC模拟器SNes9X我们就用2级指针来进行基地址搜索以及确定基址特征码,首先以0076448C作为一级指针来追踪二级指针& v+ y( {3 E( |* H/ `
(如果这里对于多级指针的查找和添加方法看不明白的,请在网上搜索CheatEngine八步教学的教程自学一下)
8 v7 d! c2 e' H8 E. t# v+ o3 u6 q0 _" B. v
; |5 w7 c, c$ ~/ s! {. ~
4 _% e; |9 l4 N2 u
重新读取一下游戏就可以找到以下代码/ E% E8 j, Q; J" C
% o. A7 c% O6 h6 y8 L
: `$ S/ |# s! M, x3 a3 R( n% P* G9 `; c, a, H
使用深度扫描可以找到对应007311C8的指针
+ N b+ o" R; f: E4 Z; x" ^, I0 A. M2 K% f
- Z, H" ?! \8 \
m4 B! {; \! u这个时候对于基地址的添加就多了一种方式,使用2级指针进行寻址,这时候【附加数据】需要填上1,才能正常寻址( D1 m& w% M f
+ O2 F# P) k- p& [2 P8 e) z
8 Z3 v' I# W. g: |6 j
0 f d3 \9 _6 e" L5 [; y1 f改完后我们可以尝试搜索平原的金钱,可以看到添加2级指针的基地址的结果和之前的效果相同0 V+ [" A2 [! j
( v7 Z7 o: v# ^# ?4 ~
9 q6 o$ u# Z' l! u
2 U' s, j( y1 G1 a但是换一个版本的模拟器时同样无法使用" R1 U2 o6 o# c% S
所以我们对于其他版本的模拟器以同样的方法和选地址方式进行这个2级指针的寻址,得到对比结果如下
% u* x" v+ L# S8 g$ `' Z4 x
4 o; Y( O' t- W
9 ` g4 g ]) U A) ] U' r' S+ q+ E& k% o; I- Y# F8 E
可以看到两个版本的模拟器,对应的2级指针地址很接近但并不相同,那么我需要一种可以自动定位到各个版本的基地址的方法,这种自动定位的方法就是“特征码寻址”) j1 s1 ]$ w* G' r# V, D$ n
我们分别对上图的红框内的2级指针地址查看反汇编的代码,如下图# [0 @. K& ^& h% n- j" o0 p' K$ @
0 ?2 u; k8 i$ x: ]9 V, c* f0 e
: N" {3 S9 e6 U" X
然后又可以得到一个对比图,下图中选中的部分就是相关到2级指针的地址的代码
8 n" T1 V [. k. w% x7 H& ~ l% X' |& v) w0 \) m: c. Q$ b
6 m* M ], H0 H2 V# Y3 l3 h' c
% }7 T+ V- ~6 h$ ?通过观察可以看到,虽然2级指针的地址和代码不同的模拟器版本有一些区别,但是在这段代码前两者确有一段共同的代码:2B C7 48 C1 E0 05 50
, b- y& I' P2 S% y! L我们不妨对这段代码进行搜索,如下图,注意反汇编里看到的字节搜索的时候要反过来,2B C7 48 C1,搜索就要搜索C148C72B
2 Q6 J2 A2 Q+ u) y
# P: J3 r& ]* l
C$ K4 P+ m. ^0 W: J' h
: d$ g6 B' S7 E5 [3 O非常幸运的,这段代码在模拟器的程序里具有唯一性,那么我们就可以通过这段代码定位到之前的2级指针的地址
2 X7 z! Y! t. P6 G! H X9 c这段代码的地址是00409664,2级指针的地址是00409674,两个地址之前相差10(H),那么我通过定位到00409664,然后增加10(H)的偏移就可以得到2级指针的基地址00409674
6 t3 g: Q2 U& h! m我们在看一下另一个版本的模拟器,这段代码定位的地址是00409FD4,而2级指针的地址是00409FE4,差值也是10(H): O& e( P) M0 l$ b2 k
那么这两个版本的模拟器的共性我们就找到了(这种共性肯定不止一处,这里只是介绍方法)
% s! c# a3 y& o) |: @3 G! j最后我们就是要把共性统一成模拟器的基地址) m) ?- ^, A" N( W- `
: S, s; U. q( e5 k" W4 w ?% v
/ d8 w/ s% |0 y3 q6 F1 {5 j
5 c9 n1 @ |' L" y这里的基址特征数据就是用刚才我们所说的不同版本模拟器的共性进行定位到基地址,而这里的附加数据“5”表示找到的基地址的指向的地址的偏移,我们再把前面添加2级指针的基地址的图来对比一下
?) ^' \* Y2 v; N
$ H! M4 P* ^% K* m# M
( y4 D: r, a+ H/ {: \- |% x! V7 I4 h4 P" l* Y
按照基址特征数据的方式,我们尝试一下对其他版本的模拟器搜索金钱地址的结果,之前我搜索的是Snes9X 1.52版,现在我来搜索1.53版
. s9 l* y0 P% u+ t+ K" B) {: G, W: ^# z( g b2 s" ~
8 I8 @3 W6 f+ q$ E' M+ X) _+ t7 K P( C1 K$ I, m
Z5 o3 Q& i1 d! G
我们可以看到,现在两个版本的模拟器通过特征地址的方式统一了搜索的结果,但是这样其实我们还只是作出了通用1.52和1.53两个版本的模拟器
1 j" t8 W/ O2 N& b# V% b6 {: o% ^如果想作出更大的兼容性,则还需要拿更多的版本的Snes9X来做这种共性的测试,最终找到一个共性地址
( K5 j u* X; r) ?( _值得说明的是,这里的基址特征数据可以用?做替代,比如2BC748C1E00550作为特征地址码的话可以写成2B??48C1????50,这样其实可以模糊出更多的版本共性出來
% ~+ m& V% I U' E2 f0 o0 M0 g7 ~1 h本文可以看作一篇实例教学,关于附加数据和添加说明写法的详细说明) Z" V1 Z5 }9 D3 ^* q& Z
还可以参考http://www.cheatmaker.org/Help/supportDoc/supportDoc.html0 D& H6 G' E: [3 q3 w4 h
这里最下面的位置写了各个基地址和附加数据的写法和作用
$ ^: i4 J; w8 V* |
+ x: O' R1 I- {# U, A |
本帖子中包含更多资源
您需要 登录 才可以下载或查看,没有账号?立即注册
x
|