Keygen 1: seVeb’s crackme05
Link: http://crackmes.cf/users/seveb/crackme05/ (now dead, see archive alt.) (binary)
This challenge is somewhat different, since our aim is to build a keygen instead of simply breaking the binary. It will therefore involve more reversing and no patching.
$ ./crackme05_32bit
Welcome to crackme05 reverser!
Your task is simple, figure out a way to generate valid serials.
Patching is as expected not allowed. Write a keygen and tell us
how you solved the crackme.
Invoke the crackme with your serial as the first argument to check
for a valid serial.
Usage: ./crackme05_32bit serial
$ ./crackme05_32bit foo
ROCK 4: Serial not 19 chars!
,--.!,
__/ -*-
,d08b. '|`
0088MM
`9MMP'
I have not failed. I've just found 10,000 ways that won't work.
- Thomas Edison
Note “ROCK 4” and the message about the length. Hopper shows that there might be multiple validation steps, with shortcuts likely going to an error function:
main
starts with a check on argc
> 1, and goes to usage
if it doesn’t match. We then have a sequence of calls to rock
, paper
, scissors
, and cracker
, all called with our input as a single parameter. Here’s the first one:
mov eax, dword [esp+0x10c+var_100]
add eax, 0x4
mov eax, dword [eax]
mov dword [esp+0x10c+var_10C], eax ; argument #1 for method rock
call rock ; rock
Since they are in sequence and with branching instructions, we can assume that they will all call the failure function if at any point the input is incorrect.
In rock
, a loop goes over the characters of our input until it reaches a null byte:
loc_80489dd:
mov edx, dword [ebp+var_C] ; CODE XREF=rock+20
mov eax, dword [ebp+arg_0]
add eax, edx
movzx eax, byte [eax]
test al, al
jne loc_80488d8
It compares the character to 0x2c (’,’) and if lower or equal, goes to a procedure that calls bomb
:
mov edx, dword [ebp+var_C]
mov eax, dword [ebp+arg_0]
add eax, edx
movzx eax, byte [eax]
cmp al, 0x2f
jg loc_8048934 ; continues below
[... bomb ... ]
loc_8048934:
mov edx, dword [ebp+var_C] ; CODE XREF=rock+53, rock+68
mov eax, dword [ebp+arg_0]
add eax, edx
movzx eax, byte [eax]
cmp al, 0x39
jle loc_804897e
There are multiple blocks comparing the current character to various constants. With the help of Hopper’s disassembly tool, we can reconstruct the logic:
int rock(char * input) {
int count = 0x0;
int offset = 0x0;
do {
eax = input[offset] & 0xff;
if (eax == 0x0) {
break;
}
if ((input[offset] > ',') && (((input[offset] <= '-') || (input[offset] > '/')))) {
if ((input[offset] > '9') && (input[offset] <= '@')) {
printf("ROCK 2: %i - %c\n", offset, input[offset]);
bomb();
}
else {
if (input[offset] > 'Z') {
if (input[offset] > '`') {
if (input[offset] > 'z') {
printf("ROCK 3: %i - %c\n", offset, input[offset]);
bomb();
}
}
else {
printf("ROCK 3: %i - %c\n", offset, input[offset]);
bomb();
}
}
else {
if (input[offset] > 'z') {
printf("ROCK 3: %i - %c\n", offset, input[offset]);
bomb();
}
}
}
}
else {
printf("ROCK 1: %i - %c\n", offset, input[offset]);
bomb();
}
count = count + 0x1;
offset = offset + 0x1;
} while (true);
if (count != 0x13) {
puts("ROCK 4: Serial not 19 chars!");
eax = bomb();
}
return eax;
}
This gives us the following constraints:
- The serial needs to be 19 characters long.
- Each character needs to be after ‘,’ and before (including) ‘-’ or after ‘/’.
- They can be between ‘0’ (’/’ + 1) and ‘9’
- They can’t be between ‘9’ and ‘@’ included, which is the range “:;<=>?@”
- They can’t be after ‘Z’ and after ‘z’, which means ‘A’-‘Z’ is fine
- They can’t be after ‘`’ and after ‘z’, which means ‘a’-‘z’ is fine
So something like f00-b4r-123-abc-DEF would pass. Let’s try it:
$ ./crackme05_32bit f001-b4r0-1234-abcD
Paper 1
We’passed the Rock challenge, we’re onto the next one. We can disassemble paper
and clean it up into the following:
int paper(char *input) {
var_10 = input[0xa] ^ input[0x8] + 0x30;
var_c = input[0xd] ^ input[0x5] + 0x30;
if (var_10 <= 0x39 && var_C <= 0x39) {
if ((var_10 <= 0x2f) || (var_C <= 0x2f)) {
puts("Paper 1 lower");
bomb();
}
}
else {
puts("Paper 1");
bomb();
}
if (input[0x3] == var_10 && input[0xf] == var_10) {
if (input[0] == var_C) {
if (input[0x12] != var_C) {
puts("Paper 3");
eax = bomb();
}
}
else {
puts("Paper 3");
eax = bomb();
}
}
else {
puts("Paper 2");
eax = bomb();
}
return eax;
}
We derive the following constraints:
- input[0xa] ^ input[0x8] + 0x30 <= 0x39
- input[0xd] ^ input[0x5] + 0x30 <= 0x39
- input[0xa] ^ input[0x8] + 0x30 > 0x2f
- input[0xd] ^ input[0x5] + 0x30 > 0x2f
- input[0x3] == input[0xa] ^ input[0x8] + 0x30
- input[0xf] == input[0xa] ^ input[0x8] + 0x30
- input[0x0] = input[0xd] ^ input[0x5] + 0x30
- input[0x12] = input[0xd] ^ input[0x5] + 0x30
This gets us closer to a keygen, but we’re not quite there yet. Let’s continue with scissors
to get more data. It disassembles to:
int scissors(char *input) {
var_10 = input[0x2] + input[0x1];
var_C = input[0x11] + input[0x10];
if ((var_10 > 0xaa) && (var_C > 0xaa)) {
if (var_10 == var_C) {
puts("Scissors 2");
eax = bomb();
}
}
else {
puts("Scissors 1");
eax = bomb();
}
return eax;
}
New constraints:
- input[0x2] + input[0x1] > 0xaa
- input[0x11] + input[0x10] > 0xaa
- input[0x2] + input[0x1] != input[0x11] + input[0x10]
Finally, the last procedure is cracker
:
int cracker(char *input) {
var_C = input[0xe] + input[0x4] + input[0x9];
if (var_C != 0x87) {
puts("cracker 1");
eax = bomb();
}
else {
eax = (HIDWORD(var_C * 0x55555556) - (SAR(var_C, 0x1f)))
+ (HIDWORD(var_C * 0x55555556) - (SAR(var_C, 0x1f)))
+ (HIDWORD(var_C * 0x55555556) - (SAR(var_C, 0x1f)));
if (var_C != eax) {
puts("cracker 1");
eax = bomb();
}
}
return eax;
}
Note that 0x87 / 3 is 45, which is the ascii code for ‘-’, one of the characters allowed. Let’s try to generate a serial at random given our constraints, and with the format “????-????-????-????”.
$ ./crackme05_32bit 2aP3-MdCK-HGgO-3bR2
In order to succeed you must fail, so that you know what not to do the next time.
-Anthony J. D'Angelo
Good Job!