CDROM check protection I'm writing you this tutorial, to share my first cracking experience with you, to reveal the cracking from the perspective of a newbye. This tut deals with cracking a "CDROM check" type protection, in wich the program searches for a particular data on the original CD. This type of protection you'll encounter in many today's games which are distributed on CD. After you complete this tutorial you'll know how to: -deal with a DOS4GW program -tackle a CD check type protection You probably won't know how to crack any CD check protection, but, at least, you'll know how to start with it. To better follow this tut you'll need: - the game "Silent Hunter", a wonderful WW II submarine simulator, a must have for the strategy freaks. I worked on a Razor distribution, and I think you'll find it on the web. It's a CD ripped version, and it's just what we need, with animation files removed. - a patch for Silent Hunter, from version 1.0 to version 1.11. You can download it from SSI web site. This version is protected, so it won't work. It must be cracked. - some tools. I recommend: SoftIce for Win95 (DOS version is useless with all new programs), KGB ( the best spy I found, and it's with source code so you can modify it), DOS Navigator (for editing large files, and for many more...) and much patience :-) Now let's get to work. I'll describe my approach. I'm quite lazy so I always try to find the easiest way in doing something. This is how we'll do it: 1. When you run the program it asks you to put the original CD in drive. What do you do now? Well, I don't know, but my first try is to run a fakeCD utility (a little TSR wich emulates a CDROM drive on your hard disk) Doesn't work. 2. What next? I'll install KGB.exe and run the game again. This TSR will log some DOS INT 21h file access functions to a file. Now sit down and study the log. Here is an extras from the log file: OpenFile-R : ART\GUI_1.FNT OpenFile-R : ART\FONT1.FNT OpenFile-R : ART\FONT1.FNT OpenFile-R : BUD\10.SMK OpenFile-R : BUD\10.SMK OpenFile-R : BUD\10.SMK As you can see the program tries to open BUD\10.smk and, since it can't find it, assumes it doesn't have the right CD in drive. This is an animation file quite big, probably, to be sure it is left apart when the program is ripped from CD. 3. As I told you, I want to do it the easy (but the right) way. I don't want to step through code. At least not until I use all other means. So I make a file 10.smk, and put some crap in it. Run again the program. Doesn't work again! How stupid I can be! 4. Well, maybe it needs a valid smk file. OK. Let's give it one. I take a smk file from another game (smk files are quite standard for animation in games), and rename it 10.smk. Run SH again. No change. Still asks for CD. 5. Now, I think it's time to do it the hard way.It's time for little assembler and crackin'. I modify KGB to log all DOS functions (not only Open, Close, Execute File) and dump registers for each function (how cool was the guy who wrote the program, cause he gave the source code too). Launch (Yamato)KGB then run SH.EXE. Then you'll have a trace of all DOS functions called by program. 6. Now it's time to remember about what +ORC has taught us. It's time to use his best advice: Take a cool Martini Vodka, sit down, and meditate. (I don't use Martini Vodka due to reasons I cannot tell you. I drink only mineral water or orange juice. But remember! Only pure, sparkling mineral water from Carpatian mountains will do) Well, the log looks like this (I'll show you only AX register, here): any other file than "10.smk" ax=3d00, bx=... open file ax=4400, bx=... get device info (why?) ax=4201, bx=... seek current ax=4202, bx=... seek end (check length?) ax=4200, bx=... seek start ax=3e05, bx=... close handle ax=3d00, bx=... reopen file ax=4400, bx=... get device info ax=3f00, bx=... read data ax=3f00, bx=... ax=3e05, bx=... close for good the 10.smk file access looks different ah=2a wow! this is weird (get current date of system, maybe for a future compare?!) ax=3d00 ... open ax=4400 ... get device info ax=4400 ... again (?) ax=5700 ... get time stamp (REALLY WEIRD!!!!) As you can see, the program checks the time stamp of the original file, and probably it compares it with a built in value. This must be the protection. But how can we crack it?????? Well, load the exe in your favourite hex editor (mine is DOS Navigator) and search the string 57,CD,21.(mov ah,57; int 21) And, bingo, only one occurrence. Now write down on a paper some bytes from that address. This will be the signature of our timestamp function. 6. Now it's time to use our programming experience. As you know probably, today's games are written in C/C++ with some few parts in assembler (for optimizing). The most used compiler for DOS games is Watcom C/C++. It cames with support libraries and debugger for DOS4GW extender. So. A typical today's game is a program written in C, compiled with Watcom and linked with Watcom 32 bit libraries. You can verify this. Just run your favourite game and you'll see a message like: "Rational DOS4GW Runtime System v1.97...." (A little parenthesis: it seems that more and more games are written to work in Windows95 with DirectX. Maybe it's easier for the guys who program the game but for us, the users, this is shit. I used to play my favourite DOS game, with 8Megs of RAM, a 486 DX2 66 processor, and it works very well. It loads in a few seconds. But now, to play a game I must have a Pentium processor 16 Megs of RAM, and to tweak my Win95, and only to load the game I need 40 seconds. And why all this? Because Micro$oft wants to sell its fat operating systems, Intel to sell its processors. But we don't want that. Who cares? This stinks! >:-P . Sorry, but I'm pissed) Well, back to cracking. A DOS program with DOS extender(like DOS4GW) consists of 3 parts: -an initialisation code, which puts the processor in protected mode -the main program (written by programmer; this is the most important part) -some exit code which puts processor back in real mode. Parts 1 and 3, usually doesn't interest us, but we have to find a method to skip it in our stepping through program, and find the real entry point, in the main() function (that main() from C language) Now remember what +ORC thaught us in lesson 6.1: "...search for INT 21, AH=4c.It should be after the call to main function". All right! Let's search! But ... :-( , what a disappointement; too much INTs 21,4c. I got to find another method to reach the main function ('cause I'm lazy) Let's search for the string "WATCOM"... and... toward the end of file I find "Watcom c/c++32 runtime system". Great , this must be the runtime library, linked with the main program. And just before this string is a "EB 76" relative jump. Search again for "int 21, ah=4c" from here. ONE occurence. Yeah. This must be the int21 +ORC talked about. But if I am wrong? Check another DOS4GW program (Warcraft2 for example). Same "watcom c/c++32" string and same EB 76. Great. Try to replace EB 76 with CD 03 (int 3) and run program. SoftIce pops up (assumed you set I3HERE on) and finnaly you're in protected mode code. Put back EB 74, and see what happens. The prog reads the environment, the command line and then, calls main(). 7. Search the signature you wrote on paper :"57 CD 21...". Found. BPX on the entry point in routine (you'll find the entry point looking up in the code till you encounter some PUSH instructions or ENTER. Here you'll find both) When BPX is toggled, fetch the calling address from the stack. In SH the at the calling address you find: E84D030A00 call get_time_stamp Now find the entry point of this routine too and BPX on the entry point. The call to protection routine looks like this: E8760B0000 call protection Then sit down, take a cool mineral water (remember, only sparkling mineral water from Carpatians will do :-) ) and watch the program running. After some observation you'll notice that the protection routine is called only for the "10.smk" file. Till then, no call. So, if the call isn't made, it's a chance the protection to be removed. Well, let's try to deactivate the call protection instruction, and see if it works. Replace E8760B0000 with 5033C05890 i.e push eax xor eax,eax pop eax nop As +ORC told us, always try to avoid using too much NOP's! Newer protection schemes smell NOP patches. Here we used only one NOP. And maybe there is a solution with no nop. But I dunno all opcodes. 8. Now it's a good chance that the program checks for CD several times, not just once. If this is the case, our disable trick won't work. To see if this is the case, set a breakpoint on memory range over the two routines, to see if there is some access on them across the game. And, incredible there is no other access. The game runs smooth, never checks the file on CD, and the player is happy. 9. Here is the complete code for the protection routine. It's pretty well explained: 000857A2: C9 leave 000857A3: 5E pop edx ▌ 000857A5: 59 pop ecx ▌ 000857A6: 5B pop ebx ▌ 000857A7: C3 retn . . . ▌ 00086151: 53 push ebx ; entry point ▌ 00086152: 51 push ecx ▌ 00086153: 52 push edx ▌ 00086154: 56 push esi ▌ 00086155: C8480000 enter 0048,00 ▌ 00086159: 31F6 xor esi,esi ▌ 0008615B: 85F6 test esi,esi ▌ 0008615D: 0F853FF6FFFF jnz 000857A2 ;if esi not 0 goto leave... ▌ 00086163: 6A0A push 0A ▌ 00086165: 68543A0100 push 000013A54 ▌ 0008616A: 6875010000 push 000000175 ▌ 0008616F: 68B48C0200 push 000028CB4 ▌ 00086174: E885000A00 call 001261FE ▌ 00086179: 83C410 add esp,010 ▌ 0008617C: 6800020000 push 000000200 ▌ 00086181: 68B48C0200 push 000028CB4 ▌ 00086186: BE01000000 mov esi,000000001 ▌ 0008618B: E815010A00 call 001262A5 ;try to open 10.smk ▌ 00086190: 83C408 add esp,008 ▌ 00086193: 89C3 mov ebx,eax ;eax=handle of opened file ▌ 00086195: 83F8FF cmp eax,-001 ;or -1 if file not found ▌ 00086198: 7504 jne 0008619E ;jmp if opened OK ▌ 0008619A: 31F6 xor esi,esi ▌ 0008619C: EB1F jmps 000861BD ▌ 0008619E: 8D55B8 lea edx,[ebp][-0048] ▌ 000861A1: E84D030A00 call 001264F3 ;GET TIME STAMP ▌ 000861A6: 83F8FF cmp eax,-001 ▌ 000861A9: 7409 je 000861B4 ▌ 000861AB: 817DCEA4D60301 cmp dword [ebp][-0032],00103D6A4 ;!!!!compare timestamp ▌ 000861B2: 7402 je 000861B6 ▌ 000861B4: 31F6 xor esi,esi ▌ 000861B6: 89D8 mov eax,ebx ▌ 000861B8: E824040A00 call 001265E1 ;close handle ▌ 000861BD: 85F6 test esi,esi ▌ 000861BF: 759A jne 0008615B ▌ 000861C1: 687E010000 push 00000017E ▌ 000861C6: B985010000 mov ecx,000000185 ▌ 000861CB: BB88010000 mov ebx,000000188 ▌ 000861D0: BA9E010000 mov edx,00000019E ▌ 000861D5: B8C0010000 mov eax,0000001C0 ▌ 000861DA: E8278F0300 call 000BF106 ;put nag screen ▌ 000861DF: 83F8FE cmp eax,-002 ;eax = user input (retry,cancel) ▌ 000861E2: 0F8573FFFFFF jne 0008615B ;if retry goto ^ test esi,esi ▌ 000861E8: 31D2 xor edx,edx ▌ 000861EA: 31C0 xor eax,eax ▌ 000861EC: E8C7590200 call 000ABBB8 ▌ 000861F1: E965FFFFFF jmp 0008615B ;goto ^test esi,esi and leave ▌ 000861F6: 0000 add [eax],al ;never reach this address??? ▌ 000861F8: C3 retn Well that's all. So I cracked my first program. Now, when I'm looking back it seems easy, but it took me one week of full work to crack it. And it was a simple call. I wonder if there was multiple CD checks across the game, how long would have taken?