首先,分析游戏的文字显示方式。玩游戏的时候最怕打开游戏看到一片乱码,现在汉化游戏却是最希望打开游戏看到一片乱码,因为一般在非日文环境下能正常显示日文的游戏都使用的字符集限定,为汉化增加了不少的难度…… 8 M/ p# ?3 N% q' a* r: F
一起采取Textout方式的年代,日文、繁体中文十个有九个是乱码,于是有了Apploc和NL,后来比尔大叔推出了CreateFontA函数……CreateFontA能够直接定义程序使用和操作系统不同的字符集,于是乱码没有了,而汉化者噩梦开始了,因为汉化后的中文内容在经过日文字符集的编译后变为日语状态下的乱码。 7 N$ c+ A( h8 L; n G
以下是比尔大叔的MSDN Library对CreateFontA给出的函数原形 * J5 m. e M* H, `6 x( B
, D C, y5 M5 U1 o5 _HFONT CreateFont( " i5 L1 e1 C/ P |% z: Y; |
int nHeight, // height of font * V$ p" R7 u2 v8 d/ @3 o
int nWidth, // average character width
' Y7 Z/ }' [$ Mint nEscapement, // angle of escapement ' v6 p* Y5 u& Q6 \, g: {! G
int nOrientation, // base-line orientation angle % v I4 W# [+ r6 u0 z. o
int fnWeight, // font weight
) {* b! z8 e+ h' zDWORD fdwItalic, // italic attribute option
6 j+ v1 v# b% _9 B x6 A5 gDWORD fdwUnderline, // underline attribute option
& T& k2 r/ D- v/ H, C$ e/ MDWORD fdwStrikeOut, // strikeout attribute option
# r N( S6 v" f4 w+ |5 ^& P; uDWORD fdwCharSet, // character set identifier ) L+ j. @2 ?8 R' K$ J! Q6 V
DWORD fdwOutputPrecision, // output precision 3 L0 E- F7 e! {6 G( o
DWORD fdwClipPrecision, // clipping precision + }. m. ~3 M/ ]% C" {) B+ }- f
DWORD fdwQuality, // output quality
1 ?! R/ k' O, }8 Q9 a0 @DWORD fdwPitchAndFamily, // pitch and family
6 t; W4 |: {4 M* v8 i/ `+ B( j% l M: jLPCTSTR lpszFace // typeface name
% e L1 H0 H6 M* \, n2 }8 ]8 y. u);
+ h$ E0 K' R, O5 H, ~& L d. ~* @4 t
HFONT CreateFontIndirect(
% \( t g: I# d0 BCONST LOGFONT* lplf // characteristics A+ j- C; m5 F- G4 e; h5 m: I3 o
);
4 K' C; h5 ?2 P# m2 A2 t# A/ c1 T其中 LOGFONT的声明如下: : ~$ g: S& R N0 D# o
% n0 L# K9 l; vtypedef struct tagLOGFONT {
5 A0 Y1 E0 K, e& k) r9 H0 PLONG lfHeight;
5 b3 ^7 |" Z" Y$ |6 Z3 @LONG lfWidth; / @; s. H: a# |
LONG lfEscapement;
* C+ W% `( T; F( \LONG lfOrientation;
1 A' c' }8 F" \$ [# DLONG lfWeight; . b. E" \3 e1 s7 O# r( _% R3 h
BYTE lfItalic; ( ~$ P# w: S, ^% B; [% M- Q3 _
BYTE lfUnderline; * r/ F B/ }; C% b: f
BYTE lfStrikeOut; 6 I. O. r* J) b- j
BYTE lfCharSet; 2 L. }: s( Q2 ^ U- [9 r% s$ f
BYTE lfOutPrecision; : a' ~; o5 G" |3 q7 ?) F
BYTE lfClipPrecision; 1 f6 F. m$ ]3 a0 b" L
BYTE lfQuality;
( @+ [" H% `7 ?* J6 z$ q2 LBYTE lfPitchAndFamily;
/ x$ F" S) f7 }TCHAR lfFaceName[LF_FACESIZE]; 0 O: z2 {* J' D; u# M# W
} LOGFONT, *PLOGFONT;
/ g# f1 b9 a8 r* d" B3 S" w' u==================分割线=================
W. f5 A z" ?- s2 E6 L- [要改变程序支持的字符集,就要改变程序调用上面两个函数时的fdwCharSet或lfCharSet的值
$ O8 Z0 c6 s2 `, y" K6 [' R其中各字符集所对应的值如下:
" a3 P# |3 G0 J8 d# y( e$ X5 y7 Z1 X8 w! W; L" G+ u" f
字符集 值(十进制)
; K" H9 K: H" ~$ E2 K* LANSI_CHARSET 0 ! V* c& p, R$ k3 n) t
DEFAULT_CHARSET 1 3 r' K: z8 ]9 B. s
SYMBOL_CHARSET 2 ' A& {# D) Y: ?- V/ x3 E) @
MAC_CHARSET 77 " \+ g0 T* M, D2 ^7 U
SHIFTJI_CHARSET 128 6 [8 n' p |9 N, }3 |2 ?6 \
HANGEUL_CHARSET 129 / q5 z# W/ r# O, F& ~1 u7 W
HANGUL_CHARSET 129
# X" x" P Y: B" l [JOHAB_CHARSET 130
, f5 K3 V# Z* XGB2312_CHARSET 134 3 e0 k4 Y( a( Q3 ?! ]. w4 L
CHINESEBIG5_CHARSET 136 , O& p' Y( g' N' z
GREEK_CHARSET 161
# |" n, i0 v. uTURKISH_CHARSET 162
; p( b5 C( k( |( F: x& K; L& X( MVIETNAMESE_CHARSET 163 # K+ ]( A B$ ]$ o# H) V
HEBREW_CHARSET 177
: q6 e5 `0 `/ _5 a& ~" Q: q6 j2 T( zARABIC_CHARSET 178 * X u" o& v7 h8 V- U
BALTIC_CHARSET 186 . ~( a2 u6 o0 o( R/ F# r& Z
RUSSIAN_CHARSET 204 ( C. i1 f( }1 f# H1 f8 o# ?; i
THAI_CHARSET 222
/ l/ Y4 M; A1 n5 S4 q2 [; eEASTEUROPE_CHARSET 238
: d, M' p4 Z+ i& Z* t2 q5 O+ c' E# gOEM_CHARSET 255
1 [: Z9 e _( g8 r+ }可以看到简体中文是86(Hex),日语是80(Hex),繁体中文是88(Hex) * F7 T8 I: c" n$ d
我们要做的就是找到游戏中定义调用字符集的部位,将80改为86,这样就能让程序正确的显示中文。 9 P; |, ~0 \: I! d4 u3 H. n+ N
用pexplorer打开HANABIRA.exe,进入反编汇模式,查找Font字符串,我们发现调用CreateFontA函数的位置是唯一的:
! y d: j0 L" y& n* [于是在以下代码我们停下来 ' ]' O/ s: Q* L$ M
L0041E365:
( [5 D0 [1 q( B' H1 }% X6 h! ?mov edi,[esp+14h]
- ?, s8 j$ f6 W& e8 F$ J Jmov ebx,[esp+28h] $ T: M7 F2 ^: q, y
mov ebp,[esp+24h]
* E6 c- U$ N0 J4 B$ E8 ?mov eax,[esp+20h] # p: \/ x1 ?- `
mov ecx,[esp+1Ch]
2 q5 \+ T( X1 N4 S9 L( U x9 k! dpush edi
! P1 l7 [3 W _( X+ ?2 Smov edx,[esp+30h] . _1 d2 w1 P" |, g& Z) I V
push 00000031h ) E# k4 o7 W" A% B1 n W
push 00000002h 5 y2 F" b' \0 H2 @; ], _" @1 C
push 00000000h & a5 C* [1 G( R
push 00000000h
6 J* z! V2 a; h/ Cpush 00000080h
. F( c( C- V5 q* W. ^( H; ^4 zpush ebx
3 c! a$ X* }$ r$ W0 l! Z) lpush ebp 5 b f1 D/ O! d# J
push eax ( P& i- Y7 S2 R: \' {. ^' f6 f3 A
mov eax,[esp+3Ch] * B2 l" a/ u+ K
push ecx - i* K2 ]3 |) Q& m, `: L
push 00000000h * b7 j' p% y! p0 |+ y& F
push edx
r8 P, D( z: `$ Q* F' V2 Cpush 00000000h
+ A- p }' o2 U0 fpush eax 1 z* k& [- O8 }' d. P
call [GDI32.dll!CreateFontA]
1 }: }5 b3 K' _4 Y8 `+ m+ Dlea ecx,[esi+28h]
8 y$ c/ h1 f* u- M2 {* l( Mmov [esi+00000138h],eax , Q7 K4 m9 o' E
mov eax,edi - b; e* W1 D# g- X+ i$ n1 n
sub ecx,edi & I+ y5 u) d( t- h2 k
lea esp,[esp+00h]
2 x: P, g g4 K& z( u- G注意这就是调用GDI32.dll中的CreateFontA函数了,我们在这个堆栈中寻找将80这个值传递给CreateFontA的部位。 * k$ \1 O/ C& B7 R' b" q" t8 x
push 00000080h
! G$ D1 y Z+ `就是这里将80值压入传递中
( ^" O% D/ J. t& E( X5 ~PE中标记了这段赋值的Hex数值,用UE打开文件,找到 & ^1 m1 r' @$ d% X2 s( C
68800005355
) a( m) X# x6 z0 z: q; W将其改为 & |0 O/ W5 Z4 f. m
68860005355
( @% j) a0 t/ i, Y) p7 t保存之。
m" O) @, R2 U7 r& \. x* w这样PE中看到这段压入就成了
* d/ E0 n$ }) `0 {+ i2 bpush 00000086h
6 @: q/ |. P9 g7 Q0 y* {初战告破,运行游戏,你会看到日文全变成了乱码,说明程序已经在使用中文的字符集了
# p( `8 U5 e- {0 B3 p修改游戏脚本,加入几个中文看看…… ; R' y1 K/ H# }, _9 q2 w, i
为什么我添加的中文全部是“□”?
7 G! L. T0 f+ p; Y9 y这就是需要解决的第二关卡,字符集边界检查。 1 X% C8 W, g9 H, t; g+ K
. A8 C' ?. U5 q& v! u/ }既然已经设定了字符集,为什么还要边界检查呢?这是为了防止当游戏文本中含有某些非法的字符串时产生缓存溢出。于是在字形传递到GDI32.dll描绘字体准备显示在屏幕之前对其进行检查,发现超出了设定的缓存大小就将其拦截下来,于是屏幕上就显示出一个“□”。我们知道,由于日文的字符比中文少得多,所以这个缓存也小的多,换言之就是边界太窄。 , _ L( |, F% ~% D% I
边界检查的例子:
9 M7 f4 I- r5 [ |4 @& Q0 S( b3 _cmp al,80
) t( @& u2 _& }/ ]5 U# zjbe xxxxxxxx 7 Z, {" |8 r- x- r5 D) @! X, Y! [: E
cmp al,09F
6 w/ E3 A: E$ zjb xxxxxxxx
6 v/ X4 E- J( l, E* j; s& Z6 i( y7 ecmp al,0E0 9 J/ Q3 M4 T1 ^3 H0 {/ s
jb xxxxxxxx
! @# o6 e9 d* f/ ]6 a3 wcmp al,0FC " j0 S x. v! A. T! l
ja xxxxxxxx ) r7 |; J, K3 q4 ~! t
: @! H) s7 @4 D3 {即看字符是否在80-9F(前两位)和e0-fc(后两位)之间
8 ]% h$ H& L2 e: ]* Z& ~! g* R=================================== ) t! _3 l* u& o* ^
如何具体查找程序的字符集边界呢?常用的方法是下断,用OllyDbg载入游戏主程序运行,一步一步断下去,在出现一堆“□”的时候停住,然后转到ASM模式查看停在哪里。 3 ~# V! |8 h2 X9 P+ H
L0043BA00:
/ d, O( p1 i/ ^3 x/ Acmp al,80h ) `6 \6 k* Q) ~! _3 b
jc L0043BA08 ! ]0 J+ }( m9 L/ M9 d. y
cmp al,9Fh
: M* z! A8 o, \: }9 G `2 r) _jbe L0043BA10
+ _; K: ]' T) @# eL0043BA08:
2 ^6 @; N# l9 f, Gcmp al,E0h 4 {! d6 @3 Q3 Z0 t) j; [$ }# Z# x
jc L0043BA30
" W7 i5 H# w+ u* \cmp al,FFh
! I$ q |! u( r% u3 K+ K7 yja L0043BA30
( J, A; P3 s% f. v m6 I8 ~) m花瓣的上边界为80~FF,而下边界为E0~FF,好,开始动手
2 U4 t5 ?8 B0 X; N( d* Q这里我们将上边界改为80~FF,而下边界范围足够宽广,就不用改了。
8 W0 k' ?$ e: N0 @( [3 u. S% s这里为什么使用80~FF而不改为20~FF呢?因为我们需要让游戏文本中原本的日语空格(8140)不显示为乱码,于是8140刚好在边界外,就不会被送去CreateFontA进行显示,就会显示为日语的缺字码,一个空白——而它刚好就是空白,在显示上两者没有任何区别。 : O" u6 h% f: |$ x* d- ^1 n6 A
如法炮制,打开EU修改之,保存测试,OK,正常显示了
' B5 v8 c) p0 N- y$ ^- Q) u9 ~2 n& \& S1 B. _
原文3 {- q4 m) E Q# ~0 Y: M+ X2 N
http://blog.potatoneko.cn/ |